blob: 5694115f61aa9d6bb0798aa6123dce5228128f1b [file] [log] [blame]
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08001/*
2 * Copyright (C) 2006-2007 The Android Open Source Project
3 *
Mark Salyzyn00adb862014-03-19 11:00:06 -07004 * 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
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08007 *
Mark Salyzyn00adb862014-03-19 11:00:06 -07008 * http://www.apache.org/licenses/LICENSE-2.0
The Android Open Source Project9066cfe2009-03-03 19:31:44 -08009 *
Mark Salyzyn00adb862014-03-19 11:00:06 -070010 * 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
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080014 * limitations under the License.
15 */
16
17#undef LOG_TAG
18#define LOG_TAG "CursorWindow"
19
Mathias Agopian49d2b182012-02-27 18:11:20 -080020#include <androidfw/CursorWindow.h>
Jeff Brown9d3b1a42013-07-01 19:07:15 -070021#include <binder/Parcel.h>
22#include <utils/Log.h>
Jeff Brown0cde89f2011-10-10 14:50:10 -070023
24#include <cutils/ashmem.h>
25#include <sys/mman.h>
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080026
27#include <assert.h>
28#include <string.h>
29#include <stdlib.h>
30
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080031namespace android {
32
Jeff Brown0cde89f2011-10-10 14:50:10 -070033CursorWindow::CursorWindow(const String8& name, int ashmemFd,
34 void* data, size_t size, bool readOnly) :
35 mName(name), mAshmemFd(ashmemFd), mData(data), mSize(size), mReadOnly(readOnly) {
36 mHeader = static_cast<Header*>(mData);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080037}
38
Jeff Brown0cde89f2011-10-10 14:50:10 -070039CursorWindow::~CursorWindow() {
40 ::munmap(mData, mSize);
41 ::close(mAshmemFd);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080042}
43
Jeff Brown5e5d6d82011-10-12 15:41:34 -070044status_t CursorWindow::create(const String8& name, size_t size, CursorWindow** outCursorWindow) {
Jeff Brown0cde89f2011-10-10 14:50:10 -070045 String8 ashmemName("CursorWindow: ");
46 ashmemName.append(name);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080047
Jeff Brown0cde89f2011-10-10 14:50:10 -070048 status_t result;
49 int ashmemFd = ashmem_create_region(ashmemName.string(), size);
50 if (ashmemFd < 0) {
51 result = -errno;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080052 } else {
Jeff Brown0cde89f2011-10-10 14:50:10 -070053 result = ashmem_set_prot_region(ashmemFd, PROT_READ | PROT_WRITE);
54 if (result >= 0) {
55 void* data = ::mmap(NULL, size, PROT_READ | PROT_WRITE, MAP_SHARED, ashmemFd, 0);
56 if (data == MAP_FAILED) {
57 result = -errno;
58 } else {
59 result = ashmem_set_prot_region(ashmemFd, PROT_READ);
60 if (result >= 0) {
61 CursorWindow* window = new CursorWindow(name, ashmemFd,
62 data, size, false /*readOnly*/);
63 result = window->clear();
64 if (!result) {
65 LOG_WINDOW("Created new CursorWindow: freeOffset=%d, "
66 "numRows=%d, numColumns=%d, mSize=%d, mData=%p",
67 window->mHeader->freeOffset,
68 window->mHeader->numRows,
69 window->mHeader->numColumns,
70 window->mSize, window->mData);
71 *outCursorWindow = window;
72 return OK;
73 }
74 delete window;
75 }
76 }
77 ::munmap(data, size);
78 }
79 ::close(ashmemFd);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080080 }
Jeff Brown0cde89f2011-10-10 14:50:10 -070081 *outCursorWindow = NULL;
82 return result;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -080083}
84
Jeff Brown0cde89f2011-10-10 14:50:10 -070085status_t CursorWindow::createFromParcel(Parcel* parcel, CursorWindow** outCursorWindow) {
86 String8 name = parcel->readString8();
87
88 status_t result;
89 int ashmemFd = parcel->readFileDescriptor();
90 if (ashmemFd == int(BAD_TYPE)) {
91 result = BAD_TYPE;
92 } else {
93 ssize_t size = ashmem_get_size_region(ashmemFd);
94 if (size < 0) {
95 result = UNKNOWN_ERROR;
96 } else {
97 int dupAshmemFd = ::dup(ashmemFd);
98 if (dupAshmemFd < 0) {
99 result = -errno;
100 } else {
Fyodor Kupolov45e2e952017-02-13 18:35:12 -0800101 // the size of the ashmem descriptor can be modified between ashmem_get_size_region
102 // call and mmap, so we'll check again immediately after memory is mapped
Jeff Brown0cde89f2011-10-10 14:50:10 -0700103 void* data = ::mmap(NULL, size, PROT_READ, MAP_SHARED, dupAshmemFd, 0);
104 if (data == MAP_FAILED) {
105 result = -errno;
Fyodor Kupolov45e2e952017-02-13 18:35:12 -0800106 } else if (ashmem_get_size_region(dupAshmemFd) != size) {
107 ::munmap(data, size);
108 result = BAD_VALUE;
Jeff Brown0cde89f2011-10-10 14:50:10 -0700109 } else {
110 CursorWindow* window = new CursorWindow(name, dupAshmemFd,
111 data, size, true /*readOnly*/);
112 LOG_WINDOW("Created CursorWindow from parcel: freeOffset=%d, "
113 "numRows=%d, numColumns=%d, mSize=%d, mData=%p",
114 window->mHeader->freeOffset,
115 window->mHeader->numRows,
116 window->mHeader->numColumns,
117 window->mSize, window->mData);
118 *outCursorWindow = window;
119 return OK;
120 }
121 ::close(dupAshmemFd);
122 }
123 }
124 }
125 *outCursorWindow = NULL;
126 return result;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800127}
128
Jeff Brown0cde89f2011-10-10 14:50:10 -0700129status_t CursorWindow::writeToParcel(Parcel* parcel) {
130 status_t status = parcel->writeString8(mName);
131 if (!status) {
132 status = parcel->writeDupFileDescriptor(mAshmemFd);
133 }
134 return status;
135}
136
137status_t CursorWindow::clear() {
138 if (mReadOnly) {
139 return INVALID_OPERATION;
140 }
141
142 mHeader->freeOffset = sizeof(Header) + sizeof(RowSlotChunk);
143 mHeader->firstChunkOffset = sizeof(Header);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800144 mHeader->numRows = 0;
145 mHeader->numColumns = 0;
Jeff Brown0cde89f2011-10-10 14:50:10 -0700146
147 RowSlotChunk* firstChunk = static_cast<RowSlotChunk*>(offsetToPtr(mHeader->firstChunkOffset));
148 firstChunk->nextChunkOffset = 0;
149 return OK;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800150}
151
Jeff Brown0cde89f2011-10-10 14:50:10 -0700152status_t CursorWindow::setNumColumns(uint32_t numColumns) {
153 if (mReadOnly) {
154 return INVALID_OPERATION;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800155 }
Jeff Brown0cde89f2011-10-10 14:50:10 -0700156
157 uint32_t cur = mHeader->numColumns;
158 if ((cur > 0 || mHeader->numRows > 0) && cur != numColumns) {
Steve Block3762c312012-01-06 19:20:56 +0000159 ALOGE("Trying to go from %d columns to %d", cur, numColumns);
Jeff Brown0cde89f2011-10-10 14:50:10 -0700160 return INVALID_OPERATION;
161 }
162 mHeader->numColumns = numColumns;
163 return OK;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800164}
165
Jeff Brown0cde89f2011-10-10 14:50:10 -0700166status_t CursorWindow::allocRow() {
167 if (mReadOnly) {
168 return INVALID_OPERATION;
169 }
170
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800171 // Fill in the row slot
Jeff Brown0cde89f2011-10-10 14:50:10 -0700172 RowSlot* rowSlot = allocRowSlot();
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800173 if (rowSlot == NULL) {
Jeff Brown0cde89f2011-10-10 14:50:10 -0700174 return NO_MEMORY;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800175 }
176
177 // Allocate the slots for the field directory
Jeff Brown0cde89f2011-10-10 14:50:10 -0700178 size_t fieldDirSize = mHeader->numColumns * sizeof(FieldSlot);
179 uint32_t fieldDirOffset = alloc(fieldDirSize, true /*aligned*/);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800180 if (!fieldDirOffset) {
181 mHeader->numRows--;
Jeff Brown0cde89f2011-10-10 14:50:10 -0700182 LOG_WINDOW("The row failed, so back out the new row accounting "
183 "from allocRowSlot %d", mHeader->numRows);
184 return NO_MEMORY;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800185 }
Jeff Brown0cde89f2011-10-10 14:50:10 -0700186 FieldSlot* fieldDir = static_cast<FieldSlot*>(offsetToPtr(fieldDirOffset));
187 memset(fieldDir, 0, fieldDirSize);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800188
Jeff Brown0cde89f2011-10-10 14:50:10 -0700189 LOG_WINDOW("Allocated row %u, rowSlot is at offset %u, fieldDir is %d bytes at offset %u\n",
190 mHeader->numRows - 1, offsetFromPtr(rowSlot), fieldDirSize, fieldDirOffset);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800191 rowSlot->offset = fieldDirOffset;
Jeff Brown0cde89f2011-10-10 14:50:10 -0700192 return OK;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800193}
194
Jeff Brown0cde89f2011-10-10 14:50:10 -0700195status_t CursorWindow::freeLastRow() {
196 if (mReadOnly) {
197 return INVALID_OPERATION;
198 }
199
200 if (mHeader->numRows > 0) {
201 mHeader->numRows--;
202 }
203 return OK;
204}
205
206uint32_t CursorWindow::alloc(size_t size, bool aligned) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800207 uint32_t padding;
208 if (aligned) {
209 // 4 byte alignment
Jeff Brown0cde89f2011-10-10 14:50:10 -0700210 padding = (~mHeader->freeOffset + 1) & 3;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800211 } else {
212 padding = 0;
213 }
214
Jeff Brown0cde89f2011-10-10 14:50:10 -0700215 uint32_t offset = mHeader->freeOffset + padding;
216 uint32_t nextFreeOffset = offset + size;
217 if (nextFreeOffset > mSize) {
Ashok Bhatf5df7002014-03-25 20:51:35 +0000218 ALOGW("Window is full: requested allocation %zu bytes, "
219 "free space %zu bytes, window size %zu bytes",
Jeff Brown0cde89f2011-10-10 14:50:10 -0700220 size, freeSpace(), mSize);
221 return 0;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800222 }
223
Jeff Brown0cde89f2011-10-10 14:50:10 -0700224 mHeader->freeOffset = nextFreeOffset;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800225 return offset;
226}
227
Jeff Brown0cde89f2011-10-10 14:50:10 -0700228CursorWindow::RowSlot* CursorWindow::getRowSlot(uint32_t row) {
229 uint32_t chunkPos = row;
230 RowSlotChunk* chunk = static_cast<RowSlotChunk*>(
231 offsetToPtr(mHeader->firstChunkOffset));
232 while (chunkPos >= ROW_SLOT_CHUNK_NUM_ROWS) {
233 chunk = static_cast<RowSlotChunk*>(offsetToPtr(chunk->nextChunkOffset));
234 chunkPos -= ROW_SLOT_CHUNK_NUM_ROWS;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800235 }
Jeff Brown0cde89f2011-10-10 14:50:10 -0700236 return &chunk->slots[chunkPos];
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800237}
238
Jeff Brown0cde89f2011-10-10 14:50:10 -0700239CursorWindow::RowSlot* CursorWindow::allocRowSlot() {
240 uint32_t chunkPos = mHeader->numRows;
241 RowSlotChunk* chunk = static_cast<RowSlotChunk*>(
242 offsetToPtr(mHeader->firstChunkOffset));
243 while (chunkPos > ROW_SLOT_CHUNK_NUM_ROWS) {
244 chunk = static_cast<RowSlotChunk*>(offsetToPtr(chunk->nextChunkOffset));
245 chunkPos -= ROW_SLOT_CHUNK_NUM_ROWS;
246 }
247 if (chunkPos == ROW_SLOT_CHUNK_NUM_ROWS) {
248 if (!chunk->nextChunkOffset) {
249 chunk->nextChunkOffset = alloc(sizeof(RowSlotChunk), true /*aligned*/);
250 if (!chunk->nextChunkOffset) {
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800251 return NULL;
252 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800253 }
Jeff Brown0cde89f2011-10-10 14:50:10 -0700254 chunk = static_cast<RowSlotChunk*>(offsetToPtr(chunk->nextChunkOffset));
255 chunk->nextChunkOffset = 0;
256 chunkPos = 0;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800257 }
Jeff Brown0cde89f2011-10-10 14:50:10 -0700258 mHeader->numRows += 1;
259 return &chunk->slots[chunkPos];
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800260}
261
Jeff Brown0cde89f2011-10-10 14:50:10 -0700262CursorWindow::FieldSlot* CursorWindow::getFieldSlot(uint32_t row, uint32_t column) {
263 if (row >= mHeader->numRows || column >= mHeader->numColumns) {
Steve Block3762c312012-01-06 19:20:56 +0000264 ALOGE("Failed to read row %d, column %d from a CursorWindow which "
Jeff Brown0cde89f2011-10-10 14:50:10 -0700265 "has %d rows, %d columns.",
266 row, column, mHeader->numRows, mHeader->numColumns);
267 return NULL;
268 }
269 RowSlot* rowSlot = getRowSlot(row);
270 if (!rowSlot) {
Steve Block3762c312012-01-06 19:20:56 +0000271 ALOGE("Failed to find rowSlot for row %d.", row);
Jeff Brown0cde89f2011-10-10 14:50:10 -0700272 return NULL;
273 }
274 FieldSlot* fieldDir = static_cast<FieldSlot*>(offsetToPtr(rowSlot->offset));
275 return &fieldDir[column];
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800276}
277
Jeff Brown0cde89f2011-10-10 14:50:10 -0700278status_t CursorWindow::putBlob(uint32_t row, uint32_t column, const void* value, size_t size) {
279 return putBlobOrString(row, column, value, size, FIELD_TYPE_BLOB);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800280}
281
Jeff Brown0cde89f2011-10-10 14:50:10 -0700282status_t CursorWindow::putString(uint32_t row, uint32_t column, const char* value,
283 size_t sizeIncludingNull) {
284 return putBlobOrString(row, column, value, sizeIncludingNull, FIELD_TYPE_STRING);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800285}
286
Jeff Brown0cde89f2011-10-10 14:50:10 -0700287status_t CursorWindow::putBlobOrString(uint32_t row, uint32_t column,
288 const void* value, size_t size, int32_t type) {
289 if (mReadOnly) {
290 return INVALID_OPERATION;
291 }
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800292
Jeff Brown0cde89f2011-10-10 14:50:10 -0700293 FieldSlot* fieldSlot = getFieldSlot(row, column);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800294 if (!fieldSlot) {
Jeff Brown0cde89f2011-10-10 14:50:10 -0700295 return BAD_VALUE;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800296 }
297
Jeff Brown0cde89f2011-10-10 14:50:10 -0700298 uint32_t offset = alloc(size);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800299 if (!offset) {
Jeff Brown0cde89f2011-10-10 14:50:10 -0700300 return NO_MEMORY;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800301 }
302
Jeff Brown0cde89f2011-10-10 14:50:10 -0700303 memcpy(offsetToPtr(offset), value, size);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800304
Jeff Brown0cde89f2011-10-10 14:50:10 -0700305 fieldSlot->type = type;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800306 fieldSlot->data.buffer.offset = offset;
Jeff Brown0cde89f2011-10-10 14:50:10 -0700307 fieldSlot->data.buffer.size = size;
308 return OK;
309}
310
311status_t CursorWindow::putLong(uint32_t row, uint32_t column, int64_t value) {
312 if (mReadOnly) {
313 return INVALID_OPERATION;
314 }
315
316 FieldSlot* fieldSlot = getFieldSlot(row, column);
317 if (!fieldSlot) {
318 return BAD_VALUE;
319 }
320
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800321 fieldSlot->type = FIELD_TYPE_INTEGER;
Jeff Brown0cde89f2011-10-10 14:50:10 -0700322 fieldSlot->data.l = value;
323 return OK;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800324}
325
Jeff Brown0cde89f2011-10-10 14:50:10 -0700326status_t CursorWindow::putDouble(uint32_t row, uint32_t column, double value) {
327 if (mReadOnly) {
328 return INVALID_OPERATION;
329 }
330
331 FieldSlot* fieldSlot = getFieldSlot(row, column);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800332 if (!fieldSlot) {
Jeff Brown0cde89f2011-10-10 14:50:10 -0700333 return BAD_VALUE;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800334 }
335
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800336 fieldSlot->type = FIELD_TYPE_FLOAT;
Jeff Brown0cde89f2011-10-10 14:50:10 -0700337 fieldSlot->data.d = value;
338 return OK;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800339}
340
Jeff Brown0cde89f2011-10-10 14:50:10 -0700341status_t CursorWindow::putNull(uint32_t row, uint32_t column) {
342 if (mReadOnly) {
343 return INVALID_OPERATION;
344 }
345
346 FieldSlot* fieldSlot = getFieldSlot(row, column);
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800347 if (!fieldSlot) {
Jeff Brown0cde89f2011-10-10 14:50:10 -0700348 return BAD_VALUE;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800349 }
350
351 fieldSlot->type = FIELD_TYPE_NULL;
352 fieldSlot->data.buffer.offset = 0;
353 fieldSlot->data.buffer.size = 0;
Jeff Brown0cde89f2011-10-10 14:50:10 -0700354 return OK;
The Android Open Source Project9066cfe2009-03-03 19:31:44 -0800355}
356
357}; // namespace android