blob: 693c7cbfab70fa893dbcef5eb0ce4dff6aa8b9b2 [file] [log] [blame]
Dan Stozad3182402014-11-17 12:03:59 -08001/*
Mathias Agopian076b1cc2009-04-10 14:24:30 -07002**
3** Copyright 2009, The Android Open Source Project
4**
Dan Stozad3182402014-11-17 12:03:59 -08005** Licensed under the Apache License, Version 2.0 (the "License");
6** you may not use this file except in compliance with the License.
7** You may obtain a copy of the License at
Mathias Agopian076b1cc2009-04-10 14:24:30 -07008**
Dan Stozad3182402014-11-17 12:03:59 -08009** http://www.apache.org/licenses/LICENSE-2.0
Mathias Agopian076b1cc2009-04-10 14:24:30 -070010**
Dan Stozad3182402014-11-17 12:03:59 -080011** Unless required by applicable law or agreed to in writing, software
12** distributed under the License is distributed on an "AS IS" BASIS,
13** WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
14** See the License for the specific language governing permissions and
Mathias Agopian076b1cc2009-04-10 14:24:30 -070015** limitations under the License.
16*/
17
Mathias Agopian5629eb12010-04-15 14:57:39 -070018#define LOG_TAG "GraphicBufferAllocator"
Mathias Agopiancf563192012-02-29 20:43:29 -080019#define ATRACE_TAG ATRACE_TAG_GRAPHICS
Mathias Agopian5629eb12010-04-15 14:57:39 -070020
Mathias Agopian076b1cc2009-04-10 14:24:30 -070021#include <cutils/log.h>
Mathias Agopian4243e662009-04-15 18:34:24 -070022
23#include <utils/Singleton.h>
Mathias Agopian076b1cc2009-04-10 14:24:30 -070024#include <utils/String8.h>
Mathias Agopiancf563192012-02-29 20:43:29 -080025#include <utils/Trace.h>
Mathias Agopian076b1cc2009-04-10 14:24:30 -070026
Mathias Agopian3330b202009-10-05 17:07:12 -070027#include <ui/GraphicBufferAllocator.h>
Dan Stoza8deb4da2016-06-01 18:21:44 -070028#include <ui/Gralloc1On0Adapter.h>
Chia-I Wu9ba189d2016-09-22 17:13:08 +080029#include <ui/GrallocAllocator.h>
30#include <ui/GrallocMapper.h>
31#include <ui/GraphicBufferMapper.h>
Mathias Agopian076b1cc2009-04-10 14:24:30 -070032
Mathias Agopian076b1cc2009-04-10 14:24:30 -070033namespace android {
34// ---------------------------------------------------------------------------
35
Mathias Agopian3330b202009-10-05 17:07:12 -070036ANDROID_SINGLETON_STATIC_INSTANCE( GraphicBufferAllocator )
Mathias Agopian4243e662009-04-15 18:34:24 -070037
Mathias Agopian3330b202009-10-05 17:07:12 -070038Mutex GraphicBufferAllocator::sLock;
Mathias Agopianb26af232009-10-05 18:19:57 -070039KeyedVector<buffer_handle_t,
40 GraphicBufferAllocator::alloc_rec_t> GraphicBufferAllocator::sAllocList;
Mathias Agopian076b1cc2009-04-10 14:24:30 -070041
Mathias Agopian3330b202009-10-05 17:07:12 -070042GraphicBufferAllocator::GraphicBufferAllocator()
Chia-I Wu9ba189d2016-09-22 17:13:08 +080043 : mAllocator(std::make_unique<Gralloc2::Allocator>()),
44 mMapper(GraphicBufferMapper::getInstance())
45{
46 if (!mAllocator->valid()) {
47 mLoader = std::make_unique<Gralloc1::Loader>();
48 mDevice = mLoader->getDevice();
49 }
50}
Mathias Agopian076b1cc2009-04-10 14:24:30 -070051
Dan Stoza8deb4da2016-06-01 18:21:44 -070052GraphicBufferAllocator::~GraphicBufferAllocator() {}
Mathias Agopian076b1cc2009-04-10 14:24:30 -070053
Mathias Agopian3330b202009-10-05 17:07:12 -070054void GraphicBufferAllocator::dump(String8& result) const
Mathias Agopian076b1cc2009-04-10 14:24:30 -070055{
56 Mutex::Autolock _l(sLock);
57 KeyedVector<buffer_handle_t, alloc_rec_t>& list(sAllocList);
58 size_t total = 0;
Erik Gilling1d21a9c2010-12-01 16:38:01 -080059 const size_t SIZE = 4096;
Mathias Agopian076b1cc2009-04-10 14:24:30 -070060 char buffer[SIZE];
61 snprintf(buffer, SIZE, "Allocated buffers:\n");
62 result.append(buffer);
63 const size_t c = list.size();
64 for (size_t i=0 ; i<c ; i++) {
65 const alloc_rec_t& rec(list.valueAt(i));
Mathias Agopiana947de82011-07-29 16:35:41 -070066 if (rec.size) {
Dan Stoza024e9312016-08-24 12:17:29 -070067 snprintf(buffer, SIZE, "%10p: %7.2f KiB | %4u (%4u) x %4u | %8X | 0x%08x | %s\n",
Mathias Agopiana947de82011-07-29 16:35:41 -070068 list.keyAt(i), rec.size/1024.0f,
Dan Stoza024e9312016-08-24 12:17:29 -070069 rec.width, rec.stride, rec.height, rec.format, rec.usage,
70 rec.requestorName.c_str());
Mathias Agopiana947de82011-07-29 16:35:41 -070071 } else {
Dan Stoza024e9312016-08-24 12:17:29 -070072 snprintf(buffer, SIZE, "%10p: unknown | %4u (%4u) x %4u | %8X | 0x%08x | %s\n",
Mathias Agopiana947de82011-07-29 16:35:41 -070073 list.keyAt(i),
Dan Stoza024e9312016-08-24 12:17:29 -070074 rec.width, rec.stride, rec.height, rec.format, rec.usage,
75 rec.requestorName.c_str());
Mathias Agopiana947de82011-07-29 16:35:41 -070076 }
Mathias Agopian076b1cc2009-04-10 14:24:30 -070077 result.append(buffer);
78 total += rec.size;
79 }
Mathias Agopiana947de82011-07-29 16:35:41 -070080 snprintf(buffer, SIZE, "Total allocated (estimate): %.2f KB\n", total/1024.0f);
Mathias Agopian076b1cc2009-04-10 14:24:30 -070081 result.append(buffer);
Chia-I Wu9ba189d2016-09-22 17:13:08 +080082
83 std::string deviceDump;
84 if (mAllocator->valid()) {
85 deviceDump = mAllocator->dumpDebugInfo();
86 } else {
87 deviceDump = mDevice->dump();
88 }
89
Dan Stoza8deb4da2016-06-01 18:21:44 -070090 result.append(deviceDump.c_str(), deviceDump.size());
Mathias Agopian076b1cc2009-04-10 14:24:30 -070091}
92
Mathias Agopian678bdd62010-12-03 17:33:09 -080093void GraphicBufferAllocator::dumpToSystemLog()
94{
95 String8 s;
96 GraphicBufferAllocator::getInstance().dump(s);
Steve Block9d453682011-12-20 16:23:08 +000097 ALOGD("%s", s.string());
Mathias Agopian678bdd62010-12-03 17:33:09 -080098}
99
Chia-I Wu9ba189d2016-09-22 17:13:08 +0800100namespace {
101
102class HalBuffer {
103public:
104 HalBuffer(const Gralloc2::Allocator* allocator,
105 uint32_t width, uint32_t height,
106 PixelFormat format, uint32_t usage)
107 : mAllocator(allocator), mBufferValid(false)
108 {
109 Gralloc2::IAllocator::BufferDescriptorInfo info = {};
110 info.width = width;
111 info.height = height;
112 info.format = static_cast<Gralloc2::PixelFormat>(format);
113 info.producerUsageMask = usage;
114 info.consumerUsageMask = usage;
115
116 Gralloc2::BufferDescriptor descriptor;
117 auto error = mAllocator->createBufferDescriptor(info, descriptor);
118 if (error != Gralloc2::Error::NONE) {
119 ALOGE("Failed to create desc (%u x %u) format %d usage %u: %d",
120 width, height, format, usage, error);
121 return;
122 }
123
124 error = mAllocator->allocate(descriptor, mBuffer);
125 if (error == Gralloc2::Error::NOT_SHARED) {
126 error = Gralloc2::Error::NONE;
127 }
128
129 if (error != Gralloc2::Error::NONE) {
130 ALOGE("Failed to allocate (%u x %u) format %d usage %u: %d",
131 width, height, format, usage, error);
132 mAllocator->destroyBufferDescriptor(descriptor);
133 return;
134 }
135
136 error = mAllocator->exportHandle(descriptor, mBuffer, mHandle);
137 if (error != Gralloc2::Error::NONE) {
138 ALOGE("Failed to export handle");
139 mAllocator->free(mBuffer);
140 mAllocator->destroyBufferDescriptor(descriptor);
141 return;
142 }
143
144 mAllocator->destroyBufferDescriptor(descriptor);
145
146 mBufferValid = true;
147 }
148
149 ~HalBuffer()
150 {
151 if (mBufferValid) {
152 if (mHandle) {
153 native_handle_close(mHandle);
154 native_handle_delete(mHandle);
155 }
156
157 mAllocator->free(mBuffer);
158 }
159 }
160
161 bool exportHandle(GraphicBufferMapper& mapper,
162 buffer_handle_t* handle, uint32_t* stride)
163 {
164 if (!mBufferValid) {
165 return false;
166 }
167
168 if (mapper.registerBuffer(mHandle)) {
169 return false;
170 }
171
172 *handle = mHandle;
173
174 auto error = mapper.getGrallocMapper().getStride(mHandle, *stride);
175 if (error != Gralloc2::Error::NONE) {
176 ALOGW("Failed to get stride from buffer: %d", error);
177 *stride = 0;
178 }
179
180 mHandle = nullptr;
181 mAllocator->free(mBuffer);
182 mBufferValid = false;
183
184 return true;
185 }
186
187private:
188 const Gralloc2::Allocator* mAllocator;
189
190 bool mBufferValid;
191 Gralloc2::Buffer mBuffer;
192 native_handle_t* mHandle;
193};
194
195} // namespace
196
Dan Stoza8deb4da2016-06-01 18:21:44 -0700197status_t GraphicBufferAllocator::allocate(uint32_t width, uint32_t height,
Dan Stozad3182402014-11-17 12:03:59 -0800198 PixelFormat format, uint32_t usage, buffer_handle_t* handle,
Dan Stoza024e9312016-08-24 12:17:29 -0700199 uint32_t* stride, uint64_t graphicBufferId, std::string requestorName)
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700200{
Mathias Agopiancf563192012-02-29 20:43:29 -0800201 ATRACE_CALL();
Dan Stozad3182402014-11-17 12:03:59 -0800202
Mathias Agopian5629eb12010-04-15 14:57:39 -0700203 // make sure to not allocate a N x 0 or 0 x N buffer, since this is
204 // allowed from an API stand-point allocate a 1x1 buffer instead.
Dan Stozad3182402014-11-17 12:03:59 -0800205 if (!width || !height)
206 width = height = 1;
Mathias Agopiancbb288b2009-09-07 16:32:45 -0700207
Dan Stoza24fa67f2015-05-29 12:48:04 -0700208 // Filter out any usage bits that should not be passed to the gralloc module
209 usage &= GRALLOC_USAGE_ALLOC_MASK;
210
Chia-I Wu9ba189d2016-09-22 17:13:08 +0800211 gralloc1_error_t error;
212 if (mAllocator->valid()) {
213 HalBuffer buffer(mAllocator.get(), width, height, format, usage);
214 if (!buffer.exportHandle(mMapper, handle, stride)) {
215 return NO_MEMORY;
216 }
217 error = GRALLOC1_ERROR_NONE;
218 } else {
219 auto descriptor = mDevice->createDescriptor();
220 error = descriptor->setDimensions(width, height);
221 if (error != GRALLOC1_ERROR_NONE) {
222 ALOGE("Failed to set dimensions to (%u, %u): %d",
223 width, height, error);
224 return BAD_VALUE;
225 }
226 error = descriptor->setFormat(
227 static_cast<android_pixel_format_t>(format));
228 if (error != GRALLOC1_ERROR_NONE) {
229 ALOGE("Failed to set format to %d: %d", format, error);
230 return BAD_VALUE;
231 }
232 error = descriptor->setProducerUsage(
233 static_cast<gralloc1_producer_usage_t>(usage));
234 if (error != GRALLOC1_ERROR_NONE) {
235 ALOGE("Failed to set producer usage to %u: %d", usage, error);
236 return BAD_VALUE;
237 }
238 error = descriptor->setConsumerUsage(
239 static_cast<gralloc1_consumer_usage_t>(usage));
240 if (error != GRALLOC1_ERROR_NONE) {
241 ALOGE("Failed to set consumer usage to %u: %d", usage, error);
242 return BAD_VALUE;
243 }
Mathias Agopiancbb288b2009-09-07 16:32:45 -0700244
Chia-I Wu9ba189d2016-09-22 17:13:08 +0800245 error = mDevice->allocate(descriptor, graphicBufferId, handle);
246 if (error != GRALLOC1_ERROR_NONE) {
247 ALOGE("Failed to allocate (%u x %u) format %d usage %u: %d",
248 width, height, format, usage, error);
249 return NO_MEMORY;
250 }
Dan Stozad3182402014-11-17 12:03:59 -0800251
Chia-I Wu9ba189d2016-09-22 17:13:08 +0800252 error = mDevice->getStride(*handle, stride);
253 if (error != GRALLOC1_ERROR_NONE) {
254 ALOGW("Failed to get stride from buffer: %d", error);
255 }
Dan Stoza8deb4da2016-06-01 18:21:44 -0700256 }
257
258 if (error == NO_ERROR) {
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700259 Mutex::Autolock _l(sLock);
260 KeyedVector<buffer_handle_t, alloc_rec_t>& list(sAllocList);
Dan Stozad3182402014-11-17 12:03:59 -0800261 uint32_t bpp = bytesPerPixel(format);
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700262 alloc_rec_t rec;
Dan Stozad3182402014-11-17 12:03:59 -0800263 rec.width = width;
264 rec.height = height;
265 rec.stride = *stride;
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700266 rec.format = format;
267 rec.usage = usage;
Dan Stozad3182402014-11-17 12:03:59 -0800268 rec.size = static_cast<size_t>(height * (*stride) * bpp);
Dan Stoza024e9312016-08-24 12:17:29 -0700269 rec.requestorName = std::move(requestorName);
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700270 list.add(*handle, rec);
271 }
272
Dan Stoza8deb4da2016-06-01 18:21:44 -0700273 return NO_ERROR;
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700274}
275
Mathias Agopian3330b202009-10-05 17:07:12 -0700276status_t GraphicBufferAllocator::free(buffer_handle_t handle)
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700277{
Mathias Agopiancf563192012-02-29 20:43:29 -0800278 ATRACE_CALL();
Mathias Agopian0a757812010-12-08 16:40:01 -0800279
Chia-I Wu9ba189d2016-09-22 17:13:08 +0800280 gralloc1_error_t error;
281 if (mAllocator->valid()) {
282 error = static_cast<gralloc1_error_t>(
283 mMapper.unregisterBuffer(handle));
284 } else {
285 error = mDevice->release(handle);
286 }
287
Dan Stoza8deb4da2016-06-01 18:21:44 -0700288 if (error != GRALLOC1_ERROR_NONE) {
289 ALOGE("Failed to free buffer: %d", error);
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700290 }
291
Dan Stoza8deb4da2016-06-01 18:21:44 -0700292 Mutex::Autolock _l(sLock);
293 KeyedVector<buffer_handle_t, alloc_rec_t>& list(sAllocList);
294 list.removeItem(handle);
295
296 return NO_ERROR;
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700297}
298
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700299// ---------------------------------------------------------------------------
300}; // namespace android