/*
 * Copyright 2016 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 io.netty.handler.codec.compression;

import io.netty.buffer.ByteBuf;
import io.netty.util.ByteProcessor;
import io.netty.util.internal.ObjectUtil;
import io.netty.util.internal.PlatformDependent;

import java.lang.reflect.Method;
import java.nio.ByteBuffer;
import java.util.zip.Adler32;
import java.util.zip.CRC32;
import java.util.zip.Checksum;

Checksum implementation which can directly act on a ByteBuf. Implementations may optimize access patterns depending on if the ByteBuf is backed by a byte array (ByteBuf.hasArray() is true) or not.
/** * {@link Checksum} implementation which can directly act on a {@link ByteBuf}. * * Implementations may optimize access patterns depending on if the {@link ByteBuf} is backed by a * byte array ({@link ByteBuf#hasArray()} is {@code true}) or not. */
abstract class ByteBufChecksum implements Checksum { private static final Method ADLER32_UPDATE_METHOD; private static final Method CRC32_UPDATE_METHOD; static { // See if we can use fast-path when using ByteBuf that is not heap based as Adler32 and CRC32 added support // for update(ByteBuffer) in JDK8. ADLER32_UPDATE_METHOD = updateByteBuffer(new Adler32()); CRC32_UPDATE_METHOD = updateByteBuffer(new CRC32()); } private final ByteProcessor updateProcessor = new ByteProcessor() { @Override public boolean process(byte value) throws Exception { update(value); return true; } }; private static Method updateByteBuffer(Checksum checksum) { if (PlatformDependent.javaVersion() >= 8) { try { Method method = checksum.getClass().getDeclaredMethod("update", ByteBuffer.class); method.invoke(method, ByteBuffer.allocate(1)); return method; } catch (Throwable ignore) { return null; } } return null; } static ByteBufChecksum wrapChecksum(Checksum checksum) { ObjectUtil.checkNotNull(checksum, "checksum"); if (checksum instanceof Adler32 && ADLER32_UPDATE_METHOD != null) { return new ReflectiveByteBufChecksum(checksum, ADLER32_UPDATE_METHOD); } if (checksum instanceof CRC32 && CRC32_UPDATE_METHOD != null) { return new ReflectiveByteBufChecksum(checksum, CRC32_UPDATE_METHOD); } return new SlowByteBufChecksum(checksum); }
See Also:
  • update(byte[], int, int)
/** * @see #update(byte[], int, int) */
public void update(ByteBuf b, int off, int len) { if (b.hasArray()) { update(b.array(), b.arrayOffset() + off, len); } else { b.forEachByte(off, len, updateProcessor); } } private static final class ReflectiveByteBufChecksum extends SlowByteBufChecksum { private final Method method; ReflectiveByteBufChecksum(Checksum checksum, Method method) { super(checksum); this.method = method; } @Override public void update(ByteBuf b, int off, int len) { if (b.hasArray()) { update(b.array(), b.arrayOffset() + off, len); } else { try { method.invoke(checksum, CompressionUtil.safeNioBuffer(b)); } catch (Throwable cause) { throw new Error(); } } } } private static class SlowByteBufChecksum extends ByteBufChecksum { protected final Checksum checksum; SlowByteBufChecksum(Checksum checksum) { this.checksum = checksum; } @Override public void update(int b) { checksum.update(b); } @Override public void update(byte[] b, int off, int len) { checksum.update(b, off, len); } @Override public long getValue() { return checksum.getValue(); } @Override public void reset() { checksum.reset(); } } }