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

org.wildfly.common.iteration.Base32DecodingByteIterator Maven / Gradle / Ivy

/*
 * JBoss, Home of Professional Open Source.
 * Copyright 2017 Red Hat, Inc., and individual contributors
 * as indicated by the @author tags.
 *
 * Licensed 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.wildfly.common.iteration;

import static org.wildfly.common._private.CommonMessages.msg;

import java.util.NoSuchElementException;

/**
 */
abstract class Base32DecodingByteIterator extends ByteIterator {
    private final CodePointIterator iter;
    private final boolean requirePadding;

    // states:
    // 0x00: nothing read
    // 0x01: five bytes to return o0..o4
    // 0x02: four bytes to return o1..o4 (o0 still populated)
    // 0x03: three byte to return o2..o4 (o0..o1 still populated)
    // 0x04: two bytes to return o3..o4 (o0..o2 still populated)
    // 0x05: one byte to return o4 (o0..o3 still populated)
    // 0x06: four bytes then eof o0..o3 =
    // 0x07: three bytes then eof o1..o3 = (o0 still populated)
    // 0x08: two bytes then eof o2..o3 = (o0..o1 still populated)
    // 0x09: one byte then eof o3 = (o0..o2 still populated)
    // 0x0a: three bytes then eof o0..o2 ===
    // 0x0b: two bytes then eof o1..o2 === (o0 still populated)
    // 0x0c: one byte then eof o2 === (o0..o1 still populated)
    // 0x0d: two bytes then eof o0..o1 ====
    // 0x0e: one byte then eof o1 ==== (o0 still populated)
    // 0x0f: one byte then eof o0 ======
    // 0x10: four bytes then eof o0..o3 no pad
    // 0x11: three bytes then eof o1..o3 no pad (o0 still populated)
    // 0x12: two bytes then eof o2..o3 no pad (o0..o1 still populated)
    // 0x13: one byte then eof o3 no pad (o0..o2 still populated)
    // 0x14: three bytes then eof o0..o2 no pad
    // 0x15: two bytes then eof o1..o2 no pad (o0 still populated)
    // 0x16: one byte then eof o2 no pad (o0..o1 still populated)
    // 0x17: two bytes then eof o0..o1 no pad
    // 0x18: one byte then eof o1 no pad (o0 still populated)
    // 0x19: one byte then eof o0 no pad
    // 0x1a: end (=) (o0..o3 still populated)
    // 0x1b: end (===) (o0..o2 still populated)
    // 0x1c: end (====) (o0..o1 still populated)
    // 0x1d: end (======) (o0 still populated)
    // 0x1e: end (= but no pad) (o0..o3 still populated)
    // 0x1f: end (=== but no pad) (o0..o2 still populated)
    // 0x20: end (==== but no pad) (o0..o1 still populated)
    // 0x21: end (====== but no pad) (o0 still populated)

    private int state = 0;
    private int o0, o1, o2, o3, o4;
    private int offset;

    Base32DecodingByteIterator(final CodePointIterator iter, final boolean requirePadding) {
        this.iter = iter;
        this.requirePadding = requirePadding;
    }

