1 /*
2  * Copyright (C) 2010 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.nfc.tech;
18 
19 import android.nfc.ErrorCodes;
20 import android.nfc.Tag;
21 import android.nfc.TagLostException;
22 import android.os.RemoteException;
23 import android.util.Log;
24 
25 import java.io.IOException;
26 import java.nio.ByteBuffer;
27 import java.nio.ByteOrder;
28 
29 /**
30  * Provides access to MIFARE Classic properties and I/O operations on a {@link Tag}.
31  *
32  * <p>Acquire a {@link MifareClassic} object using {@link #get}.
33  *
34  * <p>MIFARE Classic is also known as MIFARE Standard.
35  * <p>MIFARE Classic tags are divided into sectors, and each sector is sub-divided into
36  * blocks. Block size is always 16 bytes ({@link #BLOCK_SIZE}. Sector size varies.
37  * <ul>
38  * <li>MIFARE Classic Mini are 320 bytes ({@link #SIZE_MINI}), with 5 sectors each of 4 blocks.
39  * <li>MIFARE Classic 1k are 1024 bytes ({@link #SIZE_1K}), with 16 sectors each of 4 blocks.
40  * <li>MIFARE Classic 2k are 2048 bytes ({@link #SIZE_2K}), with 32 sectors each of 4 blocks.
41  * <li>MIFARE Classic 4k are 4096 bytes ({@link #SIZE_4K}). The first 32 sectors contain 4 blocks
42  * and the last 8 sectors contain 16 blocks.
43  * </ul>
44  *
45  * <p>MIFARE Classic tags require authentication on a per-sector basis before any
46  * other I/O operations on that sector can be performed. There are two keys per sector,
47  * and ACL bits determine what I/O operations are allowed on that sector after
48  * authenticating with a key. {@see #authenticateSectorWithKeyA} and
49  * {@see #authenticateSectorWithKeyB}.
50  *
51  * <p>Three well-known authentication keys are defined in this class:
52  * {@link #KEY_DEFAULT}, {@link #KEY_MIFARE_APPLICATION_DIRECTORY},
53  * {@link #KEY_NFC_FORUM}.
54  * <ul>
55  * <li>{@link #KEY_DEFAULT} is the default factory key for MIFARE Classic.
56  * <li>{@link #KEY_MIFARE_APPLICATION_DIRECTORY} is the well-known key for
57  * MIFARE Classic cards that have been formatted according to the
58  * MIFARE Application Directory (MAD) specification.
59  * <li>{@link #KEY_NFC_FORUM} is the well-known key for MIFARE Classic cards that
60  * have been formatted according to the NXP specification for NDEF on MIFARE Classic.
61  *
62  * <p>Implementation of this class on a Android NFC device is optional.
63  * If it is not implemented, then
64  * {@link MifareClassic} will never be enumerated in {@link Tag#getTechList}.
65  * If it is enumerated, then all {@link MifareClassic} I/O operations will be supported,
66  * and {@link Ndef#MIFARE_CLASSIC} NDEF tags will also be supported. In either case,
67  * {@link NfcA} will also be enumerated on the tag, because all MIFARE Classic tags are also
68  * {@link NfcA}.
69  *
70  * <p class="note"><strong>Note:</strong> Methods that perform I/O operations
71  * require the {@link android.Manifest.permission#NFC} permission.
72  */
73 public final class MifareClassic extends BasicTagTechnology {
74     private static final String TAG = "NFC";
75 
76     /**
77      * The default factory key.
78      */
79     public static final byte[] KEY_DEFAULT =
80             {(byte)0xFF,(byte)0xFF,(byte)0xFF,(byte)0xFF,(byte)0xFF,(byte)0xFF};
81     /**
82      * The well-known key for tags formatted according to the
83      * MIFARE Application Directory (MAD) specification.
84      */
85     public static final byte[] KEY_MIFARE_APPLICATION_DIRECTORY =
86             {(byte)0xA0,(byte)0xA1,(byte)0xA2,(byte)0xA3,(byte)0xA4,(byte)0xA5};
87     /**
88      * The well-known key for tags formatted according to the
89      * NDEF on MIFARE Classic specification.
90      */
91     public static final byte[] KEY_NFC_FORUM =
92             {(byte)0xD3,(byte)0xF7,(byte)0xD3,(byte)0xF7,(byte)0xD3,(byte)0xF7};
93 
94     /** A MIFARE Classic compatible card of unknown type */
95     public static final int TYPE_UNKNOWN = -1;
96     /** A MIFARE Classic tag */
97     public static final int TYPE_CLASSIC = 0;
98     /** A MIFARE Plus tag */
99     public static final int TYPE_PLUS = 1;
100     /** A MIFARE Pro tag */
101     public static final int TYPE_PRO = 2;
102 
103     /** Tag contains 16 sectors, each with 4 blocks. */
104     public static final int SIZE_1K = 1024;
105     /** Tag contains 32 sectors, each with 4 blocks. */
106     public static final int SIZE_2K = 2048;
107     /**
108      * Tag contains 40 sectors. The first 32 sectors contain 4 blocks and the last 8 sectors
109      * contain 16 blocks.
110      */
111     public static final int SIZE_4K = 4096;
112     /** Tag contains 5 sectors, each with 4 blocks. */
113     public static final int SIZE_MINI = 320;
114 
115     /** Size of a MIFARE Classic block (in bytes) */
116     public static final int BLOCK_SIZE = 16;
117 
118     private static final int MAX_BLOCK_COUNT = 256;
119     private static final int MAX_SECTOR_COUNT = 40;
120 
121     private boolean mIsEmulated;
122     private int mType;
123     private int mSize;
124 
125     /**
126      * Get an instance of {@link MifareClassic} for the given tag.
127      * <p>Does not cause any RF activity and does not block.
128      * <p>Returns null if {@link MifareClassic} was not enumerated in {@link Tag#getTechList}.
129      * This indicates the tag is not MIFARE Classic compatible, or this Android
130      * device does not support MIFARE Classic.
131      *
132      * @param tag an MIFARE Classic compatible tag
133      * @return MIFARE Classic object
134      */
get(Tag tag)135     public static MifareClassic get(Tag tag) {
136         if (!tag.hasTech(TagTechnology.MIFARE_CLASSIC)) return null;
137         try {
138             return new MifareClassic(tag);
139         } catch (RemoteException e) {
140             return null;
141         }
142     }
143 
144     /** @hide */
MifareClassic(Tag tag)145     public MifareClassic(Tag tag) throws RemoteException {
146         super(tag, TagTechnology.MIFARE_CLASSIC);
147 
148         NfcA a = NfcA.get(tag);  // MIFARE Classic is always based on NFC a
149 
150         mIsEmulated = false;
151 
152         switch (a.getSak()) {
153         case 0x01:
154         case 0x08:
155             mType = TYPE_CLASSIC;
156             mSize = SIZE_1K;
157             break;
158         case 0x09:
159             mType = TYPE_CLASSIC;
160             mSize = SIZE_MINI;
161             break;
162         case 0x10:
163             mType = TYPE_PLUS;
164             mSize = SIZE_2K;
165             // SecLevel = SL2
166             break;
167         case 0x11:
168             mType = TYPE_PLUS;
169             mSize = SIZE_4K;
170             // Seclevel = SL2
171             break;
172         case 0x18:
173             mType = TYPE_CLASSIC;
174             mSize = SIZE_4K;
175             break;
176         case 0x28:
177             mType = TYPE_CLASSIC;
178             mSize = SIZE_1K;
179             mIsEmulated = true;
180             break;
181         case 0x38:
182             mType = TYPE_CLASSIC;
183             mSize = SIZE_4K;
184             mIsEmulated = true;
185             break;
186         case 0x88:
187             mType = TYPE_CLASSIC;
188             mSize = SIZE_1K;
189             // NXP-tag: false
190             break;
191         case 0x98:
192         case 0xB8:
193             mType = TYPE_PRO;
194             mSize = SIZE_4K;
195             break;
196         default:
197             // Stack incorrectly reported a MifareClassic. We cannot handle this
198             // gracefully - we have no idea of the memory layout. Bail.
199             throw new RuntimeException(
200                     "Tag incorrectly enumerated as MIFARE Classic, SAK = " + a.getSak());
201         }
202     }
203 
204     /**
205      * Return the type of this MIFARE Classic compatible tag.
206      * <p>One of {@link #TYPE_UNKNOWN}, {@link #TYPE_CLASSIC}, {@link #TYPE_PLUS} or
207      * {@link #TYPE_PRO}.
208      * <p>Does not cause any RF activity and does not block.
209      *
210      * @return type
211      */
getType()212     public int getType() {
213         return mType;
214     }
215 
216     /**
217      * Return the size of the tag in bytes
218      * <p>One of {@link #SIZE_MINI}, {@link #SIZE_1K}, {@link #SIZE_2K}, {@link #SIZE_4K}.
219      * These constants are equal to their respective size in bytes.
220      * <p>Does not cause any RF activity and does not block.
221      * @return size in bytes
222      */
getSize()223     public int getSize() {
224         return mSize;
225     }
226 
227     /**
228      * Return true if the tag is emulated, determined at discovery time.
229      * These are actually smart-cards that emulate a MIFARE Classic interface.
230      * They can be treated identically to a MIFARE Classic tag.
231      * @hide
232      */
isEmulated()233     public boolean isEmulated() {
234         return mIsEmulated;
235     }
236 
237     /**
238      * Return the number of MIFARE Classic sectors.
239      * <p>Does not cause any RF activity and does not block.
240      * @return number of sectors
241      */
getSectorCount()242     public int getSectorCount() {
243         switch (mSize) {
244         case SIZE_1K:
245             return 16;
246         case SIZE_2K:
247             return 32;
248         case SIZE_4K:
249             return 40;
250         case SIZE_MINI:
251             return 5;
252         default:
253             return 0;
254         }
255     }
256 
257     /**
258      * Return the total number of MIFARE Classic blocks.
259      * <p>Does not cause any RF activity and does not block.
260      * @return total number of blocks
261      */
getBlockCount()262     public int getBlockCount() {
263         return mSize / BLOCK_SIZE;
264     }
265 
266     /**
267      * Return the number of blocks in the given sector.
268      * <p>Does not cause any RF activity and does not block.
269      *
270      * @param sectorIndex index of sector, starting from 0
271      * @return number of blocks in the sector
272      */
getBlockCountInSector(int sectorIndex)273     public int getBlockCountInSector(int sectorIndex) {
274         validateSector(sectorIndex);
275 
276         if (sectorIndex < 32) {
277             return 4;
278         } else {
279             return 16;
280         }
281     }
282 
283     /**
284      * Return the sector that contains a given block.
285      * <p>Does not cause any RF activity and does not block.
286      *
287      * @param blockIndex index of block to lookup, starting from 0
288      * @return sector index that contains the block
289      */
blockToSector(int blockIndex)290     public int blockToSector(int blockIndex) {
291         validateBlock(blockIndex);
292 
293         if (blockIndex < 32 * 4) {
294             return blockIndex / 4;
295         } else {
296             return 32 + (blockIndex - 32 * 4) / 16;
297         }
298     }
299 
300     /**
301      * Return the first block of a given sector.
302      * <p>Does not cause any RF activity and does not block.
303      *
304      * @param sectorIndex index of sector to lookup, starting from 0
305      * @return block index of first block in sector
306      */
sectorToBlock(int sectorIndex)307     public int sectorToBlock(int sectorIndex) {
308         if (sectorIndex < 32) {
309             return sectorIndex * 4;
310         } else {
311             return 32 * 4 + (sectorIndex - 32) * 16;
312         }
313     }
314 
315     /**
316      * Authenticate a sector with key A.
317      *
318      * <p>Successful authentication of a sector with key A enables other
319      * I/O operations on that sector. The set of operations granted by key A
320      * key depends on the ACL bits set in that sector. For more information
321      * see the MIFARE Classic specification on <a href="http://www.nxp.com">http://www.nxp.com</a>.
322      *
323      * <p>A failed authentication attempt causes an implicit reconnection to the
324      * tag, so authentication to other sectors will be lost.
325      *
326      * <p>This is an I/O operation and will block until complete. It must
327      * not be called from the main application thread. A blocked call will be canceled with
328      * {@link IOException} if {@link #close} is called from another thread.
329      *
330      * <p class="note">Requires the {@link android.Manifest.permission#NFC} permission.
331      *
332      * @param sectorIndex index of sector to authenticate, starting from 0
333      * @param key 6-byte authentication key
334      * @return true on success, false on authentication failure
335      * @throws TagLostException if the tag leaves the field
336      * @throws IOException if there is an I/O failure, or the operation is canceled
337      */
authenticateSectorWithKeyA(int sectorIndex, byte[] key)338     public boolean authenticateSectorWithKeyA(int sectorIndex, byte[] key) throws IOException {
339         return authenticate(sectorIndex, key, true);
340     }
341 
342     /**
343      * Authenticate a sector with key B.
344      *
345      * <p>Successful authentication of a sector with key B enables other
346      * I/O operations on that sector. The set of operations granted by key B
347      * depends on the ACL bits set in that sector. For more information
348      * see the MIFARE Classic specification on <a href="http://www.nxp.com">http://www.nxp.com</a>.
349      *
350      * <p>A failed authentication attempt causes an implicit reconnection to the
351      * tag, so authentication to other sectors will be lost.
352      *
353      * <p>This is an I/O operation and will block until complete. It must
354      * not be called from the main application thread. A blocked call will be canceled with
355      * {@link IOException} if {@link #close} is called from another thread.
356      *
357      * <p class="note">Requires the {@link android.Manifest.permission#NFC} permission.
358      *
359      * @param sectorIndex index of sector to authenticate, starting from 0
360      * @param key 6-byte authentication key
361      * @return true on success, false on authentication failure
362      * @throws TagLostException if the tag leaves the field
363      * @throws IOException if there is an I/O failure, or the operation is canceled
364      */
authenticateSectorWithKeyB(int sectorIndex, byte[] key)365     public boolean authenticateSectorWithKeyB(int sectorIndex, byte[] key) throws IOException {
366         return authenticate(sectorIndex, key, false);
367     }
368 
authenticate(int sector, byte[] key, boolean keyA)369     private boolean authenticate(int sector, byte[] key, boolean keyA) throws IOException {
370         validateSector(sector);
371         checkConnected();
372 
373         byte[] cmd = new byte[12];
374 
375         // First byte is the command
376         if (keyA) {
377             cmd[0] = 0x60; // phHal_eMifareAuthentA
378         } else {
379             cmd[0] = 0x61; // phHal_eMifareAuthentB
380         }
381 
382         // Second byte is block address
383         // Authenticate command takes a block address. Authenticating a block
384         // of a sector will authenticate the entire sector.
385         cmd[1] = (byte) sectorToBlock(sector);
386 
387         // Next 4 bytes are last 4 bytes of UID
388         byte[] uid = getTag().getId();
389         System.arraycopy(uid, uid.length - 4, cmd, 2, 4);
390 
391         // Next 6 bytes are key
392         System.arraycopy(key, 0, cmd, 6, 6);
393 
394         try {
395             if (transceive(cmd, false) != null) {
396                 return true;
397             }
398         } catch (TagLostException e) {
399             throw e;
400         } catch (IOException e) {
401             // No need to deal with, will return false anyway
402         }
403         return false;
404     }
405 
406     /**
407      * Read 16-byte block.
408      *
409      * <p>This is an I/O operation and will block until complete. It must
410      * not be called from the main application thread. A blocked call will be canceled with
411      * {@link IOException} if {@link #close} is called from another thread.
412      *
413      * <p class="note">Requires the {@link android.Manifest.permission#NFC} permission.
414      *
415      * @param blockIndex index of block to read, starting from 0
416      * @return 16 byte block
417      * @throws TagLostException if the tag leaves the field
418      * @throws IOException if there is an I/O failure, or the operation is canceled
419      */
readBlock(int blockIndex)420     public byte[] readBlock(int blockIndex) throws IOException {
421         validateBlock(blockIndex);
422         checkConnected();
423 
424         byte[] cmd = { 0x30, (byte) blockIndex };
425         return transceive(cmd, false);
426     }
427 
428     /**
429      * Write 16-byte block.
430      *
431      * <p>This is an I/O operation and will block until complete. It must
432      * not be called from the main application thread. A blocked call will be canceled with
433      * {@link IOException} if {@link #close} is called from another thread.
434      *
435      * <p class="note">Requires the {@link android.Manifest.permission#NFC} permission.
436      *
437      * @param blockIndex index of block to write, starting from 0
438      * @param data 16 bytes of data to write
439      * @throws TagLostException if the tag leaves the field
440      * @throws IOException if there is an I/O failure, or the operation is canceled
441      */
writeBlock(int blockIndex, byte[] data)442     public void writeBlock(int blockIndex, byte[] data) throws IOException {
443         validateBlock(blockIndex);
444         checkConnected();
445         if (data.length != 16) {
446             throw new IllegalArgumentException("must write 16-bytes");
447         }
448 
449         byte[] cmd = new byte[data.length + 2];
450         cmd[0] = (byte) 0xA0; // MF write command
451         cmd[1] = (byte) blockIndex;
452         System.arraycopy(data, 0, cmd, 2, data.length);
453 
454         transceive(cmd, false);
455     }
456 
457     /**
458      * Increment a value block, storing the result in the temporary block on the tag.
459      *
460      * <p>This is an I/O operation and will block until complete. It must
461      * not be called from the main application thread. A blocked call will be canceled with
462      * {@link IOException} if {@link #close} is called from another thread.
463      *
464      * <p class="note">Requires the {@link android.Manifest.permission#NFC} permission.
465      *
466      * @param blockIndex index of block to increment, starting from 0
467      * @param value non-negative to increment by
468      * @throws TagLostException if the tag leaves the field
469      * @throws IOException if there is an I/O failure, or the operation is canceled
470      */
increment(int blockIndex, int value)471     public void increment(int blockIndex, int value) throws IOException {
472         validateBlock(blockIndex);
473         validateValueOperand(value);
474         checkConnected();
475 
476         ByteBuffer cmd = ByteBuffer.allocate(6);
477         cmd.order(ByteOrder.LITTLE_ENDIAN);
478         cmd.put( (byte) 0xC1 );
479         cmd.put( (byte) blockIndex );
480         cmd.putInt(value);
481 
482         transceive(cmd.array(), false);
483     }
484 
485     /**
486      * Decrement a value block, storing the result in the temporary block on the tag.
487      *
488      * <p>This is an I/O operation and will block until complete. It must
489      * not be called from the main application thread. A blocked call will be canceled with
490      * {@link IOException} if {@link #close} is called from another thread.
491      *
492      * <p class="note">Requires the {@link android.Manifest.permission#NFC} permission.
493      *
494      * @param blockIndex index of block to decrement, starting from 0
495      * @param value non-negative to decrement by
496      * @throws TagLostException if the tag leaves the field
497      * @throws IOException if there is an I/O failure, or the operation is canceled
498      */
decrement(int blockIndex, int value)499     public void decrement(int blockIndex, int value) throws IOException {
500         validateBlock(blockIndex);
501         validateValueOperand(value);
502         checkConnected();
503 
504         ByteBuffer cmd = ByteBuffer.allocate(6);
505         cmd.order(ByteOrder.LITTLE_ENDIAN);
506         cmd.put( (byte) 0xC0 );
507         cmd.put( (byte) blockIndex );
508         cmd.putInt(value);
509 
510         transceive(cmd.array(), false);
511     }
512 
513     /**
514      * Copy from the temporary block to a value block.
515      *
516      * <p>This is an I/O operation and will block until complete. It must
517      * not be called from the main application thread. A blocked call will be canceled with
518      * {@link IOException} if {@link #close} is called from another thread.
519      *
520      * <p class="note">Requires the {@link android.Manifest.permission#NFC} permission.
521      *
522      * @param blockIndex index of block to copy to
523      * @throws TagLostException if the tag leaves the field
524      * @throws IOException if there is an I/O failure, or the operation is canceled
525      */
transfer(int blockIndex)526     public void transfer(int blockIndex) throws IOException {
527         validateBlock(blockIndex);
528         checkConnected();
529 
530         byte[] cmd = { (byte) 0xB0, (byte) blockIndex };
531 
532         transceive(cmd, false);
533     }
534 
535     /**
536      * Copy from a value block to the temporary block.
537      *
538      * <p>This is an I/O operation and will block until complete. It must
539      * not be called from the main application thread. A blocked call will be canceled with
540      * {@link IOException} if {@link #close} is called from another thread.
541      *
542      * <p class="note">Requires the {@link android.Manifest.permission#NFC} permission.
543      *
544      * @param blockIndex index of block to copy from
545      * @throws TagLostException if the tag leaves the field
546      * @throws IOException if there is an I/O failure, or the operation is canceled
547      */
restore(int blockIndex)548     public void restore(int blockIndex) throws IOException {
549         validateBlock(blockIndex);
550         checkConnected();
551 
552         byte[] cmd = { (byte) 0xC2, (byte) blockIndex };
553 
554         transceive(cmd, false);
555     }
556 
557     /**
558      * Send raw NfcA data to a tag and receive the response.
559      *
560      * <p>This is equivalent to connecting to this tag via {@link NfcA}
561      * and calling {@link NfcA#transceive}. Note that all MIFARE Classic
562      * tags are based on {@link NfcA} technology.
563      *
564      * <p>Use {@link #getMaxTransceiveLength} to retrieve the maximum number of bytes
565      * that can be sent with {@link #transceive}.
566      *
567      * <p>This is an I/O operation and will block until complete. It must
568      * not be called from the main application thread. A blocked call will be canceled with
569      * {@link IOException} if {@link #close} is called from another thread.
570      *
571      * <p class="note">Requires the {@link android.Manifest.permission#NFC} permission.
572      *
573      * @see NfcA#transceive
574      */
transceive(byte[] data)575     public byte[] transceive(byte[] data) throws IOException {
576         return transceive(data, true);
577     }
578 
579     /**
580      * Return the maximum number of bytes that can be sent with {@link #transceive}.
581      * @return the maximum number of bytes that can be sent with {@link #transceive}.
582      */
getMaxTransceiveLength()583     public int getMaxTransceiveLength() {
584         return getMaxTransceiveLengthInternal();
585     }
586 
587     /**
588      * Set the {@link #transceive} timeout in milliseconds.
589      *
590      * <p>The timeout only applies to {@link #transceive} on this object,
591      * and is reset to a default value when {@link #close} is called.
592      *
593      * <p>Setting a longer timeout may be useful when performing
594      * transactions that require a long processing time on the tag
595      * such as key generation.
596      *
597      * <p class="note">Requires the {@link android.Manifest.permission#NFC} permission.
598      *
599      * @param timeout timeout value in milliseconds
600      * @throws SecurityException if the tag object is reused after the tag has left the field
601      */
setTimeout(int timeout)602     public void setTimeout(int timeout) {
603         try {
604             int err = mTag.getTagService().setTimeout(TagTechnology.MIFARE_CLASSIC, timeout);
605             if (err != ErrorCodes.SUCCESS) {
606                 throw new IllegalArgumentException("The supplied timeout is not valid");
607             }
608         } catch (RemoteException e) {
609             Log.e(TAG, "NFC service dead", e);
610         }
611     }
612 
613     /**
614      * Get the current {@link #transceive} timeout in milliseconds.
615      *
616      * <p class="note">Requires the {@link android.Manifest.permission#NFC} permission.
617      *
618      * @return timeout value in milliseconds
619      * @throws SecurityException if the tag object is reused after the tag has left the field
620      */
getTimeout()621     public int getTimeout() {
622         try {
623             return mTag.getTagService().getTimeout(TagTechnology.MIFARE_CLASSIC);
624         } catch (RemoteException e) {
625             Log.e(TAG, "NFC service dead", e);
626             return 0;
627         }
628     }
629 
validateSector(int sector)630     private static void validateSector(int sector) {
631         // Do not be too strict on upper bounds checking, since some cards
632         // have more addressable memory than they report. For example,
633         // MIFARE Plus 2k cards will appear as MIFARE Classic 1k cards when in
634         // MIFARE Classic compatibility mode.
635         // Note that issuing a command to an out-of-bounds block is safe - the
636         // tag should report error causing IOException. This validation is a
637         // helper to guard against obvious programming mistakes.
638         if (sector < 0 || sector >= MAX_SECTOR_COUNT) {
639             throw new IndexOutOfBoundsException("sector out of bounds: " + sector);
640         }
641     }
642 
validateBlock(int block)643     private static void validateBlock(int block) {
644         // Just looking for obvious out of bounds...
645         if (block < 0 || block >= MAX_BLOCK_COUNT) {
646             throw new IndexOutOfBoundsException("block out of bounds: " + block);
647         }
648     }
649 
validateValueOperand(int value)650     private static void validateValueOperand(int value) {
651         if (value < 0) {
652             throw new IllegalArgumentException("value operand negative");
653         }
654     }
655 }
656