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

com.googlecode.d2j.dex.writer.ev.EncodedValue Maven / Gradle / Ivy

The newest version!
/*
 * dex2jar - Tools to work with android .dex and java .class files
 * Copyright (c) 2009-2013 Panxiaobo
 *
 * 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 com.googlecode.d2j.dex.writer.ev;

import com.googlecode.d2j.dex.writer.io.DataOut;
import com.googlecode.d2j.dex.writer.item.*;

public class EncodedValue {

    public final static int VALUE_ANNOTATION = 0x1d;
    public final static int VALUE_ARRAY = 0x1c;
    public final static int VALUE_BOOLEAN = 0x1f;
    public final static int VALUE_BYTE = 0x00;
    public final static int VALUE_CHAR = 0x03;
    public final static int VALUE_DOUBLE = 0x11;
    public final static int VALUE_ENUM = 0x1b;
    public final static int VALUE_FIELD = 0x19;
    public final static int VALUE_FLOAT = 0x10;
    public final static int VALUE_INT = 0x04;
    public final static int VALUE_LONG = 0x06;
    public final static int VALUE_METHOD = 0x1a;
    public final static int VALUE_NULL = 0x1e;
    public final static int VALUE_SHORT = 0x02;
    public final static int VALUE_STRING = 0x17;
    public final static int VALUE_TYPE = 0x18;
    public final int valueType;
    public Object value;

    public EncodedValue(int valueType, Object value) {
        this.valueType = valueType;
        this.value = value;
    }

    public static int lengthOfDouble(double value) {
        int requiredBits = 64 - Long.numberOfTrailingZeros(Double.doubleToRawLongBits(value));
        if (requiredBits == 0) {
            requiredBits = 1;
        }
        return (requiredBits + 0x07) >> 3;
    }

    public static int lengthOfFloat(float value) {
        int requiredBits = 64 - Long.numberOfTrailingZeros(((long) (Float.floatToRawIntBits(value))) << 32);
        if (requiredBits == 0) {
            requiredBits = 1;
        }
        return (requiredBits + 0x07) >> 3;
    }

    public static int lengthOfSint(int value) {
        int nbBits = 33 - Integer.numberOfLeadingZeros(value ^ (value >> 31));
        return (nbBits + 0x07) >> 3;
    }

    public static int lengthOfSint(long value) {
        int nbBits = 65 - Long.numberOfLeadingZeros(value ^ (value >> 63));
        return (nbBits + 0x07) >> 3;
    }

    public final static int lengthOfUint(int val) {
        int size = 1;
        if (val != 0) {
            val = val >>> 8;
            if (val != 0) {
                size++;
                val = val >>> 8;
                if (val != 0) {
                    size++;
                    val = val >>> 8;
                    if (val != 0) {
                        size++;
                    }
                }
            }
        }
        return size;
    }

    public static EncodedValue wrap(Object v) {
        if (v == null) {
            return new EncodedValue(VALUE_NULL, null);
        }
        if (v instanceof Integer) {
            return new EncodedValue(VALUE_INT, v);
        } else if (v instanceof Short) {
            return new EncodedValue(VALUE_SHORT, v);
        } else if (v instanceof Character) {
            return new EncodedValue(VALUE_CHAR, v);
        } else if (v instanceof Long) {
            return new EncodedValue(VALUE_LONG, v);
        } else if (v instanceof Float) {
            return new EncodedValue(VALUE_FLOAT, v);
        } else if (v instanceof Double) {
            return new EncodedValue(VALUE_DOUBLE, v);
        } else if (v instanceof Boolean) {
            return new EncodedValue(VALUE_BOOLEAN, v);
        } else if (v instanceof Byte) {
            return new EncodedValue(VALUE_BYTE, v);
        } else if (v instanceof TypeIdItem) {
            return new EncodedValue(VALUE_TYPE, v);
        } else if (v instanceof StringIdItem) {
            return new EncodedValue(VALUE_STRING, v);
        } else if (v instanceof FieldIdItem) {
            return new EncodedValue(VALUE_FIELD, v);
        } else if (v instanceof MethodIdItem) {
            return new EncodedValue(VALUE_METHOD, v);
        }


        throw new RuntimeException("not support");
    }

    public static EncodedValue defaultValueForType(String typeString) {
        switch (typeString.charAt(0)) {
            case '[':
            case 'L':
                return new EncodedValue(VALUE_NULL, null);
            case 'B':
                return new EncodedValue(VALUE_BYTE, (byte) 0);
            case 'Z':
                return new EncodedValue(VALUE_BOOLEAN, false);
            case 'S':
                return new EncodedValue(VALUE_SHORT, (short) 0);
            case 'C':
                return new EncodedValue(VALUE_CHAR, (char) 0);
            case 'I':
                return new EncodedValue(VALUE_INT, (int) 0);
            case 'F':
                return new EncodedValue(VALUE_FLOAT, (float) 0);
            case 'D':
                return new EncodedValue(VALUE_DOUBLE, (double) 0);
            case 'J':
                return new EncodedValue(VALUE_LONG, (long) 0);
            default:
                throw new RuntimeException();
        }
    }

    static byte[] encodeLong(int length, long value) {
        byte[] data = new byte[length];
        switch (length) {
            case 8:
                data[7] = (byte) (value >> 56);
            case 7:
                data[6] = (byte) (value >> 48);
            case 6:
                data[5] = (byte) (value >> 40);
            case 5:
                data[4] = (byte) (value >> 32);
            case 4:
                data[3] = (byte) (value >> 24);
            case 3:
                data[2] = (byte) (value >> 16);
            case 2:
                data[1] = (byte) (value >> 8);
            case 1:
                data[0] = (byte) (value >> 0);
                break;
            default:
                throw new RuntimeException();

        }
        return data;
    }

    static byte[] encodeSint(int length, int value) {
        byte[] data = new byte[length];
        switch (length) {
            case 4:
                data[3] = (byte) (value >> 24);
            case 3:
                data[2] = (byte) (value >> 16);
            case 2:
                data[1] = (byte) (value >> 8);
            case 1:
                data[0] = (byte) (value >> 0);
                break;
            default:
                throw new RuntimeException();
        }
        return data;
    }

    @Override
    public boolean equals(Object o) {
        if (this == o) return true;
        if (o == null || getClass() != o.getClass()) return false;

        EncodedValue that = (EncodedValue) o;

        if (valueType != that.valueType) return false;
        if (value != null ? !value.equals(that.value) : that.value != null) return false;

        return true;
    }

    @Override
    public int hashCode() {
        int result = valueType;
        result = 31 * result + (value != null ? value.hashCode() : 0);
        return result;
    }

    public boolean isDefaultValueForType() {
        if (valueType == VALUE_NULL) {
            return true;
        }
        switch (valueType) {
            case VALUE_CHAR:
                Character c = (Character) this.value;
                return c.charValue() == 0;
            case VALUE_BYTE:
            case VALUE_INT:
            case VALUE_SHORT:
                return ((Number) this.value).intValue() == 0;
            case VALUE_LONG:
                return ((Number) this.value).longValue() == 0;
            case VALUE_FLOAT:
                return ((Number) this.value).floatValue() == 0.0f;
            case VALUE_DOUBLE:
                return ((Number) this.value).doubleValue() == 0.0;
            case VALUE_BOOLEAN:
                Boolean z = (Boolean) this.value;
                return Boolean.FALSE.equals(z);
        }
        return false;
    }

    protected int doPlace(int offset) {
        switch (valueType) {
            case VALUE_NULL:
            case VALUE_BOOLEAN:
                return offset;
            case VALUE_ARRAY: {
                EncodedArray ea = (EncodedArray) value;
                return ea.place(offset);
            }
            case VALUE_ANNOTATION: {
                EncodedAnnotation ea = (EncodedAnnotation) value;
                return ea.place(offset);
            }
            case VALUE_STRING:
            case VALUE_TYPE:
            case VALUE_FIELD:
            case VALUE_METHOD:
            case VALUE_ENUM:
            default:
                return offset + getValueArg() + 1;
        }
    }

    protected int getValueArg() {
        switch (valueType) {
            case VALUE_NULL:
            case VALUE_ANNOTATION:
            case VALUE_ARRAY:
                return 0;
            case VALUE_BOOLEAN:
                return Boolean.TRUE.equals(value) ? 1 : 0;
            case VALUE_BYTE:
                return 0;
            case VALUE_SHORT:
            case VALUE_INT:
                return lengthOfSint(((Number) value).intValue()) - 1;
            case VALUE_CHAR:
                return lengthOfUint(((Character) value).charValue()) - 1;
            case VALUE_LONG:
                return lengthOfSint(((Number) value).longValue()) - 1;
            case VALUE_DOUBLE:
                return lengthOfDouble(((Number) value).doubleValue()) - 1;
            case VALUE_FLOAT:
                return lengthOfFloat(((Number) value).floatValue()) - 1;
            case VALUE_STRING:
            case VALUE_TYPE:
            case VALUE_FIELD:
            case VALUE_METHOD:
            case VALUE_ENUM:
                BaseItem bi = (BaseItem) value;
                return lengthOfUint(bi.index) - 1;
        }
        return 0;
    }

    final public int place(int offset) {
        offset += 1;
        return doPlace(offset);
    }

    public void write(DataOut out) {
        int valueArg = getValueArg();
        out.ubyte("(value_arg << 5 | value_type", valueArg << 5 | valueType);
        switch (valueType) {
            case VALUE_NULL:
            case VALUE_BOOLEAN:
                // nop
                break;
            case VALUE_SHORT:
                out.bytes("value_short", encodeSint(valueArg + 1, (Short) value));
                break;
            case VALUE_CHAR:
                out.bytes("value_char", encodeSint(valueArg + 1, (Character) value));
                break;
            case VALUE_INT:
                out.bytes("value_int", encodeSint(valueArg + 1, (Integer) value));
                break;
            case VALUE_LONG:
                out.bytes("value_long", encodeLong(valueArg + 1, (Long) value));
                break;
            case VALUE_DOUBLE:
                out.bytes("value_double", writeRightZeroExtendedValue(valueArg+1,Double.doubleToLongBits( ((Number) value).doubleValue())));
                break;
            case VALUE_FLOAT:
                out.bytes("value_float", writeRightZeroExtendedValue(valueArg+1,((long)Float.floatToIntBits((((Number) value).floatValue())))<<32));
                break;
            case VALUE_STRING:
            case VALUE_TYPE:
            case VALUE_FIELD:
            case VALUE_METHOD:
            case VALUE_ENUM:
                out.bytes("value_xidx", encodeLong(valueArg + 1, ((BaseItem) value).index));
                break;
            case VALUE_ARRAY: {
                EncodedArray ea = (EncodedArray) value;
                ea.write(out);
            }
            break;
            case VALUE_ANNOTATION: {
                EncodedAnnotation ea = (EncodedAnnotation) value;
                ea.write(out);
            }
            break;
            case VALUE_BYTE: {
                out.ubyte("value_byte", (Byte) value);
                break;
            }
            default:
                throw new RuntimeException();

        }
    }

    private byte[] writeRightZeroExtendedValue(int requiredBytes, long value) {
        value >>= 64 - (requiredBytes * 8);
        byte[] s = new byte[requiredBytes];
        for (int i = 0; i < requiredBytes; i++) {
            s[i] = ((byte) value);
            value >>= 8;
        }
        return s;
    }

}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy