1 /*
2  * Copyright (C) 2009 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 MEDIA_BUFFER_H_
18 
19 #define MEDIA_BUFFER_H_
20 
21 #include <atomic>
22 #include <list>
23 
24 #include <pthread.h>
25 
26 #include <binder/MemoryDealer.h>
27 #include <utils/Errors.h>
28 #include <utils/RefBase.h>
29 #include <media/stagefright/MediaBufferBase.h>
30 
31 namespace android {
32 
33 struct ABuffer;
34 class MediaBuffer;
35 class MediaBufferObserver;
36 class MetaDataBase;
37 
38 class MediaBuffer : public MediaBufferBase {
39 public:
40     // allocations larger than or equal to this will use shared memory.
41     static const size_t kSharedMemThreshold = 64 * 1024;
42 
43     // The underlying data remains the responsibility of the caller!
44     MediaBuffer(void *data, size_t size);
45 
46     explicit MediaBuffer(size_t size);
47 
48     explicit MediaBuffer(const sp<ABuffer> &buffer);
49 #ifndef NO_IMEMORY
MediaBuffer(const sp<IMemory> & mem)50     MediaBuffer(const sp<IMemory> &mem) :
51         MediaBuffer((uint8_t *)mem->pointer() + sizeof(SharedControl), mem->size()) {
52         // delegate and override mMemory
53         mMemory = mem;
54     }
55 #endif
56 
57     // If MediaBufferGroup is set, decrement the local reference count;
58     // if the local reference count drops to 0, return the buffer to the
59     // associated MediaBufferGroup.
60     //
61     // If no MediaBufferGroup is set, the local reference count must be zero
62     // when called, whereupon the MediaBuffer is deleted.
63     virtual void release();
64 
65     // Increments the local reference count.
66     // Use only when MediaBufferGroup is set.
67     virtual void add_ref();
68 
69     virtual void *data() const;
70     virtual size_t size() const;
71 
72     virtual size_t range_offset() const;
73     virtual size_t range_length() const;
74 
75     virtual void set_range(size_t offset, size_t length);
76 
77     MetaDataBase& meta_data();
78 
79     // Clears meta data and resets the range to the full extent.
80     virtual void reset();
81 
82     virtual void setObserver(MediaBufferObserver *group);
83 
84     // sum of localRefcount() and remoteRefcount()
85     // Result should be treated as approximate unless the result precludes concurrent accesses.
refcount()86     virtual int refcount() const {
87         return localRefcount() + remoteRefcount();
88     }
89 
90     // Result should be treated as approximate unless the result precludes concurrent accesses.
localRefcount()91     virtual int localRefcount() const {
92         return mRefCount.load(std::memory_order_relaxed);
93     }
94 
remoteRefcount()95     virtual int remoteRefcount() const {
96 #ifndef NO_IMEMORY
97         if (mMemory.get() == nullptr || mMemory->pointer() == nullptr) return 0;
98         int32_t remoteRefcount =
99                 reinterpret_cast<SharedControl *>(mMemory->pointer())->getRemoteRefcount();
100         // Sanity check so that remoteRefCount() is non-negative.
101         return remoteRefcount >= 0 ? remoteRefcount : 0; // do not allow corrupted data.
102 #else
103         return 0;
104 #endif
105     }
106 
107     // returns old value
addRemoteRefcount(int32_t value)108     int addRemoteRefcount(int32_t value) {
109 #ifndef NO_IMEMORY
110         if (mMemory.get() == nullptr || mMemory->pointer() == nullptr) return 0;
111         return reinterpret_cast<SharedControl *>(mMemory->pointer())->addRemoteRefcount(value);
112 #else
113         (void) value;
114         return 0;
115 #endif
116     }
117 
isDeadObject()118     bool isDeadObject() const {
119         return isDeadObject(mMemory);
120     }
121 
isDeadObject(const sp<IMemory> & memory)122     static bool isDeadObject(const sp<IMemory> &memory) {
123 #ifndef NO_IMEMORY
124         if (memory.get() == nullptr || memory->pointer() == nullptr) return false;
125         return reinterpret_cast<SharedControl *>(memory->pointer())->isDeadObject();
126 #else
127         (void) memory;
128         return false;
129 #endif
130     }
131 
132     // Sticky on enabling of shared memory MediaBuffers. By default we don't use
133     // shared memory for MediaBuffers, but we enable this for those processes
134     // that export MediaBuffers.
useSharedMemory()135     static void useSharedMemory() {
136         std::atomic_store_explicit(
137                 &mUseSharedMemory, (int_least32_t)1, std::memory_order_seq_cst);
138     }
139 
140 protected:
141     // true if MediaBuffer is observed (part of a MediaBufferGroup).
isObserved()142     inline bool isObserved() const {
143         return mObserver != nullptr;
144     }
145 
146     virtual ~MediaBuffer();
147 
148     sp<IMemory> mMemory;
149 
150 private:
151     friend class MediaBufferGroup;
152     friend class OMXDecoder;
153     friend class BnMediaSource;
154     friend class BpMediaSource;
155 
156     // For use by OMXDecoder, reference count must be 1, drop reference
157     // count to 0 without signalling the observer.
158     void claim();
159 
160     MediaBufferObserver *mObserver;
161     std::atomic<int> mRefCount;
162 
163     void *mData;
164     size_t mSize, mRangeOffset, mRangeLength;
165     sp<ABuffer> mBuffer;
166 
167     bool mOwnsData;
168 
169     MetaDataBase* mMetaData;
170 
171     static std::atomic_int_least32_t mUseSharedMemory;
172 
173     MediaBuffer(const MediaBuffer &);
174     MediaBuffer &operator=(const MediaBuffer &);
175 
176     // SharedControl block at the start of IMemory.
177     struct SharedControl {
178         enum {
179             FLAG_DEAD_OBJECT = (1 << 0),
180         };
181 
182         // returns old value
addRemoteRefcountSharedControl183         inline int32_t addRemoteRefcount(int32_t value) {
184             return std::atomic_fetch_add_explicit(
185                     &mRemoteRefcount, (int_least32_t)value, std::memory_order_seq_cst);
186         }
187 
getRemoteRefcountSharedControl188         inline int32_t getRemoteRefcount() const {
189             return std::atomic_load_explicit(&mRemoteRefcount, std::memory_order_seq_cst);
190         }
191 
setRemoteRefcountSharedControl192         inline void setRemoteRefcount(int32_t value) {
193             std::atomic_store_explicit(
194                     &mRemoteRefcount, (int_least32_t)value, std::memory_order_seq_cst);
195         }
196 
isDeadObjectSharedControl197         inline bool isDeadObject() const {
198             return (std::atomic_load_explicit(
199                     &mFlags, std::memory_order_seq_cst) & FLAG_DEAD_OBJECT) != 0;
200         }
201 
setDeadObjectSharedControl202         inline void setDeadObject() {
203             (void)std::atomic_fetch_or_explicit(
204                     &mFlags, (int_least32_t)FLAG_DEAD_OBJECT, std::memory_order_seq_cst);
205         }
206 
clearSharedControl207         inline void clear() {
208             std::atomic_store_explicit(
209                     &mFlags, (int_least32_t)0, std::memory_order_seq_cst);
210             std::atomic_store_explicit(
211                     &mRemoteRefcount, (int_least32_t)0, std::memory_order_seq_cst);
212         }
213 
214     private:
215         // Caution: atomic_int_fast32_t is 64 bits on LP64.
216         std::atomic_int_least32_t mFlags;
217         std::atomic_int_least32_t mRemoteRefcount;
218         int32_t unused[6] __attribute__((__unused__)); // additional buffer space
219     };
220 
getSharedControl()221     inline SharedControl *getSharedControl() const {
222 #ifndef NO_IMEMORY
223          return reinterpret_cast<SharedControl *>(mMemory->pointer());
224 #else
225          return nullptr;
226 #endif
227      }
228 };
229 
230 }  // namespace android
231 
232 #endif  // MEDIA_BUFFER_H_
233