Home | History | Annotate | Download | only in io
      1 /*
      2  * Licensed to the Apache Software Foundation (ASF) under one or more
      3  * contributor license agreements.  See the NOTICE file distributed with
      4  * this work for additional information regarding copyright ownership.
      5  * The ASF licenses this file to You under the Apache License, Version 2.0
      6  * (the "License"); you may not use this file except in compliance with
      7  * the License.  You may obtain a copy of the License at
      8  *
      9  *      http://www.apache.org/licenses/LICENSE-2.0
     10  *
     11  * Unless required by applicable law or agreed to in writing, software
     12  * distributed under the License is distributed on an "AS IS" BASIS,
     13  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
     14  * See the License for the specific language governing permissions and
     15  * limitations under the License.
     16  */
     17 package org.apache.commons.io;
     18 
     19 import java.io.IOException;
     20 import java.io.OutputStream;
     21 
     22 /**
     23  * Dumps data in hexadecimal format.
     24  * <p>
     25  * Provides a single function to take an array of bytes and display it
     26  * in hexadecimal form.
     27  * <p>
     28  * Origin of code: POI.
     29  *
     30  * @author Scott Sanders
     31  * @author Marc Johnson
     32  * @version $Id: HexDump.java 596667 2007-11-20 13:50:14Z niallp $
     33  */
     34 public class HexDump {
     35 
     36     /**
     37      * Instances should NOT be constructed in standard programming.
     38      */
     39     public HexDump() {
     40         super();
     41     }
     42 
     43     /**
     44      * Dump an array of bytes to an OutputStream.
     45      *
     46      * @param data  the byte array to be dumped
     47      * @param offset  its offset, whatever that might mean
     48      * @param stream  the OutputStream to which the data is to be
     49      *               written
     50      * @param index initial index into the byte array
     51      *
     52      * @throws IOException is thrown if anything goes wrong writing
     53      *         the data to stream
     54      * @throws ArrayIndexOutOfBoundsException if the index is
     55      *         outside the data array's bounds
     56      * @throws IllegalArgumentException if the output stream is null
     57      */
     58 
     59     public static void dump(byte[] data, long offset,
     60                             OutputStream stream, int index)
     61             throws IOException, ArrayIndexOutOfBoundsException,
     62             IllegalArgumentException {
     63 
     64         if ((index < 0) || (index >= data.length)) {
     65             throw new ArrayIndexOutOfBoundsException(
     66                     "illegal index: " + index + " into array of length "
     67                     + data.length);
     68         }
     69         if (stream == null) {
     70             throw new IllegalArgumentException("cannot write to nullstream");
     71         }
     72         long display_offset = offset + index;
     73         StringBuffer buffer = new StringBuffer(74);
     74 
     75         for (int j = index; j < data.length; j += 16) {
     76             int chars_read = data.length - j;
     77 
     78             if (chars_read > 16) {
     79                 chars_read = 16;
     80             }
     81             dump(buffer, display_offset).append(' ');
     82             for (int k = 0; k < 16; k++) {
     83                 if (k < chars_read) {
     84                     dump(buffer, data[k + j]);
     85                 } else {
     86                     buffer.append("  ");
     87                 }
     88                 buffer.append(' ');
     89             }
     90             for (int k = 0; k < chars_read; k++) {
     91                 if ((data[k + j] >= ' ') && (data[k + j] < 127)) {
     92                     buffer.append((char) data[k + j]);
     93                 } else {
     94                     buffer.append('.');
     95                 }
     96             }
     97             buffer.append(EOL);
     98             stream.write(buffer.toString().getBytes());
     99             stream.flush();
    100             buffer.setLength(0);
    101             display_offset += chars_read;
    102         }
    103     }
    104 
    105     /**
    106      * The line-separator (initializes to "line.separator" system property.
    107      */
    108     public static final String EOL =
    109             System.getProperty("line.separator");
    110     private static final char[] _hexcodes =
    111             {
    112                 '0', '1', '2', '3', '4', '5', '6', '7', '8', '9',
    113                 'A', 'B', 'C', 'D', 'E', 'F'
    114             };
    115     private static final int[] _shifts =
    116             {
    117                 28, 24, 20, 16, 12, 8, 4, 0
    118             };
    119 
    120     /**
    121      * Dump a long value into a StringBuffer.
    122      *
    123      * @param _lbuffer the StringBuffer to dump the value in
    124      * @param value  the long value to be dumped
    125      * @return StringBuffer containing the dumped value.
    126      */
    127     private static StringBuffer dump(StringBuffer _lbuffer, long value) {
    128         for (int j = 0; j < 8; j++) {
    129             _lbuffer
    130                     .append(_hexcodes[((int) (value >> _shifts[j])) & 15]);
    131         }
    132         return _lbuffer;
    133     }
    134 
    135     /**
    136      * Dump a byte value into a StringBuffer.
    137      *
    138      * @param _cbuffer the StringBuffer to dump the value in
    139      * @param value  the byte value to be dumped
    140      * @return StringBuffer containing the dumped value.
    141      */
    142     private static StringBuffer dump(StringBuffer _cbuffer, byte value) {
    143         for (int j = 0; j < 2; j++) {
    144             _cbuffer.append(_hexcodes[(value >> _shifts[j + 6]) & 15]);
    145         }
    146         return _cbuffer;
    147     }
    148 
    149 }
    150