1 /*
2 * Copyright (C) 2013 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 #include "RenderScript.h"
18 #include "rsCppInternal.h"
19
20 using namespace android;
21 using namespace RSC;
22
getIDSafe() const23 void * Allocation::getIDSafe() const {
24 return getID();
25 }
26
updateCacheInfo(sp<const Type> t)27 void Allocation::updateCacheInfo(sp<const Type> t) {
28 mCurrentDimX = t->getX();
29 mCurrentDimY = t->getY();
30 mCurrentDimZ = t->getZ();
31 mCurrentCount = mCurrentDimX;
32 if (mCurrentDimY > 1) {
33 mCurrentCount *= mCurrentDimY;
34 }
35 if (mCurrentDimZ > 1) {
36 mCurrentCount *= mCurrentDimZ;
37 }
38 }
39
Allocation(void * id,sp<RS> rs,sp<const Type> t,uint32_t usage)40 Allocation::Allocation(void *id, sp<RS> rs, sp<const Type> t, uint32_t usage) :
41 BaseObj(id, rs), mSelectedY(0), mSelectedZ(0), mSelectedLOD(0),
42 mSelectedFace(RS_ALLOCATION_CUBEMAP_FACE_POSITIVE_X) {
43
44 if ((usage & ~(RS_ALLOCATION_USAGE_SCRIPT |
45 RS_ALLOCATION_USAGE_GRAPHICS_TEXTURE |
46 RS_ALLOCATION_USAGE_GRAPHICS_VERTEX |
47 RS_ALLOCATION_USAGE_GRAPHICS_CONSTANTS |
48 RS_ALLOCATION_USAGE_GRAPHICS_RENDER_TARGET |
49 RS_ALLOCATION_USAGE_IO_INPUT |
50 RS_ALLOCATION_USAGE_IO_OUTPUT |
51 RS_ALLOCATION_USAGE_SHARED)) != 0) {
52 ALOGE("Unknown usage specified.");
53 }
54
55 if ((usage & RS_ALLOCATION_USAGE_IO_INPUT) != 0) {
56 mWriteAllowed = false;
57 if ((usage & ~(RS_ALLOCATION_USAGE_IO_INPUT |
58 RS_ALLOCATION_USAGE_GRAPHICS_TEXTURE |
59 RS_ALLOCATION_USAGE_SCRIPT)) != 0) {
60 ALOGE("Invalid usage combination.");
61 }
62 }
63
64 mType = t;
65 mUsage = usage;
66
67 if (t != NULL) {
68 updateCacheInfo(t);
69 }
70
71 }
72
73
74
validateIsInt32()75 void Allocation::validateIsInt32() {
76 RsDataType dt = mType->getElement()->getDataType();
77 if ((dt == RS_TYPE_SIGNED_32) || (dt == RS_TYPE_UNSIGNED_32)) {
78 return;
79 }
80 ALOGE("32 bit integer source does not match allocation type %i", dt);
81 }
82
validateIsInt16()83 void Allocation::validateIsInt16() {
84 RsDataType dt = mType->getElement()->getDataType();
85 if ((dt == RS_TYPE_SIGNED_16) || (dt == RS_TYPE_UNSIGNED_16)) {
86 return;
87 }
88 ALOGE("16 bit integer source does not match allocation type %i", dt);
89 }
90
validateIsInt8()91 void Allocation::validateIsInt8() {
92 RsDataType dt = mType->getElement()->getDataType();
93 if ((dt == RS_TYPE_SIGNED_8) || (dt == RS_TYPE_UNSIGNED_8)) {
94 return;
95 }
96 ALOGE("8 bit integer source does not match allocation type %i", dt);
97 }
98
validateIsFloat32()99 void Allocation::validateIsFloat32() {
100 RsDataType dt = mType->getElement()->getDataType();
101 if (dt == RS_TYPE_FLOAT_32) {
102 return;
103 }
104 ALOGE("32 bit float source does not match allocation type %i", dt);
105 }
106
validateIsObject()107 void Allocation::validateIsObject() {
108 RsDataType dt = mType->getElement()->getDataType();
109 if ((dt == RS_TYPE_ELEMENT) ||
110 (dt == RS_TYPE_TYPE) ||
111 (dt == RS_TYPE_ALLOCATION) ||
112 (dt == RS_TYPE_SAMPLER) ||
113 (dt == RS_TYPE_SCRIPT) ||
114 (dt == RS_TYPE_MESH) ||
115 (dt == RS_TYPE_PROGRAM_FRAGMENT) ||
116 (dt == RS_TYPE_PROGRAM_VERTEX) ||
117 (dt == RS_TYPE_PROGRAM_RASTER) ||
118 (dt == RS_TYPE_PROGRAM_STORE)) {
119 return;
120 }
121 ALOGE("Object source does not match allocation type %i", dt);
122 }
123
updateFromNative()124 void Allocation::updateFromNative() {
125 BaseObj::updateFromNative();
126
127 const void *typeID = RS::dispatch->AllocationGetType(mRS->getContext(), getID());
128 if(typeID != NULL) {
129 sp<const Type> old = mType;
130 sp<Type> t = new Type((void *)typeID, mRS);
131 t->updateFromNative();
132 updateCacheInfo(t);
133 mType = t;
134 }
135 }
136
syncAll(RsAllocationUsageType srcLocation)137 void Allocation::syncAll(RsAllocationUsageType srcLocation) {
138 switch (srcLocation) {
139 case RS_ALLOCATION_USAGE_SCRIPT:
140 case RS_ALLOCATION_USAGE_GRAPHICS_CONSTANTS:
141 case RS_ALLOCATION_USAGE_GRAPHICS_TEXTURE:
142 case RS_ALLOCATION_USAGE_GRAPHICS_VERTEX:
143 case RS_ALLOCATION_USAGE_SHARED:
144 break;
145 default:
146 mRS->throwError(RS_ERROR_INVALID_PARAMETER, "Source must be exactly one usage type.");
147 return;
148 }
149 tryDispatch(mRS, RS::dispatch->AllocationSyncAll(mRS->getContext(), getIDSafe(), srcLocation));
150 }
151
ioSendOutput()152 void Allocation::ioSendOutput() {
153 #ifndef RS_COMPATIBILITY_LIB
154 if ((mUsage & RS_ALLOCATION_USAGE_IO_OUTPUT) == 0) {
155 mRS->throwError(RS_ERROR_INVALID_PARAMETER, "Can only send buffer if IO_OUTPUT usage specified.");
156 return;
157 }
158 tryDispatch(mRS, RS::dispatch->AllocationIoSend(mRS->getContext(), getID()));
159 #endif
160 }
161
ioGetInput()162 void Allocation::ioGetInput() {
163 #ifndef RS_COMPATIBILITY_LIB
164 if ((mUsage & RS_ALLOCATION_USAGE_IO_INPUT) == 0) {
165 mRS->throwError(RS_ERROR_INVALID_PARAMETER, "Can only send buffer if IO_OUTPUT usage specified.");
166 return;
167 }
168 tryDispatch(mRS, RS::dispatch->AllocationIoReceive(mRS->getContext(), getID()));
169 #endif
170 }
171
generateMipmaps()172 void Allocation::generateMipmaps() {
173 tryDispatch(mRS, RS::dispatch->AllocationGenerateMipmaps(mRS->getContext(), getID()));
174 }
175
copy1DRangeFrom(uint32_t off,size_t count,const void * data)176 void Allocation::copy1DRangeFrom(uint32_t off, size_t count, const void *data) {
177
178 if(count < 1) {
179 mRS->throwError(RS_ERROR_INVALID_PARAMETER, "Count must be >= 1.");
180 return;
181 }
182 if((off + count) > mCurrentCount) {
183 ALOGE("Overflow, Available count %zu, got %zu at offset %zu.", mCurrentCount, count, off);
184 mRS->throwError(RS_ERROR_INVALID_PARAMETER, "Invalid copy specified");
185 return;
186 }
187
188 tryDispatch(mRS, RS::dispatch->Allocation1DData(mRS->getContext(), getIDSafe(), off, mSelectedLOD,
189 count, data, count * mType->getElement()->getSizeBytes()));
190 }
191
copy1DRangeTo(uint32_t off,size_t count,void * data)192 void Allocation::copy1DRangeTo(uint32_t off, size_t count, void *data) {
193 if(count < 1) {
194 mRS->throwError(RS_ERROR_INVALID_PARAMETER, "Count must be >= 1.");
195 return;
196 }
197 if((off + count) > mCurrentCount) {
198 ALOGE("Overflow, Available count %zu, got %zu at offset %zu.", mCurrentCount, count, off);
199 mRS->throwError(RS_ERROR_INVALID_PARAMETER, "Invalid copy specified");
200 return;
201 }
202
203 tryDispatch(mRS, RS::dispatch->Allocation1DRead(mRS->getContext(), getIDSafe(), off, mSelectedLOD,
204 count, data, count * mType->getElement()->getSizeBytes()));
205 }
206
copy1DRangeFrom(uint32_t off,size_t count,sp<const Allocation> data,uint32_t dataOff)207 void Allocation::copy1DRangeFrom(uint32_t off, size_t count, sp<const Allocation> data,
208 uint32_t dataOff) {
209
210 tryDispatch(mRS, RS::dispatch->AllocationCopy2DRange(mRS->getContext(), getIDSafe(), off, 0,
211 mSelectedLOD, mSelectedFace,
212 count, 1, data->getIDSafe(), dataOff, 0,
213 data->mSelectedLOD, data->mSelectedFace));
214 }
215
copy1DFrom(const void * data)216 void Allocation::copy1DFrom(const void* data) {
217 copy1DRangeFrom(0, mCurrentCount, data);
218 }
219
copy1DTo(void * data)220 void Allocation::copy1DTo(void* data) {
221 copy1DRangeTo(0, mCurrentCount, data);
222 }
223
224
validate2DRange(uint32_t xoff,uint32_t yoff,uint32_t w,uint32_t h)225 void Allocation::validate2DRange(uint32_t xoff, uint32_t yoff, uint32_t w, uint32_t h) {
226 if (mAdaptedAllocation != NULL) {
227
228 } else {
229 if (((xoff + w) > mCurrentDimX) || ((yoff + h) > mCurrentDimY)) {
230 mRS->throwError(RS_ERROR_INVALID_PARAMETER, "Updated region larger than allocation.");
231 }
232 }
233 }
234
copy2DRangeFrom(uint32_t xoff,uint32_t yoff,uint32_t w,uint32_t h,const void * data)235 void Allocation::copy2DRangeFrom(uint32_t xoff, uint32_t yoff, uint32_t w, uint32_t h,
236 const void *data) {
237 validate2DRange(xoff, yoff, w, h);
238 tryDispatch(mRS, RS::dispatch->Allocation2DData(mRS->getContext(), getIDSafe(), xoff,
239 yoff, mSelectedLOD, mSelectedFace,
240 w, h, data, w * h * mType->getElement()->getSizeBytes(),
241 w * mType->getElement()->getSizeBytes()));
242 }
243
copy2DRangeFrom(uint32_t xoff,uint32_t yoff,uint32_t w,uint32_t h,sp<const Allocation> data,uint32_t dataXoff,uint32_t dataYoff)244 void Allocation::copy2DRangeFrom(uint32_t xoff, uint32_t yoff, uint32_t w, uint32_t h,
245 sp<const Allocation> data, uint32_t dataXoff, uint32_t dataYoff) {
246 validate2DRange(xoff, yoff, w, h);
247 tryDispatch(mRS, RS::dispatch->AllocationCopy2DRange(mRS->getContext(), getIDSafe(), xoff, yoff,
248 mSelectedLOD, mSelectedFace,
249 w, h, data->getIDSafe(), dataXoff, dataYoff,
250 data->mSelectedLOD, data->mSelectedFace));
251 }
252
copy2DRangeTo(uint32_t xoff,uint32_t yoff,uint32_t w,uint32_t h,void * data)253 void Allocation::copy2DRangeTo(uint32_t xoff, uint32_t yoff, uint32_t w, uint32_t h,
254 void* data) {
255 validate2DRange(xoff, yoff, w, h);
256 tryDispatch(mRS, RS::dispatch->Allocation2DRead(mRS->getContext(), getIDSafe(), xoff, yoff,
257 mSelectedLOD, mSelectedFace, w, h, data,
258 w * h * mType->getElement()->getSizeBytes(),
259 w * mType->getElement()->getSizeBytes()));
260 }
261
copy2DStridedFrom(uint32_t xoff,uint32_t yoff,uint32_t w,uint32_t h,const void * data,size_t stride)262 void Allocation::copy2DStridedFrom(uint32_t xoff, uint32_t yoff, uint32_t w, uint32_t h,
263 const void *data, size_t stride) {
264 validate2DRange(xoff, yoff, w, h);
265 tryDispatch(mRS, RS::dispatch->Allocation2DData(mRS->getContext(), getIDSafe(), xoff, yoff,
266 mSelectedLOD, mSelectedFace, w, h, data,
267 w * h * mType->getElement()->getSizeBytes(), stride));
268 }
269
copy2DStridedFrom(const void * data,size_t stride)270 void Allocation::copy2DStridedFrom(const void* data, size_t stride) {
271 copy2DStridedFrom(0, 0, mCurrentDimX, mCurrentDimY, data, stride);
272 }
273
copy2DStridedTo(uint32_t xoff,uint32_t yoff,uint32_t w,uint32_t h,void * data,size_t stride)274 void Allocation::copy2DStridedTo(uint32_t xoff, uint32_t yoff, uint32_t w, uint32_t h,
275 void *data, size_t stride) {
276 validate2DRange(xoff, yoff, w, h);
277 tryDispatch(mRS, RS::dispatch->Allocation2DRead(mRS->getContext(), getIDSafe(), xoff, yoff,
278 mSelectedLOD, mSelectedFace, w, h, data,
279 w * h * mType->getElement()->getSizeBytes(), stride));
280 }
281
copy2DStridedTo(void * data,size_t stride)282 void Allocation::copy2DStridedTo(void* data, size_t stride) {
283 copy2DStridedTo(0, 0, mCurrentDimX, mCurrentDimY, data, stride);
284 }
285
validate3DRange(uint32_t xoff,uint32_t yoff,uint32_t zoff,uint32_t w,uint32_t h,uint32_t d)286 void Allocation::validate3DRange(uint32_t xoff, uint32_t yoff, uint32_t zoff, uint32_t w,
287 uint32_t h, uint32_t d) {
288 if (mAdaptedAllocation != NULL) {
289
290 } else {
291 if (((xoff + w) > mCurrentDimX) || ((yoff + h) > mCurrentDimY) || ((zoff + d) > mCurrentDimZ)) {
292 mRS->throwError(RS_ERROR_INVALID_PARAMETER, "Updated region larger than allocation.");
293 }
294 }
295 }
296
copy3DRangeFrom(uint32_t xoff,uint32_t yoff,uint32_t zoff,uint32_t w,uint32_t h,uint32_t d,const void * data)297 void Allocation::copy3DRangeFrom(uint32_t xoff, uint32_t yoff, uint32_t zoff, uint32_t w,
298 uint32_t h, uint32_t d, const void* data) {
299 validate3DRange(xoff, yoff, zoff, w, h, d);
300 tryDispatch(mRS, RS::dispatch->Allocation3DData(mRS->getContext(), getIDSafe(), xoff, yoff, zoff,
301 mSelectedLOD, w, h, d, data,
302 w * h * d * mType->getElement()->getSizeBytes(),
303 w * mType->getElement()->getSizeBytes()));
304 }
305
copy3DRangeFrom(uint32_t xoff,uint32_t yoff,uint32_t zoff,uint32_t w,uint32_t h,uint32_t d,sp<const Allocation> data,uint32_t dataXoff,uint32_t dataYoff,uint32_t dataZoff)306 void Allocation::copy3DRangeFrom(uint32_t xoff, uint32_t yoff, uint32_t zoff, uint32_t w, uint32_t h, uint32_t d,
307 sp<const Allocation> data, uint32_t dataXoff, uint32_t dataYoff, uint32_t dataZoff) {
308 validate3DRange(xoff, yoff, zoff, dataXoff, dataYoff, dataZoff);
309 tryDispatch(mRS, RS::dispatch->AllocationCopy3DRange(mRS->getContext(), getIDSafe(), xoff, yoff, zoff,
310 mSelectedLOD, w, h, d, data->getIDSafe(),
311 dataXoff, dataYoff, dataZoff, data->mSelectedLOD));
312 }
313
314
createTyped(sp<RS> rs,sp<const Type> type,RsAllocationMipmapControl mips,uint32_t usage)315 sp<Allocation> Allocation::createTyped(sp<RS> rs, sp<const Type> type,
316 RsAllocationMipmapControl mips, uint32_t usage) {
317 void *id = 0;
318 if (rs->getError() == RS_SUCCESS) {
319 id = RS::dispatch->AllocationCreateTyped(rs->getContext(), type->getID(), mips, usage, 0);
320 }
321 if (id == 0) {
322 rs->throwError(RS_ERROR_RUNTIME_ERROR, "Allocation creation failed");
323 return NULL;
324 }
325 return new Allocation(id, rs, type, usage);
326 }
327
createTyped(sp<RS> rs,sp<const Type> type,RsAllocationMipmapControl mips,uint32_t usage,void * pointer)328 sp<Allocation> Allocation::createTyped(sp<RS> rs, sp<const Type> type,
329 RsAllocationMipmapControl mips, uint32_t usage,
330 void *pointer) {
331 void *id = 0;
332 if (rs->getError() == RS_SUCCESS) {
333 id = RS::dispatch->AllocationCreateTyped(rs->getContext(), type->getID(), mips, usage,
334 (uintptr_t)pointer);
335 }
336 if (id == 0) {
337 rs->throwError(RS_ERROR_RUNTIME_ERROR, "Allocation creation failed");
338 return NULL;
339 }
340 return new Allocation(id, rs, type, usage);
341 }
342
createTyped(sp<RS> rs,sp<const Type> type,uint32_t usage)343 sp<Allocation> Allocation::createTyped(sp<RS> rs, sp<const Type> type,
344 uint32_t usage) {
345 return createTyped(rs, type, RS_ALLOCATION_MIPMAP_NONE, usage);
346 }
347
createSized(sp<RS> rs,sp<const Element> e,size_t count,uint32_t usage)348 sp<Allocation> Allocation::createSized(sp<RS> rs, sp<const Element> e,
349 size_t count, uint32_t usage) {
350 Type::Builder b(rs, e);
351 b.setX(count);
352 sp<const Type> t = b.create();
353
354 return createTyped(rs, t, usage);
355 }
356
createSized2D(sp<RS> rs,sp<const Element> e,size_t x,size_t y,uint32_t usage)357 sp<Allocation> Allocation::createSized2D(sp<RS> rs, sp<const Element> e,
358 size_t x, size_t y, uint32_t usage) {
359 Type::Builder b(rs, e);
360 b.setX(x);
361 b.setY(y);
362 sp<const Type> t = b.create();
363
364 return createTyped(rs, t, usage);
365 }
366