blob: b14110e5992a427df978ed3c853f164d139259d1 [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
Mark Salyzyn7823e122016-09-29 08:08:05 -070021#include <log/log.h>
Mathias Agopian4243e662009-04-15 18:34:24 -070022#include <utils/Singleton.h>
Mathias Agopian076b1cc2009-04-10 14:24:30 -070023#include <utils/String8.h>
Mathias Agopiancf563192012-02-29 20:43:29 -080024#include <utils/Trace.h>
Mathias Agopian076b1cc2009-04-10 14:24:30 -070025
Mathias Agopian3330b202009-10-05 17:07:12 -070026#include <ui/GraphicBufferAllocator.h>
Dan Stoza8deb4da2016-06-01 18:21:44 -070027#include <ui/Gralloc1On0Adapter.h>
Chia-I Wu9ba189d2016-09-22 17:13:08 +080028#include <ui/GrallocAllocator.h>
29#include <ui/GrallocMapper.h>
30#include <ui/GraphicBufferMapper.h>
Mathias Agopian076b1cc2009-04-10 14:24:30 -070031
Mathias Agopian076b1cc2009-04-10 14:24:30 -070032namespace android {
33// ---------------------------------------------------------------------------
34
Mathias Agopian3330b202009-10-05 17:07:12 -070035ANDROID_SINGLETON_STATIC_INSTANCE( GraphicBufferAllocator )
Mathias Agopian4243e662009-04-15 18:34:24 -070036
Mathias Agopian3330b202009-10-05 17:07:12 -070037Mutex GraphicBufferAllocator::sLock;
Mathias Agopianb26af232009-10-05 18:19:57 -070038KeyedVector<buffer_handle_t,
39 GraphicBufferAllocator::alloc_rec_t> GraphicBufferAllocator::sAllocList;
Mathias Agopian076b1cc2009-04-10 14:24:30 -070040
Mathias Agopian3330b202009-10-05 17:07:12 -070041GraphicBufferAllocator::GraphicBufferAllocator()
Chia-I Wu9ba189d2016-09-22 17:13:08 +080042 : mAllocator(std::make_unique<Gralloc2::Allocator>()),
43 mMapper(GraphicBufferMapper::getInstance())
44{
45 if (!mAllocator->valid()) {
46 mLoader = std::make_unique<Gralloc1::Loader>();
47 mDevice = mLoader->getDevice();
48 }
49}
Mathias Agopian076b1cc2009-04-10 14:24:30 -070050
Dan Stoza8deb4da2016-06-01 18:21:44 -070051GraphicBufferAllocator::~GraphicBufferAllocator() {}
Mathias Agopian076b1cc2009-04-10 14:24:30 -070052
Mathias Agopian3330b202009-10-05 17:07:12 -070053void GraphicBufferAllocator::dump(String8& result) const
Mathias Agopian076b1cc2009-04-10 14:24:30 -070054{
55 Mutex::Autolock _l(sLock);
56 KeyedVector<buffer_handle_t, alloc_rec_t>& list(sAllocList);
57 size_t total = 0;
Erik Gilling1d21a9c2010-12-01 16:38:01 -080058 const size_t SIZE = 4096;
Mathias Agopian076b1cc2009-04-10 14:24:30 -070059 char buffer[SIZE];
60 snprintf(buffer, SIZE, "Allocated buffers:\n");
61 result.append(buffer);
62 const size_t c = list.size();
63 for (size_t i=0 ; i<c ; i++) {
64 const alloc_rec_t& rec(list.valueAt(i));
Mathias Agopiana947de82011-07-29 16:35:41 -070065 if (rec.size) {
Craig Donnere96a3252017-02-02 12:13:34 -080066 snprintf(buffer, SIZE, "%10p: %7.2f KiB | %4u (%4u) x %4u | %4u | %8X | 0x%" PRIx64
67 ", 0x%" PRIx64 " | %s\n",
George Burgess IV1093cf32017-01-25 17:09:18 -080068 list.keyAt(i), rec.size/1024.0,
Craig Donner6ebc46a2016-10-21 15:23:44 -070069 rec.width, rec.stride, rec.height, rec.layerCount, rec.format,
Craig Donnere96a3252017-02-02 12:13:34 -080070 rec.producerUsage, rec.consumerUsage,
71 rec.requestorName.c_str());
Mathias Agopiana947de82011-07-29 16:35:41 -070072 } else {
Craig Donnere96a3252017-02-02 12:13:34 -080073 snprintf(buffer, SIZE, "%10p: unknown | %4u (%4u) x %4u | %4u | %8X | 0x%" PRIx64
74 ", 0x%" PRIx64 " | %s\n",
Mathias Agopiana947de82011-07-29 16:35:41 -070075 list.keyAt(i),
Craig Donner6ebc46a2016-10-21 15:23:44 -070076 rec.width, rec.stride, rec.height, rec.layerCount, rec.format,
Craig Donnere96a3252017-02-02 12:13:34 -080077 rec.producerUsage, rec.consumerUsage,
78 rec.requestorName.c_str());
Mathias Agopiana947de82011-07-29 16:35:41 -070079 }
Mathias Agopian076b1cc2009-04-10 14:24:30 -070080 result.append(buffer);
81 total += rec.size;
82 }
George Burgess IV1093cf32017-01-25 17:09:18 -080083 snprintf(buffer, SIZE, "Total allocated (estimate): %.2f KB\n", total/1024.0);
Mathias Agopian076b1cc2009-04-10 14:24:30 -070084 result.append(buffer);
Chia-I Wu9ba189d2016-09-22 17:13:08 +080085
86 std::string deviceDump;
87 if (mAllocator->valid()) {
88 deviceDump = mAllocator->dumpDebugInfo();
89 } else {
90 deviceDump = mDevice->dump();
91 }
92
Dan Stoza8deb4da2016-06-01 18:21:44 -070093 result.append(deviceDump.c_str(), deviceDump.size());
Mathias Agopian076b1cc2009-04-10 14:24:30 -070094}
95
Mathias Agopian678bdd62010-12-03 17:33:09 -080096void GraphicBufferAllocator::dumpToSystemLog()
97{
98 String8 s;
99 GraphicBufferAllocator::getInstance().dump(s);
Steve Block9d453682011-12-20 16:23:08 +0000100 ALOGD("%s", s.string());
Mathias Agopian678bdd62010-12-03 17:33:09 -0800101}
102
Chia-I Wu9ba189d2016-09-22 17:13:08 +0800103namespace {
104
105class HalBuffer {
106public:
107 HalBuffer(const Gralloc2::Allocator* allocator,
108 uint32_t width, uint32_t height,
Craig Donnere96a3252017-02-02 12:13:34 -0800109 PixelFormat format, uint32_t layerCount, uint64_t producerUsage,
110 uint64_t consumerUsage)
Chia-I Wu9ba189d2016-09-22 17:13:08 +0800111 : mAllocator(allocator), mBufferValid(false)
112 {
Chia-I Wub018bf02016-11-22 13:29:49 +0800113 Gralloc2::IAllocatorClient::BufferDescriptorInfo info = {};
Chia-I Wu9ba189d2016-09-22 17:13:08 +0800114 info.width = width;
115 info.height = height;
116 info.format = static_cast<Gralloc2::PixelFormat>(format);
Craig Donner6ebc46a2016-10-21 15:23:44 -0700117 info.layerCount = layerCount;
Craig Donnere96a3252017-02-02 12:13:34 -0800118 info.producerUsageMask = producerUsage;
119 info.consumerUsageMask = consumerUsage;
Chia-I Wu9ba189d2016-09-22 17:13:08 +0800120
121 Gralloc2::BufferDescriptor descriptor;
Chia-I Wu67e376d2016-12-19 11:36:22 +0800122 auto error = mAllocator->createBufferDescriptor(info, &descriptor);
Chia-I Wu9ba189d2016-09-22 17:13:08 +0800123 if (error != Gralloc2::Error::NONE) {
Craig Donnere96a3252017-02-02 12:13:34 -0800124 ALOGE("Failed to create desc (%u x %u) layerCount %u format %d producerUsage %" PRIx64
125 " consumerUsage %" PRIx64 ": %d",
126 width, height, layerCount, format, producerUsage,
127 consumerUsage, error);
Chia-I Wu9ba189d2016-09-22 17:13:08 +0800128 return;
129 }
130
Chia-I Wu67e376d2016-12-19 11:36:22 +0800131 error = mAllocator->allocate(descriptor, &mBuffer);
Chia-I Wu9ba189d2016-09-22 17:13:08 +0800132 if (error == Gralloc2::Error::NOT_SHARED) {
133 error = Gralloc2::Error::NONE;
134 }
135
136 if (error != Gralloc2::Error::NONE) {
Craig Donnere96a3252017-02-02 12:13:34 -0800137 ALOGE("Failed to allocate (%u x %u) layerCount %u format %d producerUsage %" PRIx64
138 " consumerUsage %" PRIx64 ": %d",
139 width, height, layerCount, format, producerUsage,
140 consumerUsage, error);
Chia-I Wu9ba189d2016-09-22 17:13:08 +0800141 mAllocator->destroyBufferDescriptor(descriptor);
142 return;
143 }
144
Chia-I Wu67e376d2016-12-19 11:36:22 +0800145 error = mAllocator->exportHandle(descriptor, mBuffer, &mHandle);
Chia-I Wu9ba189d2016-09-22 17:13:08 +0800146 if (error != Gralloc2::Error::NONE) {
147 ALOGE("Failed to export handle");
148 mAllocator->free(mBuffer);
149 mAllocator->destroyBufferDescriptor(descriptor);
150 return;
151 }
152
153 mAllocator->destroyBufferDescriptor(descriptor);
154
155 mBufferValid = true;
156 }
157
158 ~HalBuffer()
159 {
160 if (mBufferValid) {
161 if (mHandle) {
162 native_handle_close(mHandle);
163 native_handle_delete(mHandle);
164 }
165
166 mAllocator->free(mBuffer);
167 }
168 }
169
170 bool exportHandle(GraphicBufferMapper& mapper,
171 buffer_handle_t* handle, uint32_t* stride)
172 {
173 if (!mBufferValid) {
174 return false;
175 }
176
177 if (mapper.registerBuffer(mHandle)) {
178 return false;
179 }
180
181 *handle = mHandle;
182
Chia-I Wu67e376d2016-12-19 11:36:22 +0800183 auto error = mapper.getGrallocMapper().getStride(mHandle, stride);
Chia-I Wu9ba189d2016-09-22 17:13:08 +0800184 if (error != Gralloc2::Error::NONE) {
185 ALOGW("Failed to get stride from buffer: %d", error);
186 *stride = 0;
187 }
188
189 mHandle = nullptr;
190 mAllocator->free(mBuffer);
191 mBufferValid = false;
192
193 return true;
194 }
195
196private:
197 const Gralloc2::Allocator* mAllocator;
198
199 bool mBufferValid;
200 Gralloc2::Buffer mBuffer;
201 native_handle_t* mHandle;
202};
203
204} // namespace
205
Dan Stoza8deb4da2016-06-01 18:21:44 -0700206status_t GraphicBufferAllocator::allocate(uint32_t width, uint32_t height,
Craig Donnere96a3252017-02-02 12:13:34 -0800207 PixelFormat format, uint32_t layerCount, uint64_t producerUsage,
208 uint64_t consumerUsage, buffer_handle_t* handle, uint32_t* stride,
209 uint64_t graphicBufferId, std::string requestorName)
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700210{
Mathias Agopiancf563192012-02-29 20:43:29 -0800211 ATRACE_CALL();
Dan Stozad3182402014-11-17 12:03:59 -0800212
Mathias Agopian5629eb12010-04-15 14:57:39 -0700213 // make sure to not allocate a N x 0 or 0 x N buffer, since this is
214 // allowed from an API stand-point allocate a 1x1 buffer instead.
Dan Stozad3182402014-11-17 12:03:59 -0800215 if (!width || !height)
216 width = height = 1;
Mathias Agopiancbb288b2009-09-07 16:32:45 -0700217
Craig Donner6ebc46a2016-10-21 15:23:44 -0700218 // Ensure that layerCount is valid.
219 if (layerCount < 1)
220 layerCount = 1;
221
Chia-I Wu9ba189d2016-09-22 17:13:08 +0800222 gralloc1_error_t error;
223 if (mAllocator->valid()) {
Craig Donner6ebc46a2016-10-21 15:23:44 -0700224 HalBuffer buffer(mAllocator.get(), width, height, format, layerCount,
Craig Donnere96a3252017-02-02 12:13:34 -0800225 producerUsage, consumerUsage);
Chia-I Wu9ba189d2016-09-22 17:13:08 +0800226 if (!buffer.exportHandle(mMapper, handle, stride)) {
227 return NO_MEMORY;
228 }
229 error = GRALLOC1_ERROR_NONE;
230 } else {
231 auto descriptor = mDevice->createDescriptor();
232 error = descriptor->setDimensions(width, height);
233 if (error != GRALLOC1_ERROR_NONE) {
234 ALOGE("Failed to set dimensions to (%u, %u): %d",
235 width, height, error);
236 return BAD_VALUE;
237 }
238 error = descriptor->setFormat(
239 static_cast<android_pixel_format_t>(format));
240 if (error != GRALLOC1_ERROR_NONE) {
241 ALOGE("Failed to set format to %d: %d", format, error);
242 return BAD_VALUE;
243 }
Craig Donner6ebc46a2016-10-21 15:23:44 -0700244 if (mDevice->hasCapability(GRALLOC1_CAPABILITY_LAYERED_BUFFERS)) {
245 error = descriptor->setLayerCount(layerCount);
246 if (error != GRALLOC1_ERROR_NONE) {
247 ALOGE("Failed to set layer count to %u: %d", layerCount, error);
248 return BAD_VALUE;
249 }
250 } else if (layerCount > 1) {
251 ALOGE("Failed to set layer count to %u: capability unsupported",
252 layerCount);
253 return BAD_VALUE;
254 }
Chia-I Wu9ba189d2016-09-22 17:13:08 +0800255 error = descriptor->setProducerUsage(
Craig Donnere96a3252017-02-02 12:13:34 -0800256 static_cast<gralloc1_producer_usage_t>(producerUsage));
Chia-I Wu9ba189d2016-09-22 17:13:08 +0800257 if (error != GRALLOC1_ERROR_NONE) {
Craig Donnere96a3252017-02-02 12:13:34 -0800258 ALOGE("Failed to set producer usage to %" PRIx64 ": %d",
259 producerUsage, error);
Chia-I Wu9ba189d2016-09-22 17:13:08 +0800260 return BAD_VALUE;
261 }
262 error = descriptor->setConsumerUsage(
Craig Donnere96a3252017-02-02 12:13:34 -0800263 static_cast<gralloc1_consumer_usage_t>(consumerUsage));
Chia-I Wu9ba189d2016-09-22 17:13:08 +0800264 if (error != GRALLOC1_ERROR_NONE) {
Craig Donnere96a3252017-02-02 12:13:34 -0800265 ALOGE("Failed to set consumer usage to %" PRIx64 ": %d",
266 consumerUsage, error);
Chia-I Wu9ba189d2016-09-22 17:13:08 +0800267 return BAD_VALUE;
268 }
Mathias Agopiancbb288b2009-09-07 16:32:45 -0700269
Chia-I Wu9ba189d2016-09-22 17:13:08 +0800270 error = mDevice->allocate(descriptor, graphicBufferId, handle);
271 if (error != GRALLOC1_ERROR_NONE) {
Craig Donnere96a3252017-02-02 12:13:34 -0800272 ALOGE("Failed to allocate (%u x %u) layerCount %u format %d "
273 "producerUsage %" PRIx64 " consumerUsage %" PRIx64 ": %d",
274 width, height, layerCount, format, producerUsage,
275 consumerUsage, error);
Chia-I Wu9ba189d2016-09-22 17:13:08 +0800276 return NO_MEMORY;
277 }
Dan Stozad3182402014-11-17 12:03:59 -0800278
Chia-I Wu9ba189d2016-09-22 17:13:08 +0800279 error = mDevice->getStride(*handle, stride);
280 if (error != GRALLOC1_ERROR_NONE) {
281 ALOGW("Failed to get stride from buffer: %d", error);
282 }
Dan Stoza8deb4da2016-06-01 18:21:44 -0700283 }
284
285 if (error == NO_ERROR) {
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700286 Mutex::Autolock _l(sLock);
287 KeyedVector<buffer_handle_t, alloc_rec_t>& list(sAllocList);
Dan Stozad3182402014-11-17 12:03:59 -0800288 uint32_t bpp = bytesPerPixel(format);
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700289 alloc_rec_t rec;
Dan Stozad3182402014-11-17 12:03:59 -0800290 rec.width = width;
291 rec.height = height;
292 rec.stride = *stride;
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700293 rec.format = format;
Craig Donner6ebc46a2016-10-21 15:23:44 -0700294 rec.layerCount = layerCount;
Craig Donnere96a3252017-02-02 12:13:34 -0800295 rec.producerUsage = producerUsage;
296 rec.consumerUsage = consumerUsage;
Dan Stozad3182402014-11-17 12:03:59 -0800297 rec.size = static_cast<size_t>(height * (*stride) * bpp);
Dan Stoza024e9312016-08-24 12:17:29 -0700298 rec.requestorName = std::move(requestorName);
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700299 list.add(*handle, rec);
300 }
301
Dan Stoza8deb4da2016-06-01 18:21:44 -0700302 return NO_ERROR;
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700303}
304
Mathias Agopian3330b202009-10-05 17:07:12 -0700305status_t GraphicBufferAllocator::free(buffer_handle_t handle)
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700306{
Mathias Agopiancf563192012-02-29 20:43:29 -0800307 ATRACE_CALL();
Mathias Agopian0a757812010-12-08 16:40:01 -0800308
Chia-I Wu9ba189d2016-09-22 17:13:08 +0800309 gralloc1_error_t error;
310 if (mAllocator->valid()) {
311 error = static_cast<gralloc1_error_t>(
312 mMapper.unregisterBuffer(handle));
313 } else {
314 error = mDevice->release(handle);
315 }
316
Dan Stoza8deb4da2016-06-01 18:21:44 -0700317 if (error != GRALLOC1_ERROR_NONE) {
318 ALOGE("Failed to free buffer: %d", error);
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700319 }
320
Dan Stoza8deb4da2016-06-01 18:21:44 -0700321 Mutex::Autolock _l(sLock);
322 KeyedVector<buffer_handle_t, alloc_rec_t>& list(sAllocList);
323 list.removeItem(handle);
324
325 return NO_ERROR;
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700326}
327
Mathias Agopian076b1cc2009-04-10 14:24:30 -0700328// ---------------------------------------------------------------------------
329}; // namespace android