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

org.apache.tomcat.spdy.SpdyFrame Maven / Gradle / Ivy

/*
 *  Licensed to the Apache Software Foundation (ASF) under one or more
 *  contributor license agreements.  See the NOTICE file distributed with
 *  this work for additional information regarding copyright ownership.
 *  The ASF 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.apache.tomcat.spdy;

import java.nio.charset.StandardCharsets;
import java.util.Map;

public class SpdyFrame {
    public static final byte[] STATUS = "status".getBytes();

    public static final byte[] VERSION = "version".getBytes();

    public static final byte[] HTTP11 = "HTTP/1.1".getBytes();

    public static final byte[] OK200 = "200 OK".getBytes();


    // This is a bit more complicated, to avoid multiple reads/writes.
    // We'll read as much as possible - possible past frame end. This may
    // cost an extra copy - or even more complexity for dealing with slices
    // if we want to save the copy.
    public byte[] data;

    public int off = 8; // used when reading - current offset

    int endReadData; // how much has been read ( may be more or less than a frame )

    // On write it is incremented.

    /**
     *  end of data in the buffer.
     */
    public int endData;

    // Processed data from the frame
    boolean c; // for control

    int version;

    int flags;

    public int type;

    // For control frames
    public int streamId;

    public int pri;

    public int associated;

    public int nvCount;

    public SpdyStream stream;

    public SpdyFrame(int size) {
        data = new byte[size];
    }

    public int getDataSize() {
        return endData - 8;
    }

    public void recyle() {
        type = 0;
        c = false;
        endReadData = 0;
        off = 8;
        streamId = 0;
        nvCount = 0;
        endData = 0;
    }

    @Override
    public String toString() {
        if (c) {
            if (type == 6) {
                return "C PING " + read32(data, 0);
            }
            return "C" + " S=" + streamId + (flags != 0 ? " F=" + flags : "")
                    + (version != 2 ? "  v" + version : "") + " t=" + type
                    + " L=" + endData + "/" + off;
        } else {
            return "D" + " S=" + streamId + (flags != 0 ? " F=" + flags : "")
                    + " L=" + endData + "/" + off;
        }
    }

    public int serializeHead() {
        if (c) {
            data[0] = (byte) 0x80;
            data[1] = 2;
            data[2] = 0;
            data[3] = (byte) type;
            data[4] = (byte) flags;
            append24(data, 5, endData - 8);
            if (type == SpdyConnection.TYPE_SYN_STREAM) {
                // nvcount is added before
                append32(data, 8, streamId);
                append32(data, 12, associated);
                data[16] = 0; // TODO: priority
                data[17] = 0;
                return 18;
            } else if (type == SpdyConnection.TYPE_SYN_REPLY) {
                append32(data, 8, streamId);
                data[12] = 0;
                data[13] = 0;
                return 14;
            } else if (type == SpdyConnection.TYPE_HEADERS) {
                append32(data, 8, streamId);
                data[12] = 0;
                data[13] = 0;
                return 14;
            }
        } else {
            append32(data, 0, streamId);
            data[4] = (byte) flags;
            append24(data, 5, endData - 8);
        }
        return 8;
    }

    public boolean parse() {
        endData = 0;
        streamId = 0;
        nvCount = 0;

        int b0 = data[0] & 0xFF;
        if (b0 < 128) {
            // data frame
            c = false;
            streamId = read32(data, 0);
            version = 2;
        } else {
            c = true;
            b0 -= 128;
            version = ((b0 << 8) | data[1] & 0xFF);
            if (version > 2) {
                return false;
            }
            b0 = data[2] & 0xFF;
            type = ((b0 << 8) | (data[3] & 0xFF));
        }

        flags = data[4] & 0xFF;
        for (int i = 5; i < 8; i++) {
            b0 = data[i] & 0xFF;
            endData = endData << 8 | b0;
        }

        // size will represent the end of the data ( header is held in same
        // buffer)
        endData += 8;

        return true;
    }

    public boolean isHalfClose() {
        return (flags & SpdyConnection.FLAG_HALF_CLOSE) != 0;
    }

    public void halfClose() {
        flags = SpdyConnection.FLAG_HALF_CLOSE;
    }

    public boolean closed() {
        return (flags & SpdyConnection.FLAG_HALF_CLOSE) != 0;
    }

    static void append24(byte[] buff, int off, int v) {
        buff[off++] = (byte) ((v & 0xFF0000) >> 16);
        buff[off++] = (byte) ((v & 0xFF00) >> 8);
        buff[off++] = (byte) ((v & 0xFF));
    }

    static void append32(byte[] buff, int off, int v) {
        buff[off++] = (byte) ((v & 0xFF000000) >> 24);
        buff[off++] = (byte) ((v & 0xFF0000) >> 16);
        buff[off++] = (byte) ((v & 0xFF00) >> 8);
        buff[off++] = (byte) ((v & 0xFF));
    }

    public void append32(int v) {
        makeSpace(4);
        data[off++] = (byte) ((v & 0xFF000000) >> 24);
        data[off++] = (byte) ((v & 0xFF0000) >> 16);
        data[off++] = (byte) ((v & 0xFF00) >> 8);
        data[off++] = (byte) ((v & 0xFF));
    }

    public void append16(int v) {
        makeSpace(2);
        data[off++] = (byte) ((v & 0xFF00) >> 8);
        data[off++] = (byte) ((v & 0xFF));
    }

    void fixNV(int nvPos) {
        data[nvPos++] = (byte) ((nvCount & 0xFF00) >> 8);
        data[nvPos] = (byte) ((nvCount & 0xFF));
    }

    public void append(byte[] buf, int soff, int len) {
        makeSpace(len + off);
        System.arraycopy(buf, soff, data, off, len);
        off += len;
    }

    public void headerValue(byte[] buf, int soff, int len) {
        makeSpace(len + 4);
        append16(len);
        System.arraycopy(buf, soff, data, off, len);
        off += len;
    }

    public void headerName(byte[] buf, int soff, int len) {
        // if it's the first header, leave space for extra params and NV count.
        // they'll be filled in by send.
        if (off == 8) {
            if (type == SpdyConnection.TYPE_SYN_REPLY) {
                off = 16;
            } else if (type == SpdyConnection.TYPE_SYN_STREAM) {
                off = 20;
            } else if (type != SpdyConnection.TYPE_HEADERS) {
                off = 16;
            } else {
                throw new RuntimeException("Wrong frame type");
            }
        }
        nvCount++;
        headerValue(buf, soff, len);
    }

    public void addHeader(String name, String value) {
        byte[] nameB = name.getBytes();
        headerName(nameB, 0, nameB.length);
        nameB = value.getBytes();
        headerValue(nameB, 0, nameB.length);
    }

    public void addHeader(byte[] nameB, String value) {
        headerName(nameB, 0, nameB.length);
        nameB = value.getBytes();
        headerValue(nameB, 0, nameB.length);
    }

    public void addHeader(byte[] nameB, byte[] valueB) {
        headerName(nameB, 0, nameB.length);
        headerValue(valueB, 0, valueB.length);
    }

    public void getHeaders(Map resHeaders) {
        for (int i = 0; i < nvCount; i++) {
            int len = read16();
            String n = new String(data, off, len, StandardCharsets.UTF_8);
            advance(len);
            len = read16();
            String v = new String(data, off, len, StandardCharsets.UTF_8);
            advance(len);
            resHeaders.put(n, v);
        }
    }


    // TODO: instead of that, use byte[][]
    void makeSpace(int len) {
        if (len < 256) {
            len = 256;
        }
        if (data == null) {
            data = new byte[len];
            return;
        }
        int newEnd = off + len;

        if (data.length < newEnd) {
            byte[] tmp = new byte[newEnd];
            System.err.println("cp " + off + " " + data.length + " " + len
                    + " " + tmp.length);
            System.arraycopy(data, 0, tmp, 0, off);
            data = tmp;
        }

    }

    public int read16() {
        int res = data[off++] & 0xFF;
        return res << 8 | (data[off++] & 0xFF);
    }

    int readInt() {
        int res = 0;
        for (int i = 0; i < 4; i++) {
            int b0 = data[off++] & 0xFF;
            res = res << 8 | b0;
        }
        return res;
    }

    int read24() {
        int res = 0;
        for (int i = 0; i < 3; i++) {
            int b0 = data[off++] & 0xFF;
            res = res << 8 | b0;
        }
        return res;
    }

    int read32(byte[] data, int off) {
        int res = 0;
        for (int i = 0; i < 4; i++) {
            int b0 = data[off++] & 0xFF;
            res = res << 8 | b0;
        }
        return res;
    }

    int read32() {
        int res = 0;
        for (int i = 0; i < 4; i++) {
            int b0 = data[off++] & 0xFF;
            res = res << 8 | b0;
        }
        return res;
    }

    public int readByte() {
        return data[off++] & 0xFF;
    }

    public int remaining() {
        return endData - off;
    }

    public void advance(int cnt) {
        off += cnt;
    }

    public boolean isData() {
        return !c;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy