Home | History | Annotate | Download | only in renderscript
      1 /*
      2  * Copyright (C) 2008 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 package android.renderscript;
     18 
     19 /**
     20  * Only intended for use by generated reflected code.
     21  *
     22  **/
     23 public class AllocationAdapter extends Allocation {
     24     Type mWindow;
     25 
     26     AllocationAdapter(long id, RenderScript rs, Allocation alloc, Type t) {
     27         super(id, rs, alloc.mType, alloc.mUsage);
     28         mAdaptedAllocation = alloc;
     29         mWindow = t;
     30     }
     31 
     32     /*
     33     long getID(RenderScript rs) {
     34         throw new RSInvalidStateException(
     35             "This operation is not supported with adapters at this time.");
     36     }
     37     */
     38 
     39     void initLOD(int lod) {
     40         if (lod < 0) {
     41             throw new RSIllegalArgumentException("Attempting to set negative lod (" + lod + ").");
     42         }
     43 
     44         int tx = mAdaptedAllocation.mType.getX();
     45         int ty = mAdaptedAllocation.mType.getY();
     46         int tz = mAdaptedAllocation.mType.getZ();
     47 
     48         for (int ct=0; ct < lod; ct++) {
     49             if ((tx==1) && (ty == 1) && (tz == 1)) {
     50                 throw new RSIllegalArgumentException("Attempting to set lod (" + lod + ") out of range.");
     51             }
     52 
     53             if (tx > 1) tx >>= 1;
     54             if (ty > 1) ty >>= 1;
     55             if (tz > 1) tz >>= 1;
     56         }
     57 
     58         mCurrentDimX = tx;
     59         mCurrentDimY = ty;
     60         mCurrentDimZ = tz;
     61         mCurrentCount = mCurrentDimX;
     62         if (mCurrentDimY > 1) {
     63             mCurrentCount *= mCurrentDimY;
     64         }
     65         if (mCurrentDimZ > 1) {
     66             mCurrentCount *= mCurrentDimZ;
     67         }
     68         mSelectedY = 0;
     69         mSelectedZ = 0;
     70     }
     71 
     72     private void updateOffsets() {
     73         int a1 = 0, a2 = 0, a3 = 0, a4 = 0;
     74 
     75         if (mSelectedArray != null) {
     76             if (mSelectedArray.length > 0) {
     77                 a1 = mSelectedArray[0];
     78             }
     79             if (mSelectedArray.length > 1) {
     80                 a2 = mSelectedArray[2];
     81             }
     82             if (mSelectedArray.length > 2) {
     83                 a3 = mSelectedArray[2];
     84             }
     85             if (mSelectedArray.length > 3) {
     86                 a4 = mSelectedArray[3];
     87             }
     88         }
     89         mRS.nAllocationAdapterOffset(getID(mRS), mSelectedX, mSelectedY, mSelectedZ,
     90                                      mSelectedLOD, mSelectedFace.mID, a1, a2, a3, a4);
     91 
     92     }
     93 
     94     /**
     95      * Set the active LOD.  The LOD must be within the range for the
     96      * type being adapted.  The base allocation must have mipmaps.
     97      *
     98      * Because this changes the dimensions of the adapter the
     99      * current Y and Z will be reset.
    100      *
    101      * @param lod The LOD to make active.
    102      */
    103     public void setLOD(int lod) {
    104         if (!mAdaptedAllocation.getType().hasMipmaps()) {
    105             throw new RSInvalidStateException("Cannot set LOD when the allocation type does not include mipmaps.");
    106         }
    107         if (mWindow.hasMipmaps()) {
    108             throw new RSInvalidStateException("Cannot set LOD when the adapter includes mipmaps.");
    109         }
    110 
    111         initLOD(lod);
    112         mSelectedLOD = lod;
    113         updateOffsets();
    114     }
    115 
    116     /**
    117      * Set the active Face.  The base allocation must be of a type
    118      * that includes faces.
    119      *
    120      * @param cf The face to make active.
    121      */
    122     public void setFace(Type.CubemapFace cf) {
    123         if (!mAdaptedAllocation.getType().hasFaces()) {
    124             throw new RSInvalidStateException("Cannot set Face when the allocation type does not include faces.");
    125         }
    126         if (mWindow.hasFaces()) {
    127             throw new RSInvalidStateException("Cannot set face when the adapter includes faces.");
    128         }
    129         if (cf == null) {
    130             throw new RSIllegalArgumentException("Cannot set null face.");
    131         }
    132 
    133         mSelectedFace = cf;
    134         updateOffsets();
    135     }
    136 
    137 
    138     /**
    139      *
    140      * Set the active X.  The x value must be within the range for
    141      * the allocation being adapted.
    142      *
    143      * @param x The x to make active.
    144      */
    145     public void setX(int x) {
    146         if (mAdaptedAllocation.getType().getX() <= x) {
    147             throw new RSInvalidStateException("Cannot set X greater than dimension of allocation.");
    148         }
    149         if (mWindow.getX() == mAdaptedAllocation.getType().getX()) {
    150             throw new RSInvalidStateException("Cannot set X when the adapter includes X.");
    151         }
    152         if ((mWindow.getX() + x) >= mAdaptedAllocation.getType().getX()) {
    153             throw new RSInvalidStateException("Cannot set (X + window) which would be larger than dimension of allocation.");
    154         }
    155 
    156         mSelectedX = x;
    157         updateOffsets();
    158     }
    159 
    160     /**
    161      * Set the active Y.  The y value must be within the range for
    162      * the allocation being adapted.  The base allocation must
    163      * contain the Y dimension.
    164      *
    165      * @param y The y to make active.
    166      */
    167     public void setY(int y) {
    168         if (mAdaptedAllocation.getType().getY() == 0) {
    169             throw new RSInvalidStateException("Cannot set Y when the allocation type does not include Y dim.");
    170         }
    171         if (mAdaptedAllocation.getType().getY() <= y) {
    172             throw new RSInvalidStateException("Cannot set Y greater than dimension of allocation.");
    173         }
    174         if (mWindow.getY() == mAdaptedAllocation.getType().getY()) {
    175             throw new RSInvalidStateException("Cannot set Y when the adapter includes Y.");
    176         }
    177         if ((mWindow.getY() + y) >= mAdaptedAllocation.getType().getY()) {
    178             throw new RSInvalidStateException("Cannot set (Y + window) which would be larger than dimension of allocation.");
    179         }
    180 
    181         mSelectedY = y;
    182         updateOffsets();
    183     }
    184 
    185     /**
    186      * Set the active Z.  The z value must be within the range for
    187      * the allocation being adapted.  The base allocation must
    188      * contain the Z dimension.
    189      *
    190      * @param z The z to make active.
    191      */
    192     public void setZ(int z) {
    193         if (mAdaptedAllocation.getType().getZ() == 0) {
    194             throw new RSInvalidStateException("Cannot set Z when the allocation type does not include Z dim.");
    195         }
    196         if (mAdaptedAllocation.getType().getZ() <= z) {
    197             throw new RSInvalidStateException("Cannot set Z greater than dimension of allocation.");
    198         }
    199         if (mWindow.getZ() == mAdaptedAllocation.getType().getZ()) {
    200             throw new RSInvalidStateException("Cannot set Z when the adapter includes Z.");
    201         }
    202         if ((mWindow.getZ() + z) >= mAdaptedAllocation.getType().getZ()) {
    203             throw new RSInvalidStateException("Cannot set (Z + window) which would be larger than dimension of allocation.");
    204         }
    205 
    206         mSelectedZ = z;
    207         updateOffsets();
    208     }
    209 
    210     /**
    211      * @hide
    212      */
    213     public void setArray(int arrayNum, int arrayVal) {
    214         if (mAdaptedAllocation.getType().getArray(arrayNum) == 0) {
    215             throw new RSInvalidStateException("Cannot set arrayNum when the allocation type does not include arrayNum dim.");
    216         }
    217         if (mAdaptedAllocation.getType().getArray(arrayNum) <= arrayVal) {
    218             throw new RSInvalidStateException("Cannot set arrayNum greater than dimension of allocation.");
    219         }
    220         if (mWindow.getArray(arrayNum) == mAdaptedAllocation.getType().getArray(arrayNum)) {
    221             throw new RSInvalidStateException("Cannot set arrayNum when the adapter includes arrayNum.");
    222         }
    223         if ((mWindow.getArray(arrayNum) + arrayVal) >= mAdaptedAllocation.getType().getArray(arrayNum)) {
    224             throw new RSInvalidStateException("Cannot set (arrayNum + window) which would be larger than dimension of allocation.");
    225         }
    226 
    227         mSelectedArray[arrayNum] = arrayVal;
    228         updateOffsets();
    229     }
    230 
    231     static public AllocationAdapter create1D(RenderScript rs, Allocation a) {
    232         rs.validate();
    233         Type t = Type.createX(rs, a.getElement(), a.getType().getX());
    234         return createTyped(rs, a, t);
    235     }
    236 
    237 
    238     static public AllocationAdapter create2D(RenderScript rs, Allocation a) {
    239         rs.validate();
    240         Type t = Type.createXY(rs, a.getElement(), a.getType().getX(), a.getType().getY());
    241         return createTyped(rs, a, t);
    242     }
    243 
    244     /**
    245      *
    246      *
    247      * Create an arbitrary window into the base allocation.
    248      * The type describes the shape of the window.
    249      *
    250      * Any dimensions present in the type must be equal or smaller
    251      * to the dimensions in the source allocation.  A dimension
    252      * present in the allocation that is not present in the type
    253      * will be constrained away with the selectors.
    254      *
    255      * If a dimension is present in both the type and allocation, one of
    256      * two things will happen.
    257      *
    258      * If the type is smaller than the allocation, a window will be
    259      * created, the selected value in the adapter for that dimension
    260      * will act as the base address, and the type will describe the
    261      * size of the view starting at that point.
    262      *
    263      * If the type and allocation dimension are of the same size,
    264      * then setting the selector for the dimension will be an error.
    265      */
    266     static public AllocationAdapter createTyped(RenderScript rs, Allocation a, Type t) {
    267         rs.validate();
    268 
    269         if (a.mAdaptedAllocation != null) {
    270             throw new RSInvalidStateException("Adapters cannot be nested.");
    271         }
    272 
    273         if (!a.getType().getElement().equals(t.getElement())) {
    274             throw new RSInvalidStateException("Element must match Allocation type.");
    275         }
    276 
    277         if (t.hasFaces() || t.hasMipmaps()) {
    278             throw new RSInvalidStateException("Adapters do not support window types with Mipmaps or Faces.");
    279         }
    280 
    281         Type at = a.getType();
    282         if ((t.getX() > at.getX()) ||
    283             (t.getY() > at.getY()) ||
    284             (t.getZ() > at.getZ()) ||
    285             (t.getArrayCount() > at.getArrayCount())) {
    286 
    287             throw new RSInvalidStateException("Type cannot have dimension larger than the source allocation.");
    288         }
    289 
    290         if (t.getArrayCount() > 0) {
    291             for (int i = 0; i < t.getArray(i); i++) {
    292                 if (t.getArray(i) > at.getArray(i)) {
    293                     throw new RSInvalidStateException("Type cannot have dimension larger than the source allocation.");
    294                 }
    295             }
    296         }
    297 
    298         // Create the object
    299         long id = rs.nAllocationAdapterCreate(a.getID(rs), t.getID(rs));
    300         if (id == 0) {
    301             throw new RSRuntimeException("AllocationAdapter creation failed.");
    302         }
    303         return new AllocationAdapter(id, rs, a, t);
    304     }
    305 
    306     /**
    307      * Override the Allocation resize.  Resizing adapters is not
    308      * allowed and will throw a RSInvalidStateException.
    309      *
    310      * @param dimX ignored.
    311      */
    312     public synchronized void resize(int dimX) {
    313         throw new RSInvalidStateException("Resize not allowed for Adapters.");
    314     }
    315 
    316 }
    317 
    318 
    319