1917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul/*
2917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul * Copyright (C) 2007 The Android Open Source Project
3917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul *
4917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul * Licensed under the Apache License, Version 2.0 (the "License");
5917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul * you may not use this file except in compliance with the License.
6917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul * You may obtain a copy of the License at
7917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul *
8917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul *      http://www.apache.org/licenses/LICENSE-2.0
9917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul *
10917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul * Unless required by applicable law or agreed to in writing, software
11917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul * distributed under the License is distributed on an "AS IS" BASIS,
12917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul * See the License for the specific language governing permissions and
14917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul * limitations under the License.
15917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul */
16917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul
17917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgulpackage com.android.dexgen.dex.code.form;
18917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul
19917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgulimport com.android.dexgen.dex.code.DalvInsn;
20917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgulimport com.android.dexgen.dex.code.HighRegisterPrefix;
21917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgulimport com.android.dexgen.dex.code.InsnFormat;
22917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgulimport com.android.dexgen.dex.code.SimpleInsn;
23917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgulimport com.android.dexgen.rop.code.RegisterSpec;
24917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgulimport com.android.dexgen.rop.code.RegisterSpecList;
25917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgulimport com.android.dexgen.util.AnnotatedOutput;
26917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul
27917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul/**
28917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul * Instruction format {@code 12x}. See the instruction format spec
29917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul * for details.
30917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul */
31917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgulpublic final class Form12x extends InsnFormat {
32917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    /** {@code non-null;} unique instance of this class */
33917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    public static final InsnFormat THE_ONE = new Form12x();
34917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul
35917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    /**
36917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul     * Constructs an instance. This class is not publicly
37917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul     * instantiable. Use {@link #THE_ONE}.
38917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul     */
39917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    private Form12x() {
40917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul        // This space intentionally left blank.
41917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    }
42917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul
43917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    /** {@inheritDoc} */
44917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    @Override
45917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    public String insnArgString(DalvInsn insn) {
46917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul        RegisterSpecList regs = insn.getRegisters();
47917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul        int sz = regs.size();
48917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul
49917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul        /*
50917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul         * The (sz - 2) and (sz - 1) below makes this code work for
51917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul         * both the two- and three-register ops. (See "case 3" in
52917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul         * isCompatible(), below.)
53917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul         */
54917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul
55917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul        return regs.get(sz - 2).regString() + ", " +
56917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul            regs.get(sz - 1).regString();
57917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    }
58917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul
59917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    /** {@inheritDoc} */
60917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    @Override
61917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    public String insnCommentString(DalvInsn insn, boolean noteIndices) {
62917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul        // This format has no comment.
63917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul        return "";
64917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    }
65917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul
66917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    /** {@inheritDoc} */
67917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    @Override
68917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    public int codeSize() {
69917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul        return 1;
70917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    }
71917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul
72917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    /** {@inheritDoc} */
73917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    @Override
74917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    public boolean isCompatible(DalvInsn insn) {
75917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul        if (!(insn instanceof SimpleInsn)) {
76917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul            return false;
77917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul        }
78917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul
79917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul        RegisterSpecList regs = insn.getRegisters();
80917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul        RegisterSpec rs1;
81917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul        RegisterSpec rs2;
82917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul
83917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul        switch (regs.size()) {
84917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul            case 2: {
85917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul                rs1 = regs.get(0);
86917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul                rs2 = regs.get(1);
87917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul                break;
88917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul            }
89917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul            case 3: {
90917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul                /*
91917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul                 * This format is allowed for ops that are effectively
92917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul                 * 3-arg but where the first two args are identical.
93917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul                 */
94917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul                rs1 = regs.get(1);
95917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul                rs2 = regs.get(2);
96917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul                if (rs1.getReg() != regs.get(0).getReg()) {
97917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul                    return false;
98917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul                }
99917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul                break;
100917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul            }
101917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul            default: {
102917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul                return false;
103917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul            }
104917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul        }
105917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul
106917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul        return unsignedFitsInNibble(rs1.getReg()) &&
107917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul            unsignedFitsInNibble(rs2.getReg());
108917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    }
109917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul
110917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    /** {@inheritDoc} */
111917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    @Override
112917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    public InsnFormat nextUp() {
113917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul        return Form22x.THE_ONE;
114917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    }
115917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul
116917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    /** {@inheritDoc} */
117917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    @Override
118917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    public void writeTo(AnnotatedOutput out, DalvInsn insn) {
119917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul        RegisterSpecList regs = insn.getRegisters();
120917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul        int sz = regs.size();
121917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul
122917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul        /*
123917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul         * The (sz - 2) and (sz - 1) below makes this code work for
124917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul         * both the two- and three-register ops. (See "case 3" in
125917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul         * isCompatible(), above.)
126917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul         */
127917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul
128917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul        write(out, opcodeUnit(insn,
129917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul                              makeByte(regs.get(sz - 2).getReg(),
130917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul                                       regs.get(sz - 1).getReg())));
131917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul    }
132917cb222329ee8c035c3ffaf947e4265761b9367Piotr Gurgul}
133