    public boolean hasNext() {
        if (state == 0) {
            if (! iter.hasNext()) {
                return false;
            }
            int b0 = iter.next();
            if (b0 == '=') {
                throw msg.unexpectedPadding();
            }
            if (! iter.hasNext()) {
                if (requirePadding) {
                    throw msg.expectedPadding();
                } else {
                    throw msg.incompleteDecode();
                }
            }
            int b1 = iter.next();
            if (b1 == '=') {
                throw msg.unexpectedPadding();
            }
            o0 = calc0(b0, b1);
            if (! iter.hasNext()) {
                if (requirePadding) {
                    throw msg.expectedPadding();
                }
                state = 0x19;
                return true;
            }
            int b2 = iter.next();
            if (b2 == '=') {
                for (int i = 0; i < 5; i++) {
                    if (! iter.hasNext()) {
                        throw msg.expectedPaddingCharacters(6);
                    }
                    if (iter.next() != '=') {
                        throw msg.expectedPaddingCharacters(6);
                    }
                }
                state = 0x0f;
                return true;
            }
            if (! iter.hasNext()) {
                if (requirePadding) {
                    throw msg.expectedPadding();
                } else {
                    throw msg.incompleteDecode();
                }
            }
            int b3 = iter.next();
            if (b3 == '=') {
                throw msg.unexpectedPadding();
            }
            o1 = calc1(b1, b2, b3);
            if (! iter.hasNext()) {
                if (requirePadding) {
                    throw msg.expectedPadding();
                }
                state = 0x17;
                return true;
            }
            int b4 = iter.next();
            if (b4 == '=') {
                for (int i = 0; i < 3; i++) {
                    if (! iter.hasNext()) {
                        throw msg.expectedPaddingCharacters(4);
                    }
                    if (iter.next() != '=') {
                        throw msg.expectedPaddingCharacters(4);
                    }
                }
                state = 0x0d;
                return true;
            }
            o2 = calc2(b3, b4);
            if (! iter.hasNext()) {
                if (requirePadding) {
                    throw msg.expectedPadding();
                }
                state = 0x14;
                return true;
            }
            int b5 = iter.next();
            if (b5 == '=') {
                for (int i = 0; i < 2; i++) {
                    if (! iter.hasNext()) {
                        throw msg.expectedPaddingCharacters(3);
                    }
                    if (iter.next() != '=') {
                        throw msg.expectedPaddingCharacters(3);
                    }
                }
                state = 0x0a;
                return true;
            }
            if (! iter.hasNext()) {
                if (requirePadding) {
                    throw msg.expectedPadding();
                } else {
                    throw msg.incompleteDecode();
                }
            }
            int b6 = iter.next();
            if (b6 == '=') {
                throw msg.unexpectedPadding();
            }
            o3 = calc3(b4, b5, b6);
            if (! iter.hasNext()) {
                if (requirePadding) {
                    throw msg.expectedPadding();
                }
                state = 0x10;
                return true;
            }
            int b7 = iter.next();
            if (b7 == '=') {
                state = 0x06;
                return true;
            }
            o4 = calc4(b6, b7);
            state = 1;
            return true;
        } else {
            return state < 0x1a;
        }
    }

    public boolean hasPrevious() {
        return offset > 0;
    }

    abstract int calc0(int b0, int b1);

    abstract int calc1(int b1, int b2, int b3);

    abstract int calc2(int b3, int b4);

    abstract int calc3(int b4, int b5, int b6);

    abstract int calc4(int b6, int b7);

    public int next() {
        if (! hasNext()) {
            throw new NoSuchElementException();
        }
        switch (state) {
            case 1:
            case 6:
            case 0x0a:
            case 0x0d:
            case 0x10:
            case 0x14:
            case 0x17: {
                state++;
                offset++;
                return o0;
            }
            case 2:
            case 7:
            case 0x0b:
            case 0x11:
            case 0x15: {
                state++;
                offset++;
                return o1;
            }
            case 3:
            case 8:
            case 0x12: {
                state++;
                offset++;
                return o2;
            }
            case 4: {
                state = 5;
                offset++;
                return o3;
            }
            case 5: {
                state = 0;
                offset++;
                return o4;
            }
            case 9: {
                state = 0x1a;
                offset++;
                return o3;
            }
            case 0x0c: {
                state = 0x1b;
                offset++;
                return o2;
            }
            case 0x0e: {
                state = 0x1c;
                offset++;
                return o1;
            }
            case 0x0f: {
                state = 0x1d;
                offset++;
                return o0;
            }
            case 0x13: {
                state = 0x1e;
                offset++;
                return o3;
            }
            case 0x16: {
                state = 0x1f;
                offset++;
                return o2;
            }
            case 0x18: {
                state = 0x20;
                offset++;
                return o1;
            }
            case 0x19: {
                state = 0x21;
                offset++;
                return o0;
            }
            default: {
                // padding
                throw new NoSuchElementException();
            }
        }
    }

    public int peekNext() throws NoSuchElementException {
        if (! hasNext()) {
            throw new NoSuchElementException();
        }
        switch (state) {
            case 1:
            case 6:
            case 0x0a:
            case 0x0d:
            case 0x0f:
            case 0x10:
            case 0x14:
            case 0x17:
            case 0x19: {
                return o0;
            }
            case 2:
            case 7:
            case 0x0b:
            case 0x0e:
            case 0x11:
            case 0x15:
            case 0x18: {
                return o1;
            }
            case 3:
            case 8:
            case 0x0c:
            case 0x12:
            case 0x16: {
                return o2;
            }
            case 4:
            case 9:
            case 0x13: {
                return o3;
            }
            case 5: {
                return o4;
            }
            default: {
                // padding
                throw new NoSuchElementException();
            }
        }
    }

    public int previous() {
        if (! hasPrevious()) {
            throw new NoSuchElementException();
        }
        int skipChars = 0;
        switch (state) {
            case 0:
            case 1:
            case 6:
            case 0x0a:
            case 0x0d:
            case 0x0f:
            case 0x10:
            case 0x14:
            case 0x17:
            case 0x19: {
                if (state == 6 || state == 0x0a || state == 0x0d || state == 0x0f) {
                    skipChars = 8;
                } else if (state == 0x10) {
                    skipChars = 7;
                } else if (state == 0x14) {
                    skipChars = 5;
                } else if (state == 0x17) {
                    skipChars = 4;
                } else if (state == 0x19) {
                    skipChars = 2;
                }
                for (int i = 0; i < skipChars; i++) {
                    iter.previous(); // consume character
                }
                int b7 = iter.previous();
                int b6 = iter.previous();
                int b5 = iter.previous();
                int b4 = iter.previous();
                int b3 = iter.previous();
                int b2 = iter.previous();
                int b1 = iter.previous();
                int b0 = iter.previous();
                o0 = calc0(b0, b1);
                o1 = calc1(b1, b2, b3);
                o2 = calc2(b3, b4);
                o3 = calc3(b4, b5, b6);
                o4 = calc4(b6, b7);
                state = 5;
                offset--;
                return o4;
            }
            case 2:
            case 7:
            case 0x0b:
            case 0x0e:
            case 0x11:
            case 0x15:
            case 0x18: {
                state--;
                offset--;
                return o0;
            }
            case 3:
            case 8:
            case 0x0c:
            case 0x12:
            case 0x16: {
                state--;
                offset--;
                return o1;
            }
            case 4:
            case 9:
            case 0x13: {
                state--;
                offset--;
                return o2;
            }
            case 5: {
                state = 4;
                offset--;
                return o3;
            }
            case 0x1a: {
                state = 9;
                offset--;
                return o3;
            }
            case 0x1b: {
                state = 0x0c;
                offset--;
                return o2;
            }
            case 0x1c: {
                state = 0x0e;
                offset--;
                return o1;
            }
            case 0x1d: {
                state = 0x0f;
                offset--;
                return o0;
            }
            case 0x1e: {
                state = 0x13;
                offset--;
                return o3;
            }
            case 0x1f: {
                state = 0x16;
                offset--;
                return o2;
            }
            case 0x20: {
                state = 0x18;
                offset--;
                return o1;
            }
            case 0x21: {
                state = 0x19;
                offset--;
                return o0;
            }
            default: {
                throw new NoSuchElementException();
            }
        }
    }

    public int peekPrevious() throws NoSuchElementException {
        if (! hasPrevious()) {
            throw new NoSuchElementException();
        }
        int skipChars = 0;
        switch (state) {
            case 0:
            case 1:
            case 6:
            case 0x0a:
            case 0x0d:
            case 0x0f:
            case 0x10:
            case 0x14:
            case 0x17:
            case 0x19: {
                if (state == 6 || state == 0x0a || state == 0x0d || state == 0x0f) {
                    skipChars = 8;
                } else if (state == 0x10) {
                    skipChars = 7;
                } else if (state == 0x14) {
                    skipChars = 5;
                } else if (state == 0x17) {
                    skipChars = 4;
                } else if (state == 0x19) {
                    skipChars = 2;
                }
                for (int i = 0; i < skipChars; i++) {
                    iter.previous(); // consume character
                }
                int b7 = iter.previous();
                int b6 = iter.peekPrevious();
                iter.next();
                for (int i = 0; i < skipChars; i++) {
                    iter.next();
                }
                return calc4(b6, b7);
            }
            case 2:
            case 7:
            case 0x0b:
            case 0x0e:
            case 0x11:
            case 0x15:
            case 0x18:
            case 0x1d:
            case 0x21: {
                return o0;
            }
            case 3:
            case 8:
            case 0x0c:
            case 0x12:
            case 0x16:
            case 0x1c:
            case 0x20: {
                return o1;
            }
            case 4:
            case 9:
            case 0x13:
            case 0x1b:
            case 0x1f: {
                return o2;
            }
            case 5:
            case 0x1a:
            case 0x1e: {
                return o3;
            }
            default: {
                throw new NoSuchElementException();
            }
        }
    }

    public long getIndex() {
        return offset;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy