1579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson/*
2579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson * Copyright (C) 2007 The Android Open Source Project
3579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson *
4579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson * Licensed under the Apache License, Version 2.0 (the "License");
5579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson * you may not use this file except in compliance with the License.
6579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson * You may obtain a copy of the License at
7579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson *
8579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson *      http://www.apache.org/licenses/LICENSE-2.0
9579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson *
10579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson * Unless required by applicable law or agreed to in writing, software
11579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson * distributed under the License is distributed on an "AS IS" BASIS,
12579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson * See the License for the specific language governing permissions and
14579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson * limitations under the License.
15579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson */
16579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson
17579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilsonpackage com.android.dx.dex.code.form;
18579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson
19579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilsonimport com.android.dx.dex.code.CstInsn;
20579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilsonimport com.android.dx.dex.code.DalvInsn;
21579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilsonimport com.android.dx.dex.code.InsnFormat;
22579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilsonimport com.android.dx.rop.code.RegisterSpecList;
23579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilsonimport com.android.dx.rop.cst.Constant;
24579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilsonimport com.android.dx.rop.cst.CstLiteralBits;
25579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilsonimport com.android.dx.util.AnnotatedOutput;
26579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson
27579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilsonimport java.util.BitSet;
28579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson
29579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson/**
30579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson * Instruction format {@code 21h}. See the instruction format spec
31579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson * for details.
32579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson */
33579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilsonpublic final class Form21h extends InsnFormat {
34579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    /** {@code non-null;} unique instance of this class */
35579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    public static final InsnFormat THE_ONE = new Form21h();
36579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson
37579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    /**
38579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson     * Constructs an instance. This class is not publicly
39579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson     * instantiable. Use {@link #THE_ONE}.
40579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson     */
41579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    private Form21h() {
42579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        // This space intentionally left blank.
43579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    }
44579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson
45579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    /** {@inheritDoc} */
46579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    @Override
47579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    public String insnArgString(DalvInsn insn) {
48579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        RegisterSpecList regs = insn.getRegisters();
49579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        CstLiteralBits value = (CstLiteralBits) ((CstInsn) insn).getConstant();
50579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson
51579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        return regs.get(0).regString() + ", " + literalBitsString(value);
52579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    }
53579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson
54579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    /** {@inheritDoc} */
55579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    @Override
56579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    public String insnCommentString(DalvInsn insn, boolean noteIndices) {
57579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        RegisterSpecList regs = insn.getRegisters();
58579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        CstLiteralBits value = (CstLiteralBits) ((CstInsn) insn).getConstant();
59579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson
60579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        return
61579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson            literalBitsComment(value,
62579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson                    (regs.get(0).getCategory() == 1) ? 32 : 64);
63579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    }
64579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson
65579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    /** {@inheritDoc} */
66579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    @Override
67579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    public int codeSize() {
68579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        return 2;
69579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    }
70579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson
71579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    /** {@inheritDoc} */
72579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    @Override
73579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    public boolean isCompatible(DalvInsn insn) {
74579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        RegisterSpecList regs = insn.getRegisters();
75579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        if (!((insn instanceof CstInsn) &&
76579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson              (regs.size() == 1) &&
77579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson              unsignedFitsInByte(regs.get(0).getReg()))) {
78579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson            return false;
79579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        }
80579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson
81579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        CstInsn ci = (CstInsn) insn;
82579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        Constant cst = ci.getConstant();
83579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson
84579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        if (!(cst instanceof CstLiteralBits)) {
85579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson            return false;
86579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        }
87579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson
88579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        CstLiteralBits cb = (CstLiteralBits) cst;
89579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson
90579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        // Where the high bits are depends on the category of the target.
91579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        if (regs.get(0).getCategory() == 1) {
92579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson            int bits = cb.getIntBits();
93579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson            return ((bits & 0xffff) == 0);
94579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        } else {
95579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson            long bits = cb.getLongBits();
96579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson            return ((bits & 0xffffffffffffL) == 0);
97579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        }
98579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    }
99579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson
100579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    /** {@inheritDoc} */
101579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    @Override
102579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    public BitSet compatibleRegs(DalvInsn insn) {
103579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        RegisterSpecList regs = insn.getRegisters();
104579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        BitSet bits = new BitSet(1);
105579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson
106579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        bits.set(0, unsignedFitsInByte(regs.get(0).getReg()));
107579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        return bits;
108579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    }
109579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson
110579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    /** {@inheritDoc} */
111579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    @Override
112579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    public void writeTo(AnnotatedOutput out, DalvInsn insn) {
113579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        RegisterSpecList regs = insn.getRegisters();
114579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        CstLiteralBits cb = (CstLiteralBits) ((CstInsn) insn).getConstant();
115579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        short bits;
116579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson
117579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        // Where the high bits are depends on the category of the target.
118579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        if (regs.get(0).getCategory() == 1) {
119579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson            bits = (short) (cb.getIntBits() >>> 16);
120579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        } else {
121579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson            bits = (short) (cb.getLongBits() >>> 48);
122579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        }
123579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson
124579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson        write(out, opcodeUnit(insn, regs.get(0).getReg()), bits);
125579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson    }
126579d7739c53a2707ad711a2d2cae46d7d782f06Jesse Wilson}
127