/* * Copyright (C) 2008 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ package android.renderscript; /** * Only intended for use by generated reflected code. * **/ public class AllocationAdapter extends Allocation { Type mWindow; AllocationAdapter(long id, RenderScript rs, Allocation alloc, Type t) { super(id, rs, alloc.mType, alloc.mUsage); mAdaptedAllocation = alloc; mWindow = t; } /* long getID(RenderScript rs) { throw new RSInvalidStateException( "This operation is not supported with adapters at this time."); } */ void initLOD(int lod) { if (lod < 0) { throw new RSIllegalArgumentException("Attempting to set negative lod (" + lod + ")."); } int tx = mAdaptedAllocation.mType.getX(); int ty = mAdaptedAllocation.mType.getY(); int tz = mAdaptedAllocation.mType.getZ(); for (int ct=0; ct < lod; ct++) { if ((tx==1) && (ty == 1) && (tz == 1)) { throw new RSIllegalArgumentException("Attempting to set lod (" + lod + ") out of range."); } if (tx > 1) tx >>= 1; if (ty > 1) ty >>= 1; if (tz > 1) tz >>= 1; } mCurrentDimX = tx; mCurrentDimY = ty; mCurrentDimZ = tz; mCurrentCount = mCurrentDimX; if (mCurrentDimY > 1) { mCurrentCount *= mCurrentDimY; } if (mCurrentDimZ > 1) { mCurrentCount *= mCurrentDimZ; } mSelectedY = 0; mSelectedZ = 0; } private void updateOffsets() { int a1 = 0, a2 = 0, a3 = 0, a4 = 0; if (mSelectedArray != null) { if (mSelectedArray.length > 0) { a1 = mSelectedArray[0]; } if (mSelectedArray.length > 1) { a2 = mSelectedArray[2]; } if (mSelectedArray.length > 2) { a3 = mSelectedArray[2]; } if (mSelectedArray.length > 3) { a4 = mSelectedArray[3]; } } mRS.nAllocationAdapterOffset(getID(mRS), mSelectedX, mSelectedY, mSelectedZ, mSelectedLOD, mSelectedFace.mID, a1, a2, a3, a4); } /** * Set the active LOD. The LOD must be within the range for the * type being adapted. The base allocation must have mipmaps. * * Because this changes the dimensions of the adapter the * current Y and Z will be reset. * * @param lod The LOD to make active. */ public void setLOD(int lod) { if (!mAdaptedAllocation.getType().hasMipmaps()) { throw new RSInvalidStateException("Cannot set LOD when the allocation type does not include mipmaps."); } if (mWindow.hasMipmaps()) { throw new RSInvalidStateException("Cannot set LOD when the adapter includes mipmaps."); } initLOD(lod); mSelectedLOD = lod; updateOffsets(); } /** * Set the active Face. The base allocation must be of a type * that includes faces. * * @param cf The face to make active. */ public void setFace(Type.CubemapFace cf) { if (!mAdaptedAllocation.getType().hasFaces()) { throw new RSInvalidStateException("Cannot set Face when the allocation type does not include faces."); } if (mWindow.hasFaces()) { throw new RSInvalidStateException("Cannot set face when the adapter includes faces."); } if (cf == null) { throw new RSIllegalArgumentException("Cannot set null face."); } mSelectedFace = cf; updateOffsets(); } /** * * Set the active X. The x value must be within the range for * the allocation being adapted. * * @param x The x to make active. */ public void setX(int x) { if (mAdaptedAllocation.getType().getX() <= x) { throw new RSInvalidStateException("Cannot set X greater than dimension of allocation."); } if (mWindow.getX() == mAdaptedAllocation.getType().getX()) { throw new RSInvalidStateException("Cannot set X when the adapter includes X."); } if ((mWindow.getX() + x) >= mAdaptedAllocation.getType().getX()) { throw new RSInvalidStateException("Cannot set (X + window) which would be larger than dimension of allocation."); } mSelectedX = x; updateOffsets(); } /** * Set the active Y. The y value must be within the range for * the allocation being adapted. The base allocation must * contain the Y dimension. * * @param y The y to make active. */ public void setY(int y) { if (mAdaptedAllocation.getType().getY() == 0) { throw new RSInvalidStateException("Cannot set Y when the allocation type does not include Y dim."); } if (mAdaptedAllocation.getType().getY() <= y) { throw new RSInvalidStateException("Cannot set Y greater than dimension of allocation."); } if (mWindow.getY() == mAdaptedAllocation.getType().getY()) { throw new RSInvalidStateException("Cannot set Y when the adapter includes Y."); } if ((mWindow.getY() + y) >= mAdaptedAllocation.getType().getY()) { throw new RSInvalidStateException("Cannot set (Y + window) which would be larger than dimension of allocation."); } mSelectedY = y; updateOffsets(); } /** * Set the active Z. The z value must be within the range for * the allocation being adapted. The base allocation must * contain the Z dimension. * * @param z The z to make active. */ public void setZ(int z) { if (mAdaptedAllocation.getType().getZ() == 0) { throw new RSInvalidStateException("Cannot set Z when the allocation type does not include Z dim."); } if (mAdaptedAllocation.getType().getZ() <= z) { throw new RSInvalidStateException("Cannot set Z greater than dimension of allocation."); } if (mWindow.getZ() == mAdaptedAllocation.getType().getZ()) { throw new RSInvalidStateException("Cannot set Z when the adapter includes Z."); } if ((mWindow.getZ() + z) >= mAdaptedAllocation.getType().getZ()) { throw new RSInvalidStateException("Cannot set (Z + window) which would be larger than dimension of allocation."); } mSelectedZ = z; updateOffsets(); } /** * @hide */ public void setArray(int arrayNum, int arrayVal) { if (mAdaptedAllocation.getType().getArray(arrayNum) == 0) { throw new RSInvalidStateException("Cannot set arrayNum when the allocation type does not include arrayNum dim."); } if (mAdaptedAllocation.getType().getArray(arrayNum) <= arrayVal) { throw new RSInvalidStateException("Cannot set arrayNum greater than dimension of allocation."); } if (mWindow.getArray(arrayNum) == mAdaptedAllocation.getType().getArray(arrayNum)) { throw new RSInvalidStateException("Cannot set arrayNum when the adapter includes arrayNum."); } if ((mWindow.getArray(arrayNum) + arrayVal) >= mAdaptedAllocation.getType().getArray(arrayNum)) { throw new RSInvalidStateException("Cannot set (arrayNum + window) which would be larger than dimension of allocation."); } mSelectedArray[arrayNum] = arrayVal; updateOffsets(); } static public AllocationAdapter create1D(RenderScript rs, Allocation a) { rs.validate(); Type t = Type.createX(rs, a.getElement(), a.getType().getX()); return createTyped(rs, a, t); } static public AllocationAdapter create2D(RenderScript rs, Allocation a) { rs.validate(); Type t = Type.createXY(rs, a.getElement(), a.getType().getX(), a.getType().getY()); return createTyped(rs, a, t); } /** * * * Create an arbitrary window into the base allocation. * The type describes the shape of the window. * * Any dimensions present in the type must be equal or smaller * to the dimensions in the source allocation. A dimension * present in the allocation that is not present in the type * will be constrained away with the selectors. * * If a dimension is present in both the type and allocation, one of * two things will happen. * * If the type is smaller than the allocation, a window will be * created, the selected value in the adapter for that dimension * will act as the base address, and the type will describe the * size of the view starting at that point. * * If the type and allocation dimension are of the same size, * then setting the selector for the dimension will be an error. */ static public AllocationAdapter createTyped(RenderScript rs, Allocation a, Type t) { rs.validate(); if (a.mAdaptedAllocation != null) { throw new RSInvalidStateException("Adapters cannot be nested."); } if (!a.getType().getElement().equals(t.getElement())) { throw new RSInvalidStateException("Element must match Allocation type."); } if (t.hasFaces() || t.hasMipmaps()) { throw new RSInvalidStateException("Adapters do not support window types with Mipmaps or Faces."); } Type at = a.getType(); if ((t.getX() > at.getX()) || (t.getY() > at.getY()) || (t.getZ() > at.getZ()) || (t.getArrayCount() > at.getArrayCount())) { throw new RSInvalidStateException("Type cannot have dimension larger than the source allocation."); } if (t.getArrayCount() > 0) { for (int i = 0; i < t.getArray(i); i++) { if (t.getArray(i) > at.getArray(i)) { throw new RSInvalidStateException("Type cannot have dimension larger than the source allocation."); } } } // Create the object long id = rs.nAllocationAdapterCreate(a.getID(rs), t.getID(rs)); if (id == 0) { throw new RSRuntimeException("AllocationAdapter creation failed."); } return new AllocationAdapter(id, rs, a, t); } /** * Override the Allocation resize. Resizing adapters is not * allowed and will throw a RSInvalidStateException. * * @param dimX ignored. */ public synchronized void resize(int dimX) { throw new RSInvalidStateException("Resize not allowed for Adapters."); } }