1 /*
2  * Copyright (C) 2017 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #ifndef C_CODEC_H_
18 #define C_CODEC_H_
19 
20 #include <atomic>
21 #include <chrono>
22 #include <list>
23 #include <memory>
24 #include <set>
25 
26 #include <C2Component.h>
27 #include <codec2/hidl/client.h>
28 
29 #include <android/native_window.h>
30 #include <media/hardware/MetadataBufferType.h>
31 #include <media/stagefright/foundation/Mutexed.h>
32 #include <media/stagefright/CodecBase.h>
33 #include <media/stagefright/FrameRenderTracker.h>
34 #include <media/stagefright/MediaDefs.h>
35 #include <media/stagefright/SkipCutBuffer.h>
36 #include <utils/NativeHandle.h>
37 #include <hardware/gralloc.h>
38 #include <nativebase/nativebase.h>
39 
40 namespace android {
41 
42 class CCodecBufferChannel;
43 class InputSurfaceWrapper;
44 struct CCodecConfig;
45 struct MediaCodecInfo;
46 
47 class CCodec : public CodecBase {
48 public:
49     CCodec();
50 
51     virtual std::shared_ptr<BufferChannelBase> getBufferChannel() override;
52     virtual void initiateAllocateComponent(const sp<AMessage> &msg) override;
53     virtual void initiateConfigureComponent(const sp<AMessage> &msg) override;
54     virtual void initiateCreateInputSurface() override;
55     virtual void initiateSetInputSurface(const sp<PersistentSurface> &surface) override;
56     virtual void initiateStart() override;
57     virtual void initiateShutdown(bool keepComponentAllocated = false) override;
58 
59     virtual status_t setSurface(const sp<Surface> &surface, uint32_t generation) override;
60 
61     virtual void signalFlush() override;
62     virtual void signalResume() override;
63 
64     virtual void signalSetParameters(const sp<AMessage> &params) override;
65     virtual void signalEndOfInputStream() override;
66     virtual void signalRequestIDRFrame() override;
67 
68     virtual status_t querySupportedParameters(std::vector<std::string> *names) override;
69     virtual status_t describeParameter(
70             const std::string &name, CodecParameterDescriptor *desc) override;
71     virtual status_t subscribeToParameters(const std::vector<std::string> &names) override;
72     virtual status_t unsubscribeFromParameters(const std::vector<std::string> &names) override;
73 
74     void initiateReleaseIfStuck();
75     void onWorkDone(std::list<std::unique_ptr<C2Work>> &workItems);
76     void onInputBufferDone(uint64_t frameIndex, size_t arrayIndex);
77 
78     static PersistentSurface *CreateInputSurface();
79 
80     static status_t CanFetchLinearBlock(
81             const std::vector<std::string> &names, const C2MemoryUsage &usage, bool *isCompatible);
82 
83     static std::shared_ptr<C2LinearBlock> FetchLinearBlock(
84             size_t capacity, const C2MemoryUsage &usage, const std::vector<std::string> &names);
85 
86     static status_t CanFetchGraphicBlock(
87             const std::vector<std::string> &names, bool *isCompatible);
88 
89     static std::shared_ptr<C2GraphicBlock> FetchGraphicBlock(
90             int32_t width,
91             int32_t height,
92             int32_t format,
93             uint64_t usage,
94             const std::vector<std::string> &names);
95 
96 protected:
97     virtual ~CCodec();
98 
99     virtual void onMessageReceived(const sp<AMessage> &msg) override;
100 
101 private:
102     typedef std::chrono::steady_clock::time_point TimePoint;
103 
104     status_t tryAndReportOnError(std::function<status_t()> job);
105 
106     void initiateStop();
107     void initiateRelease(bool sendCallback = true);
108 
109     void allocate(const sp<MediaCodecInfo> &codecInfo);
110     void configure(const sp<AMessage> &msg);
111     void start();
112     void stop(bool pushBlankBuffer);
113     void flush();
114     void release(bool sendCallback, bool pushBlankBuffer);
115 
116     /**
117      * Creates an input surface for the current device configuration compatible with CCodec.
118      * This could be backed by the C2 HAL or the OMX HAL.
119      */
120     static sp<PersistentSurface> CreateCompatibleInputSurface();
121 
122     /// Creates an input surface to the OMX HAL
123     static sp<PersistentSurface> CreateOmxInputSurface();
124 
125     /// handle a create input surface call
126     void createInputSurface();
127     void setInputSurface(const sp<PersistentSurface> &surface);
128     status_t setupInputSurface(const std::shared_ptr<InputSurfaceWrapper> &surface);
129 
130     void setDeadline(
131             const TimePoint &now,
132             const std::chrono::milliseconds &timeout,
133             const char *name);
134 
135     status_t configureTunneledVideoPlayback(
136             const std::shared_ptr<Codec2Client::Component> comp,
137             sp<NativeHandle> *sidebandHandle,
138             const sp<AMessage> &msg);
139 
140     enum {
141         kWhatAllocate,
142         kWhatConfigure,
143         kWhatStart,
144         kWhatFlush,
145         kWhatStop,
146         kWhatRelease,
147         kWhatCreateInputSurface,
148         kWhatSetInputSurface,
149         kWhatSetParameters,
150 
151         kWhatWorkDone,
152         kWhatWatch,
153     };
154 
155     enum {
156         RELEASED,
157         ALLOCATED,
158         FLUSHED,
159         RUNNING,
160 
161         ALLOCATING,  // RELEASED -> ALLOCATED
162         STARTING,    // ALLOCATED -> RUNNING
163         STOPPING,    // RUNNING -> ALLOCATED
164         FLUSHING,    // RUNNING -> FLUSHED
165         RESUMING,    // FLUSHED -> RUNNING
166         RELEASING,   // {ANY EXCEPT RELEASED} -> RELEASED
167     };
168 
169     struct State {
StateState170         inline State() : mState(RELEASED) {}
getState171         inline int get() const { return mState; }
setState172         inline void set(int newState) { mState = newState; }
173 
174         std::shared_ptr<Codec2Client::Component> comp;
175     private:
176         int mState;
177     };
178 
179     struct NamedTimePoint {
NamedTimePointNamedTimePoint180         NamedTimePoint() : mTimePoint(TimePoint::max()), mName("") {}
181 
setNamedTimePoint182         inline void set(
183                 const TimePoint &timePoint,
184                 const char *name) {
185             mTimePoint = timePoint;
186             mName = name;
187         }
188 
getNamedTimePoint189         inline TimePoint get() const { return mTimePoint; }
getNameNamedTimePoint190         inline const char *getName() const { return mName; }
191     private:
192         TimePoint mTimePoint;
193         const char *mName;
194     };
195 
196     Mutexed<State> mState;
197     std::shared_ptr<CCodecBufferChannel> mChannel;
198 
199     std::shared_ptr<Codec2Client> mClient;
200     std::shared_ptr<Codec2Client::Listener> mClientListener;
201     struct ClientListener;
202 
203     Mutexed<NamedTimePoint> mDeadline;
204 
205     Mutexed<std::unique_ptr<CCodecConfig>> mConfig;
206     Mutexed<std::list<std::unique_ptr<C2Work>>> mWorkDoneQueue;
207 
208     sp<AMessage> mMetrics;
209 
210     friend class CCodecCallbackImpl;
211 
212     DISALLOW_EVIL_CONSTRUCTORS(CCodec);
213 };
214 
215 }  // namespace android
216 
217 #endif  // C_CODEC_H_
218