All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.signalfx.shaded.jetty.http.compression.NBitIntegerEncoder Maven / Gradle / Ivy

//
//  ========================================================================
//  Copyright (c) 1995-2022 Mort Bay Consulting Pty Ltd and others.
//  ------------------------------------------------------------------------
//  All rights reserved. This program and the accompanying materials
//  are made available under the terms of the Eclipse Public License v1.0
//  and Apache License v2.0 which accompanies this distribution.
//
//      The Eclipse Public License is available at
//      http://www.eclipse.org/legal/epl-v10.html
//
//      The Apache License v2.0 is available at
//      http://www.opensource.org/licenses/apache2.0.php
//
//  You may elect to redistribute this code under either of these licenses.
//  ========================================================================
//

package com.signalfx.shaded.jetty.http.compression;

import java.nio.ByteBuffer;

/**
 * Used to encode integers as described in RFC7541.
 */
public class NBitIntegerEncoder
{
    private NBitIntegerEncoder()
    {
    }

    /**
     * @param prefix the prefix used to encode this long.
     * @param value the integer to encode.
     * @return the number of octets it would take to encode the long.
     */
    public static int octetsNeeded(int prefix, long value)
    {
        if (prefix <= 0 || prefix > 8)
            throw new IllegalArgumentException();

        int nbits = 0xFF >>> (8 - prefix);
        value = value - nbits;
        if (value < 0)
            return 1;
        if (value == 0)
            return 2;
        int lz = Long.numberOfLeadingZeros(value);
        int log = 64 - lz;

        // The return value is 1 for the prefix + the number of 7-bit groups necessary to encode the value.
        return 1 + (log + 6) / 7;
    }

    /**
     *
     * @param buffer the buffer to encode into.
     * @param prefix the prefix used to encode this long.
     * @param value the long to encode into the buffer.
     */
    public static void encode(ByteBuffer buffer, int prefix, long value)
    {
        if (prefix <= 0 || prefix > 8)
            throw new IllegalArgumentException();

        // If prefix is 8 we add an empty byte as we initially modify last byte from the buffer.
        if (prefix == 8)
            buffer.put((byte)0x00);

        int bits = 0xFF >>> (8 - prefix);
        int p = buffer.position() - 1;
        if (value < bits)
        {
            buffer.put(p, (byte)((buffer.get(p) & ~bits) | value));
        }
        else
        {
            buffer.put(p, (byte)(buffer.get(p) | bits));
            long length = value - bits;
            while (true)
            {
                // The value of ~0x7F is different to 0x80 because of all the 1s from the MSB.
                if ((length & ~0x7FL) == 0)
                {
                    buffer.put((byte)length);
                    return;
                }
                else
                {
                    buffer.put((byte)((length & 0x7F) | 0x80));
                    length >>>= 7;
                }
            }
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy