| /* |
| * 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."); |
| } |
| |
| } |
| |
| |