1326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams/*
2bc0ca6ba4e31239bf77060578d0bdf1a10e04168Jason Sams * Copyright (C) 2013 The Android Open Source Project
3326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams *
4326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams * Licensed under the Apache License, Version 2.0 (the "License");
5326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams * you may not use this file except in compliance with the License.
6326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams * You may obtain a copy of the License at
7326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams *
8326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams *      http://www.apache.org/licenses/LICENSE-2.0
9326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams *
10326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams * Unless required by applicable law or agreed to in writing, software
11326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams * distributed under the License is distributed on an "AS IS" BASIS,
12326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams * See the License for the specific language governing permissions and
14326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams * limitations under the License.
15326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams */
16326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams
17326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams#include "rsContext.h"
18326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams
1910f317038dd53543dc8c7f5afe26a6360adfb5f3Stephen Hines#if !defined(RS_SERVER) && !defined(RS_COMPATIBILITY_LIB)
20bc0ca6ba4e31239bf77060578d0bdf1a10e04168Jason Sams#include "system/graphics.h"
210b575de8ed0b628d84d256f5846500b0385979bdTim Murray#endif
22bc0ca6ba4e31239bf77060578d0bdf1a10e04168Jason Sams
2310f317038dd53543dc8c7f5afe26a6360adfb5f3Stephen Hines#ifdef RS_COMPATIBILITY_LIB
2410f317038dd53543dc8c7f5afe26a6360adfb5f3Stephen Hines#include "rsCompatibilityLib.h"
2510f317038dd53543dc8c7f5afe26a6360adfb5f3Stephen Hines#endif
2610f317038dd53543dc8c7f5afe26a6360adfb5f3Stephen Hines
27326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsusing namespace android;
28326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsusing namespace android::renderscript;
29326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams
30afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex SakhartchoukType::Type(Context *rsc) : ObjectBase(rsc) {
31246fbee66ccb17cb1f08a5420e104a709183bce1Alex Sakhartchouk    memset(&mHal, 0, sizeof(mHal));
32246fbee66ccb17cb1f08a5420e104a709183bce1Alex Sakhartchouk    mDimLOD = false;
33326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams}
34326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams
35c700e649ca44d0dcff8b271e42d949ea72fe3c63Alex Sakhartchoukvoid Type::preDestroy() const {
368154954868694e1f233d87d4933a474518b1cb81Jason Sams    for (uint32_t ct = 0; ct < mRSC->mStateType.mTypes.size(); ct++) {
378154954868694e1f233d87d4933a474518b1cb81Jason Sams        if (mRSC->mStateType.mTypes[ct] == this) {
388154954868694e1f233d87d4933a474518b1cb81Jason Sams            mRSC->mStateType.mTypes.removeAt(ct);
398154954868694e1f233d87d4933a474518b1cb81Jason Sams            break;
408154954868694e1f233d87d4933a474518b1cb81Jason Sams        }
418154954868694e1f233d87d4933a474518b1cb81Jason Sams    }
42225afd317e101a7be5fe02c0a86361146ea89f05Jason Sams}
43225afd317e101a7be5fe02c0a86361146ea89f05Jason Sams
44afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex SakhartchoukType::~Type() {
45246fbee66ccb17cb1f08a5420e104a709183bce1Alex Sakhartchouk    clear();
46326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams}
47326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams
4849a87772b9abe620d28cbe110260424b8950055dTim Murrayvoid Type::operator delete(void* ptr) {
4949a87772b9abe620d28cbe110260424b8950055dTim Murray    if (ptr) {
5049a87772b9abe620d28cbe110260424b8950055dTim Murray        Type *t = (Type*) ptr;
5149a87772b9abe620d28cbe110260424b8950055dTim Murray        t->getContext()->mHal.funcs.freeRuntimeMem(ptr);
5249a87772b9abe620d28cbe110260424b8950055dTim Murray    }
5349a87772b9abe620d28cbe110260424b8950055dTim Murray}
5449a87772b9abe620d28cbe110260424b8950055dTim Murray
55afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex Sakhartchoukvoid Type::clear() {
56246fbee66ccb17cb1f08a5420e104a709183bce1Alex Sakhartchouk    if (mHal.state.lodCount) {
57246fbee66ccb17cb1f08a5420e104a709183bce1Alex Sakhartchouk        delete [] mHal.state.lodDimX;
58246fbee66ccb17cb1f08a5420e104a709183bce1Alex Sakhartchouk        delete [] mHal.state.lodDimY;
59246fbee66ccb17cb1f08a5420e104a709183bce1Alex Sakhartchouk        delete [] mHal.state.lodDimZ;
60326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams    }
61326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams    mElement.clear();
6250bfc354e61e174a465893fd0dafe913f1954478Alex Sakhartchouk    memset(&mHal, 0, sizeof(mHal));
63326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams}
64326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams
65afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex SakhartchoukTypeState::TypeState() {
66326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams}
67326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams
68afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex SakhartchoukTypeState::~TypeState() {
69c700e649ca44d0dcff8b271e42d949ea72fe3c63Alex Sakhartchouk    rsAssert(!mTypes.size());
70326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams}
71326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams
72afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex Sakhartchoukvoid Type::compute() {
73246fbee66ccb17cb1f08a5420e104a709183bce1Alex Sakhartchouk    uint32_t oldLODCount = mHal.state.lodCount;
74246fbee66ccb17cb1f08a5420e104a709183bce1Alex Sakhartchouk    if (mDimLOD) {
7550bfc354e61e174a465893fd0dafe913f1954478Alex Sakhartchouk        uint32_t l2x = rsFindHighBit(mHal.state.dimX) + 1;
7650bfc354e61e174a465893fd0dafe913f1954478Alex Sakhartchouk        uint32_t l2y = rsFindHighBit(mHal.state.dimY) + 1;
7750bfc354e61e174a465893fd0dafe913f1954478Alex Sakhartchouk        uint32_t l2z = rsFindHighBit(mHal.state.dimZ) + 1;
78326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams
79246fbee66ccb17cb1f08a5420e104a709183bce1Alex Sakhartchouk        mHal.state.lodCount = rsMax(l2x, l2y);
80246fbee66ccb17cb1f08a5420e104a709183bce1Alex Sakhartchouk        mHal.state.lodCount = rsMax(mHal.state.lodCount, l2z);
81326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams    } else {
82d06653c70a67a987a2a1deb1bfb139e927fb7bd4Jason Sams        if (mHal.state.dimYuv) {
83d06653c70a67a987a2a1deb1bfb139e927fb7bd4Jason Sams            mHal.state.lodCount = 3;
84d06653c70a67a987a2a1deb1bfb139e927fb7bd4Jason Sams        } else {
85d06653c70a67a987a2a1deb1bfb139e927fb7bd4Jason Sams            mHal.state.lodCount = 1;
86d06653c70a67a987a2a1deb1bfb139e927fb7bd4Jason Sams        }
87326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams    }
88246fbee66ccb17cb1f08a5420e104a709183bce1Alex Sakhartchouk    if (mHal.state.lodCount != oldLODCount) {
89246fbee66ccb17cb1f08a5420e104a709183bce1Alex Sakhartchouk        if (oldLODCount) {
90246fbee66ccb17cb1f08a5420e104a709183bce1Alex Sakhartchouk            delete [] mHal.state.lodDimX;
91246fbee66ccb17cb1f08a5420e104a709183bce1Alex Sakhartchouk            delete [] mHal.state.lodDimY;
92246fbee66ccb17cb1f08a5420e104a709183bce1Alex Sakhartchouk            delete [] mHal.state.lodDimZ;
93417e6a486adb02b3b29ada9725286f554cc6d0d3Alex Sakhartchouk        }
94246fbee66ccb17cb1f08a5420e104a709183bce1Alex Sakhartchouk        mHal.state.lodDimX = new uint32_t[mHal.state.lodCount];
95246fbee66ccb17cb1f08a5420e104a709183bce1Alex Sakhartchouk        mHal.state.lodDimY = new uint32_t[mHal.state.lodCount];
96246fbee66ccb17cb1f08a5420e104a709183bce1Alex Sakhartchouk        mHal.state.lodDimZ = new uint32_t[mHal.state.lodCount];
97326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams    }
98326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams
9950bfc354e61e174a465893fd0dafe913f1954478Alex Sakhartchouk    uint32_t tx = mHal.state.dimX;
10050bfc354e61e174a465893fd0dafe913f1954478Alex Sakhartchouk    uint32_t ty = mHal.state.dimY;
10150bfc354e61e174a465893fd0dafe913f1954478Alex Sakhartchouk    uint32_t tz = mHal.state.dimZ;
10261656a7c6fc13421679d0a1cdf8b5b861e286892Jason Sams    mCellCount = 0;
103d06653c70a67a987a2a1deb1bfb139e927fb7bd4Jason Sams    if (!mHal.state.dimYuv) {
104d06653c70a67a987a2a1deb1bfb139e927fb7bd4Jason Sams        for (uint32_t lod=0; lod < mHal.state.lodCount; lod++) {
105d06653c70a67a987a2a1deb1bfb139e927fb7bd4Jason Sams            mHal.state.lodDimX[lod] = tx;
106d06653c70a67a987a2a1deb1bfb139e927fb7bd4Jason Sams            mHal.state.lodDimY[lod] = ty;
107d06653c70a67a987a2a1deb1bfb139e927fb7bd4Jason Sams            mHal.state.lodDimZ[lod]  = tz;
108d06653c70a67a987a2a1deb1bfb139e927fb7bd4Jason Sams            mCellCount += tx * rsMax(ty, 1u) * rsMax(tz, 1u);
109d06653c70a67a987a2a1deb1bfb139e927fb7bd4Jason Sams            if (tx > 1) tx >>= 1;
110d06653c70a67a987a2a1deb1bfb139e927fb7bd4Jason Sams            if (ty > 1) ty >>= 1;
111d06653c70a67a987a2a1deb1bfb139e927fb7bd4Jason Sams            if (tz > 1) tz >>= 1;
112d06653c70a67a987a2a1deb1bfb139e927fb7bd4Jason Sams        }
113326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams    }
114326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams
11550bfc354e61e174a465893fd0dafe913f1954478Alex Sakhartchouk    if (mHal.state.faces) {
11661656a7c6fc13421679d0a1cdf8b5b861e286892Jason Sams        mCellCount *= 6;
117326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams    }
1180b575de8ed0b628d84d256f5846500b0385979bdTim Murray#ifndef RS_SERVER
119bc0ca6ba4e31239bf77060578d0bdf1a10e04168Jason Sams    // YUV only supports basic 2d
120bc0ca6ba4e31239bf77060578d0bdf1a10e04168Jason Sams    // so we can stash the plane pointers in the mipmap levels.
121bc0ca6ba4e31239bf77060578d0bdf1a10e04168Jason Sams    if (mHal.state.dimYuv) {
12261656a7c6fc13421679d0a1cdf8b5b861e286892Jason Sams        mHal.state.lodDimX[1] = mHal.state.lodDimX[0] / 2;
12361656a7c6fc13421679d0a1cdf8b5b861e286892Jason Sams        mHal.state.lodDimY[1] = mHal.state.lodDimY[0] / 2;
12461656a7c6fc13421679d0a1cdf8b5b861e286892Jason Sams        mHal.state.lodDimX[2] = mHal.state.lodDimX[0] / 2;
12561656a7c6fc13421679d0a1cdf8b5b861e286892Jason Sams        mHal.state.lodDimY[2] = mHal.state.lodDimY[0] / 2;
126d06653c70a67a987a2a1deb1bfb139e927fb7bd4Jason Sams        mCellCount += mHal.state.lodDimX[0] * mHal.state.lodDimY[0];
12761656a7c6fc13421679d0a1cdf8b5b861e286892Jason Sams        mCellCount += mHal.state.lodDimX[1] * mHal.state.lodDimY[1];
12861656a7c6fc13421679d0a1cdf8b5b861e286892Jason Sams        mCellCount += mHal.state.lodDimX[2] * mHal.state.lodDimY[2];
12961656a7c6fc13421679d0a1cdf8b5b861e286892Jason Sams
130bc0ca6ba4e31239bf77060578d0bdf1a10e04168Jason Sams        switch(mHal.state.dimYuv) {
131bc0ca6ba4e31239bf77060578d0bdf1a10e04168Jason Sams        case HAL_PIXEL_FORMAT_YV12:
132bc0ca6ba4e31239bf77060578d0bdf1a10e04168Jason Sams            break;
133bc0ca6ba4e31239bf77060578d0bdf1a10e04168Jason Sams        case HAL_PIXEL_FORMAT_YCrCb_420_SP:  // NV21
134bc0ca6ba4e31239bf77060578d0bdf1a10e04168Jason Sams            mHal.state.lodDimX[1] = mHal.state.lodDimX[0];
135bc0ca6ba4e31239bf77060578d0bdf1a10e04168Jason Sams            break;
13661656a7c6fc13421679d0a1cdf8b5b861e286892Jason Sams#ifndef RS_COMPATIBILITY_LIB
13761656a7c6fc13421679d0a1cdf8b5b861e286892Jason Sams        case HAL_PIXEL_FORMAT_YCbCr_420_888:
13861656a7c6fc13421679d0a1cdf8b5b861e286892Jason Sams            break;
13961656a7c6fc13421679d0a1cdf8b5b861e286892Jason Sams#endif
140bc0ca6ba4e31239bf77060578d0bdf1a10e04168Jason Sams        default:
141bc0ca6ba4e31239bf77060578d0bdf1a10e04168Jason Sams            rsAssert(0);
142bc0ca6ba4e31239bf77060578d0bdf1a10e04168Jason Sams        }
143bc0ca6ba4e31239bf77060578d0bdf1a10e04168Jason Sams    }
1440b575de8ed0b628d84d256f5846500b0385979bdTim Murray#endif
14550bfc354e61e174a465893fd0dafe913f1954478Alex Sakhartchouk    mHal.state.element = mElement.get();
146326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams}
147326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams
148afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex Sakhartchoukvoid Type::dumpLOGV(const char *prefix) const {
149e12c1c591b4219e80f29c6c0e0c62c9578b75450Jason Sams    char buf[1024];
150e12c1c591b4219e80f29c6c0e0c62c9578b75450Jason Sams    ObjectBase::dumpLOGV(prefix);
15150bfc354e61e174a465893fd0dafe913f1954478Alex Sakhartchouk    ALOGV("%s   Type: x=%u y=%u z=%u mip=%i face=%i", prefix,
15250bfc354e61e174a465893fd0dafe913f1954478Alex Sakhartchouk                                                      mHal.state.dimX,
15350bfc354e61e174a465893fd0dafe913f1954478Alex Sakhartchouk                                                      mHal.state.dimY,
15450bfc354e61e174a465893fd0dafe913f1954478Alex Sakhartchouk                                                      mHal.state.dimZ,
155246fbee66ccb17cb1f08a5420e104a709183bce1Alex Sakhartchouk                                                      mHal.state.lodCount,
15650bfc354e61e174a465893fd0dafe913f1954478Alex Sakhartchouk                                                      mHal.state.faces);
15787319de2b16a185cf360827c96a42cf1fcaae744Jason Sams    snprintf(buf, sizeof(buf), "%s element: ", prefix);
158e12c1c591b4219e80f29c6c0e0c62c9578b75450Jason Sams    mElement->dumpLOGV(buf);
159e12c1c591b4219e80f29c6c0e0c62c9578b75450Jason Sams}
160e12c1c591b4219e80f29c6c0e0c62c9578b75450Jason Sams
161e3150cfb3edb028407669e4a65e087eae77e718cJason Samsvoid Type::serialize(Context *rsc, OStream *stream) const {
162fb6b614bcea88a587a7ea4530be45ff0ffa0210eAlex Sakhartchouk    // Need to identify ourselves
163fb6b614bcea88a587a7ea4530be45ff0ffa0210eAlex Sakhartchouk    stream->addU32((uint32_t)getClassId());
16448ecf6a5e85a9a832f41393ed2802385bb8b5db8Jason Sams    stream->addString(getName());
165fb6b614bcea88a587a7ea4530be45ff0ffa0210eAlex Sakhartchouk
166e3150cfb3edb028407669e4a65e087eae77e718cJason Sams    mElement->serialize(rsc, stream);
167fb6b614bcea88a587a7ea4530be45ff0ffa0210eAlex Sakhartchouk
16850bfc354e61e174a465893fd0dafe913f1954478Alex Sakhartchouk    stream->addU32(mHal.state.dimX);
16950bfc354e61e174a465893fd0dafe913f1954478Alex Sakhartchouk    stream->addU32(mHal.state.dimY);
17050bfc354e61e174a465893fd0dafe913f1954478Alex Sakhartchouk    stream->addU32(mHal.state.dimZ);
171fb6b614bcea88a587a7ea4530be45ff0ffa0210eAlex Sakhartchouk
172246fbee66ccb17cb1f08a5420e104a709183bce1Alex Sakhartchouk    stream->addU8((uint8_t)(mHal.state.lodCount ? 1 : 0));
17350bfc354e61e174a465893fd0dafe913f1954478Alex Sakhartchouk    stream->addU8((uint8_t)(mHal.state.faces ? 1 : 0));
174fb6b614bcea88a587a7ea4530be45ff0ffa0210eAlex Sakhartchouk}
175fb6b614bcea88a587a7ea4530be45ff0ffa0210eAlex Sakhartchouk
176afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex SakhartchoukType *Type::createFromStream(Context *rsc, IStream *stream) {
177fb6b614bcea88a587a7ea4530be45ff0ffa0210eAlex Sakhartchouk    // First make sure we are reading the correct object
178b825f67adb5d1e1751fe108e6dbf9c6f2555c283Alex Sakhartchouk    RsA3DClassID classID = (RsA3DClassID)stream->loadU32();
179afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex Sakhartchouk    if (classID != RS_A3D_CLASS_ID_TYPE) {
180af12ac6a08651464f8d823add667c706f993b587Steve Block        ALOGE("type loading skipped due to invalid class id\n");
181fb6b614bcea88a587a7ea4530be45ff0ffa0210eAlex Sakhartchouk        return NULL;
182fb6b614bcea88a587a7ea4530be45ff0ffa0210eAlex Sakhartchouk    }
183fb6b614bcea88a587a7ea4530be45ff0ffa0210eAlex Sakhartchouk
18448ecf6a5e85a9a832f41393ed2802385bb8b5db8Jason Sams    const char *name = stream->loadString();
185fb6b614bcea88a587a7ea4530be45ff0ffa0210eAlex Sakhartchouk
186fb6b614bcea88a587a7ea4530be45ff0ffa0210eAlex Sakhartchouk    Element *elem = Element::createFromStream(rsc, stream);
187afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex Sakhartchouk    if (!elem) {
188fb6b614bcea88a587a7ea4530be45ff0ffa0210eAlex Sakhartchouk        return NULL;
189fb6b614bcea88a587a7ea4530be45ff0ffa0210eAlex Sakhartchouk    }
190fb6b614bcea88a587a7ea4530be45ff0ffa0210eAlex Sakhartchouk
191f0c1df480304a72ce41e7d4b088319cbd7f0938aJason Sams    uint32_t x = stream->loadU32();
192f0c1df480304a72ce41e7d4b088319cbd7f0938aJason Sams    uint32_t y = stream->loadU32();
193f0c1df480304a72ce41e7d4b088319cbd7f0938aJason Sams    uint32_t z = stream->loadU32();
194f0c1df480304a72ce41e7d4b088319cbd7f0938aJason Sams    uint8_t lod = stream->loadU8();
195f0c1df480304a72ce41e7d4b088319cbd7f0938aJason Sams    uint8_t faces = stream->loadU8();
196a572aca4eb4ddb32c10baa1f529431cfefd756b8Jason Sams    Type *type = Type::getType(rsc, elem, x, y, z, lod != 0, faces !=0, 0);
197c700e649ca44d0dcff8b271e42d949ea72fe3c63Alex Sakhartchouk    elem->decUserRef();
19848ecf6a5e85a9a832f41393ed2802385bb8b5db8Jason Sams
19948ecf6a5e85a9a832f41393ed2802385bb8b5db8Jason Sams    delete [] name;
200c700e649ca44d0dcff8b271e42d949ea72fe3c63Alex Sakhartchouk    return type;
201fb6b614bcea88a587a7ea4530be45ff0ffa0210eAlex Sakhartchouk}
202fb6b614bcea88a587a7ea4530be45ff0ffa0210eAlex Sakhartchouk
203afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex Sakhartchoukbool Type::getIsNp2() const {
204ef21edcc70fc2734a3dc7995d3c3af1f90d16ef8Jason Sams    uint32_t x = getDimX();
205ef21edcc70fc2734a3dc7995d3c3af1f90d16ef8Jason Sams    uint32_t y = getDimY();
206ef21edcc70fc2734a3dc7995d3c3af1f90d16ef8Jason Sams    uint32_t z = getDimZ();
207ef21edcc70fc2734a3dc7995d3c3af1f90d16ef8Jason Sams
208ef21edcc70fc2734a3dc7995d3c3af1f90d16ef8Jason Sams    if (x && (x & (x-1))) {
209ef21edcc70fc2734a3dc7995d3c3af1f90d16ef8Jason Sams        return true;
210ef21edcc70fc2734a3dc7995d3c3af1f90d16ef8Jason Sams    }
211ef21edcc70fc2734a3dc7995d3c3af1f90d16ef8Jason Sams    if (y && (y & (y-1))) {
212ef21edcc70fc2734a3dc7995d3c3af1f90d16ef8Jason Sams        return true;
213ef21edcc70fc2734a3dc7995d3c3af1f90d16ef8Jason Sams    }
214ef21edcc70fc2734a3dc7995d3c3af1f90d16ef8Jason Sams    if (z && (z & (z-1))) {
215ef21edcc70fc2734a3dc7995d3c3af1f90d16ef8Jason Sams        return true;
216ef21edcc70fc2734a3dc7995d3c3af1f90d16ef8Jason Sams    }
217ef21edcc70fc2734a3dc7995d3c3af1f90d16ef8Jason Sams    return false;
218ef21edcc70fc2734a3dc7995d3c3af1f90d16ef8Jason Sams}
219ef21edcc70fc2734a3dc7995d3c3af1f90d16ef8Jason Sams
220c700e649ca44d0dcff8b271e42d949ea72fe3c63Alex SakhartchoukObjectBaseRef<Type> Type::getTypeRef(Context *rsc, const Element *e,
221c700e649ca44d0dcff8b271e42d949ea72fe3c63Alex Sakhartchouk                                     uint32_t dimX, uint32_t dimY, uint32_t dimZ,
222a572aca4eb4ddb32c10baa1f529431cfefd756b8Jason Sams                                     bool dimLOD, bool dimFaces, uint32_t dimYuv) {
223c700e649ca44d0dcff8b271e42d949ea72fe3c63Alex Sakhartchouk    ObjectBaseRef<Type> returnRef;
224e12c1c591b4219e80f29c6c0e0c62c9578b75450Jason Sams
22596abf819e50b59ba8cf886c13f894633eb0a24baJason Sams    TypeState * stc = &rsc->mStateType;
226f0c1df480304a72ce41e7d4b088319cbd7f0938aJason Sams
227f0c1df480304a72ce41e7d4b088319cbd7f0938aJason Sams    ObjectBase::asyncLock();
22896abf819e50b59ba8cf886c13f894633eb0a24baJason Sams    for (uint32_t ct=0; ct < stc->mTypes.size(); ct++) {
22996abf819e50b59ba8cf886c13f894633eb0a24baJason Sams        Type *t = stc->mTypes[ct];
230f0c1df480304a72ce41e7d4b088319cbd7f0938aJason Sams        if (t->getElement() != e) continue;
23196abf819e50b59ba8cf886c13f894633eb0a24baJason Sams        if (t->getDimX() != dimX) continue;
232f0c1df480304a72ce41e7d4b088319cbd7f0938aJason Sams        if (t->getDimY() != dimY) continue;
233f0c1df480304a72ce41e7d4b088319cbd7f0938aJason Sams        if (t->getDimZ() != dimZ) continue;
234f0c1df480304a72ce41e7d4b088319cbd7f0938aJason Sams        if (t->getDimLOD() != dimLOD) continue;
235f0c1df480304a72ce41e7d4b088319cbd7f0938aJason Sams        if (t->getDimFaces() != dimFaces) continue;
236a572aca4eb4ddb32c10baa1f529431cfefd756b8Jason Sams        if (t->getDimYuv() != dimYuv) continue;
237c700e649ca44d0dcff8b271e42d949ea72fe3c63Alex Sakhartchouk        returnRef.set(t);
238f0c1df480304a72ce41e7d4b088319cbd7f0938aJason Sams        ObjectBase::asyncUnlock();
239c700e649ca44d0dcff8b271e42d949ea72fe3c63Alex Sakhartchouk        return returnRef;
24096abf819e50b59ba8cf886c13f894633eb0a24baJason Sams    }
241f0c1df480304a72ce41e7d4b088319cbd7f0938aJason Sams    ObjectBase::asyncUnlock();
242f0c1df480304a72ce41e7d4b088319cbd7f0938aJason Sams
243665eafe0287f0d12c3f049a4f73b95db6e38b3f0Tim Murray    // Type objects must use allocator specified by the driver
244665eafe0287f0d12c3f049a4f73b95db6e38b3f0Tim Murray    void* allocMem = rsc->mHal.funcs.allocRuntimeMem(sizeof(Type), 0);
245665eafe0287f0d12c3f049a4f73b95db6e38b3f0Tim Murray    if (!allocMem) {
246665eafe0287f0d12c3f049a4f73b95db6e38b3f0Tim Murray        rsc->setError(RS_ERROR_FATAL_DRIVER, "Couldn't allocate memory for Type");
247665eafe0287f0d12c3f049a4f73b95db6e38b3f0Tim Murray        return NULL;
248665eafe0287f0d12c3f049a4f73b95db6e38b3f0Tim Murray    }
24996abf819e50b59ba8cf886c13f894633eb0a24baJason Sams
250665eafe0287f0d12c3f049a4f73b95db6e38b3f0Tim Murray    Type *nt = new (allocMem) Type(rsc);
251e3af53b643677c40d228ffd3624cf259f4dc68edTim Murray
252e3af53b643677c40d228ffd3624cf259f4dc68edTim Murray#ifdef RS_FIND_OFFSETS
253e3af53b643677c40d228ffd3624cf259f4dc68edTim Murray    ALOGE("pointer for type: %p", nt);
254e3af53b643677c40d228ffd3624cf259f4dc68edTim Murray    ALOGE("pointer for type.drv: %p", &nt->mHal.drv);
255e3af53b643677c40d228ffd3624cf259f4dc68edTim Murray#endif
256e3af53b643677c40d228ffd3624cf259f4dc68edTim Murray
257246fbee66ccb17cb1f08a5420e104a709183bce1Alex Sakhartchouk    nt->mDimLOD = dimLOD;
258c700e649ca44d0dcff8b271e42d949ea72fe3c63Alex Sakhartchouk    returnRef.set(nt);
259f0c1df480304a72ce41e7d4b088319cbd7f0938aJason Sams    nt->mElement.set(e);
26050bfc354e61e174a465893fd0dafe913f1954478Alex Sakhartchouk    nt->mHal.state.dimX = dimX;
26150bfc354e61e174a465893fd0dafe913f1954478Alex Sakhartchouk    nt->mHal.state.dimY = dimY;
26250bfc354e61e174a465893fd0dafe913f1954478Alex Sakhartchouk    nt->mHal.state.dimZ = dimZ;
26350bfc354e61e174a465893fd0dafe913f1954478Alex Sakhartchouk    nt->mHal.state.faces = dimFaces;
264a572aca4eb4ddb32c10baa1f529431cfefd756b8Jason Sams    nt->mHal.state.dimYuv = dimYuv;
26596abf819e50b59ba8cf886c13f894633eb0a24baJason Sams    nt->compute();
266f0c1df480304a72ce41e7d4b088319cbd7f0938aJason Sams
267f0c1df480304a72ce41e7d4b088319cbd7f0938aJason Sams    ObjectBase::asyncLock();
268f0c1df480304a72ce41e7d4b088319cbd7f0938aJason Sams    stc->mTypes.push(nt);
269f0c1df480304a72ce41e7d4b088319cbd7f0938aJason Sams    ObjectBase::asyncUnlock();
270f0c1df480304a72ce41e7d4b088319cbd7f0938aJason Sams
271c700e649ca44d0dcff8b271e42d949ea72fe3c63Alex Sakhartchouk    return returnRef;
27296abf819e50b59ba8cf886c13f894633eb0a24baJason Sams}
27396abf819e50b59ba8cf886c13f894633eb0a24baJason Sams
274c700e649ca44d0dcff8b271e42d949ea72fe3c63Alex SakhartchoukObjectBaseRef<Type> Type::cloneAndResize1D(Context *rsc, uint32_t dimX) const {
275c700e649ca44d0dcff8b271e42d949ea72fe3c63Alex Sakhartchouk    return getTypeRef(rsc, mElement.get(), dimX,
276a572aca4eb4ddb32c10baa1f529431cfefd756b8Jason Sams                      getDimY(), getDimZ(), getDimLOD(), getDimFaces(), getDimYuv());
277f0c1df480304a72ce41e7d4b088319cbd7f0938aJason Sams}
27896abf819e50b59ba8cf886c13f894633eb0a24baJason Sams
279c700e649ca44d0dcff8b271e42d949ea72fe3c63Alex SakhartchoukObjectBaseRef<Type> Type::cloneAndResize2D(Context *rsc,
280afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex Sakhartchouk                              uint32_t dimX,
281afb743aca56c18beb7ab924e75cb6e070ef3e55aAlex Sakhartchouk                              uint32_t dimY) const {
282c700e649ca44d0dcff8b271e42d949ea72fe3c63Alex Sakhartchouk    return getTypeRef(rsc, mElement.get(), dimX, dimY,
283a572aca4eb4ddb32c10baa1f529431cfefd756b8Jason Sams                      getDimZ(), getDimLOD(), getDimFaces(), getDimYuv());
28496abf819e50b59ba8cf886c13f894633eb0a24baJason Sams}
28596abf819e50b59ba8cf886c13f894633eb0a24baJason Sams
28696abf819e50b59ba8cf886c13f894633eb0a24baJason Sams
2879f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hinesvoid Type::incRefs(const void *ptr, size_t ct, size_t startOff) const {
2889f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines    const uint8_t *p = static_cast<const uint8_t *>(ptr);
2899f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines    const Element *e = mHal.state.element;
2909f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines    uint32_t stride = e->getSizeBytes();
2919f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines
2929f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines    p += stride * startOff;
2939f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines    while (ct > 0) {
2949f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines        e->incRefs(p);
2959f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines        ct--;
2969f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines        p += stride;
2979f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines    }
2989f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines}
2999f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines
3009f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines
3019f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hinesvoid Type::decRefs(const void *ptr, size_t ct, size_t startOff) const {
3029f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines    if (!mHal.state.element->getHasReferences()) {
3039f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines        return;
3049f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines    }
3059f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines    const uint8_t *p = static_cast<const uint8_t *>(ptr);
3069f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines    const Element *e = mHal.state.element;
3079f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines    uint32_t stride = e->getSizeBytes();
3089f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines
3099f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines    p += stride * startOff;
3109f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines    while (ct > 0) {
3119f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines        e->decRefs(p);
3129f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines        ct--;
3139f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines        p += stride;
3149f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines    }
3159f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines}
3169f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines
317a36c50a6ab87f4c9049318d4c6c8ec7b0a1e6e12Jason Samsvoid Type::callUpdateCacheObject(const Context *rsc, void *dstObj) const {
318a36c50a6ab87f4c9049318d4c6c8ec7b0a1e6e12Jason Sams    if (rsc->mHal.funcs.type.updateCachedObject != NULL) {
319a36c50a6ab87f4c9049318d4c6c8ec7b0a1e6e12Jason Sams        rsc->mHal.funcs.type.updateCachedObject(rsc, this, (rs_type *)dstObj);
320a36c50a6ab87f4c9049318d4c6c8ec7b0a1e6e12Jason Sams    } else {
321a36c50a6ab87f4c9049318d4c6c8ec7b0a1e6e12Jason Sams        *((const void **)dstObj) = this;
322a36c50a6ab87f4c9049318d4c6c8ec7b0a1e6e12Jason Sams    }
323a36c50a6ab87f4c9049318d4c6c8ec7b0a1e6e12Jason Sams}
3249f70a4e63825afe9f786483722e1669b3625f5e9Stephen Hines
325326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams//////////////////////////////////////////////////
326e5ffb879ae535a899a486285a23bea05e912480fJason Sams//
327326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsnamespace android {
328326e0ddf89e8df2837752fbfd7a014814b32082cJason Samsnamespace renderscript {
329326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams
330c975cf4a71b63ccbd20f1f3b3341c5f2e6025b45Jason SamsRsType rsi_TypeCreate(Context *rsc, RsElement _e, uint32_t dimX,
3318f615d682f9e7e2cd4de2e4478e0e76fc359922cStephen Hines                     uint32_t dimY, uint32_t dimZ, bool mipmaps, bool faces, uint32_t yuv) {
3322353ae303868d04e3a26002b2f2dc456c15e8170Jason Sams    Element *e = static_cast<Element *>(_e);
3332353ae303868d04e3a26002b2f2dc456c15e8170Jason Sams
3348f615d682f9e7e2cd4de2e4478e0e76fc359922cStephen Hines    return Type::getType(rsc, e, dimX, dimY, dimZ, mipmaps, faces, yuv);
335326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams}
336326e0ddf89e8df2837752fbfd7a014814b32082cJason Sams
337c975cf4a71b63ccbd20f1f3b3341c5f2e6025b45Jason Sams}
338c975cf4a71b63ccbd20f1f3b3341c5f2e6025b45Jason Sams}
339c975cf4a71b63ccbd20f1f3b3341c5f2e6025b45Jason Sams
340c2ce707a3d1a8eae79bcf1c749afc6d6e7969ad9Tim Murrayextern "C" void rsaTypeGetNativeData(RsContext con, RsType type, uintptr_t *typeData, uint32_t typeDataSize) {
341dc763f345db3e796efc28dc4b4d8edffda5a803eAlex Sakhartchouk    rsAssert(typeDataSize == 6);
34250bfc354e61e174a465893fd0dafe913f1954478Alex Sakhartchouk    // Pack the data in the follofing way mHal.state.dimX; mHal.state.dimY; mHal.state.dimZ;
343246fbee66ccb17cb1f08a5420e104a709183bce1Alex Sakhartchouk    // mHal.state.lodCount; mHal.state.faces; mElement; into typeData
344dc763f345db3e796efc28dc4b4d8edffda5a803eAlex Sakhartchouk    Type *t = static_cast<Type *>(type);
345dc763f345db3e796efc28dc4b4d8edffda5a803eAlex Sakhartchouk
346dc763f345db3e796efc28dc4b4d8edffda5a803eAlex Sakhartchouk    (*typeData++) = t->getDimX();
347dc763f345db3e796efc28dc4b4d8edffda5a803eAlex Sakhartchouk    (*typeData++) = t->getDimY();
348dc763f345db3e796efc28dc4b4d8edffda5a803eAlex Sakhartchouk    (*typeData++) = t->getDimZ();
349246fbee66ccb17cb1f08a5420e104a709183bce1Alex Sakhartchouk    (*typeData++) = t->getDimLOD() ? 1 : 0;
350dc763f345db3e796efc28dc4b4d8edffda5a803eAlex Sakhartchouk    (*typeData++) = t->getDimFaces() ? 1 : 0;
351099bc262f862cdeb547cf8a78fe9e0e92560f437Tim Murray    (*typeData++) = (uintptr_t)t->getElement();
352dc763f345db3e796efc28dc4b4d8edffda5a803eAlex Sakhartchouk    t->getElement()->incUserRef();
353dc763f345db3e796efc28dc4b4d8edffda5a803eAlex Sakhartchouk}
354