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