/* * Copyright 2014 The Netty Project * * The Netty Project licenses this file to you 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 org.jboss.netty.handler.codec.spdy; import org.jboss.netty.buffer.ChannelBuffer; import org.jboss.netty.buffer.ChannelBuffers; import java.util.zip.DataFormatException; import java.util.zip.Inflater; import static org.jboss.netty.handler.codec.spdy.SpdyCodecUtil.*; final class SpdyHeaderBlockZlibDecoder extends SpdyHeaderBlockRawDecoder { private static final int DEFAULT_BUFFER_CAPACITY = 4096; private static final SpdyProtocolException INVALID_HEADER_BLOCK = new SpdyProtocolException("Invalid Header Block"); private final Inflater decompressor = new Inflater(); private ChannelBuffer decompressed; SpdyHeaderBlockZlibDecoder(SpdyVersion spdyVersion, int maxHeaderSize) { super(spdyVersion, maxHeaderSize); } @Override void decode(ChannelBuffer headerBlock, SpdyHeadersFrame frame) throws Exception { if (headerBlock == null) { throw new NullPointerException("headerBlock"); } if (frame == null) { throw new NullPointerException("frame"); } int len = setInput(headerBlock); int numBytes; do { numBytes = decompress(frame); } while (numBytes > 0); // z_stream has an internal 64-bit hold buffer // it is always capable of consuming the entire input if (decompressor.getRemaining() != 0) { // we reached the end of the deflate stream throw INVALID_HEADER_BLOCK; } headerBlock.skipBytes(len); } private int setInput(ChannelBuffer compressed) { int len = compressed.readableBytes(); if (compressed.hasArray()) { decompressor.setInput(compressed.array(), compressed.arrayOffset() + compressed.readerIndex(), len); } else { byte[] in = new byte[len]; compressed.getBytes(compressed.readerIndex(), in); decompressor.setInput(in, 0, in.length); } return len; } private int decompress(SpdyHeadersFrame frame) throws Exception { ensureBuffer(); byte[] out = decompressed.array(); int off = decompressed.arrayOffset() + decompressed.writerIndex(); try { int numBytes = decompressor.inflate(out, off, decompressed.writableBytes()); if (numBytes == 0 && decompressor.needsDictionary()) { try { decompressor.setDictionary(SPDY_DICT); } catch (IllegalArgumentException e) { throw INVALID_HEADER_BLOCK; } numBytes = decompressor.inflate(out, off, decompressed.writableBytes()); } decompressed.writerIndex(decompressed.writerIndex() + numBytes); super.decode(decompressed, frame); decompressed.discardReadBytes(); return numBytes; } catch (DataFormatException e) { throw INVALID_HEADER_BLOCK; } } private void ensureBuffer() { if (decompressed == null) { decompressed = ChannelBuffers.dynamicBuffer(DEFAULT_BUFFER_CAPACITY); } decompressed.ensureWritableBytes(1); } @Override void endHeaderBlock(SpdyHeadersFrame frame) throws Exception { super.endHeaderBlock(frame); decompressed = null; } @Override public void end() { super.end(); decompressed = null; decompressor.end(); } }