BasicRegisterMapper.java revision f6c387128427e121477c1b32ad35cdcaa5101ba3
1f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project/*
2f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project * Copyright (C) 2007 The Android Open Source Project
3f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project *
4f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project * Licensed under the Apache License, Version 2.0 (the "License");
5f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project * you may not use this file except in compliance with the License.
6f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project * You may obtain a copy of the License at
7f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project *
8f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project *      http://www.apache.org/licenses/LICENSE-2.0
9f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project *
10f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project * Unless required by applicable law or agreed to in writing, software
11f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project * distributed under the License is distributed on an "AS IS" BASIS,
12f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project * See the License for the specific language governing permissions and
14f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project * limitations under the License.
15f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project */
16f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
17f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Projectpackage com.android.dx.ssa;
18f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
19f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Projectimport com.android.dx.rop.code.RegisterSpec;
20f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Projectimport com.android.dx.rop.code.RegisterSpecList;
21f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Projectimport com.android.dx.util.IntList;
22f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
23f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project/**
24f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project * This class maps one register space into another, with
25f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project * each mapping built up individually and added via addMapping()
26f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project */
27f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Projectpublic class BasicRegisterMapper
28f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        extends RegisterMapper {
29f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
30f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    /** indexed by old register, containing new name */
31f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    private IntList oldToNew;
32f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
33f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    /** Running count of used registers in new namespace */
34f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    private int runningCountNewRegisters;
35f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
36f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    /**
37f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * Creates a new OneToOneRegisterMapper
38f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * @param countOldRegisters the number of registers in the old name space
39f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     */
40f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    public BasicRegisterMapper(int countOldRegisters) {
41f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        oldToNew = new IntList(countOldRegisters);
42f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    }
43f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
44f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    /** {@inheritDoc} */
45f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    @Override
46f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    public int getNewRegisterCount() {
47f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        return runningCountNewRegisters;
48f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    }
49f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
50f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    /** {@inheritDoc} */
51f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    @Override
52f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    public RegisterSpec map(RegisterSpec registerSpec) {
53f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        if (registerSpec == null) {
54f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            return null;
55f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        }
56f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
57f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        int newReg;
58f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        try {
59f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            newReg = oldToNew.get(registerSpec.getReg());
60f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        } catch (IndexOutOfBoundsException ex) {
61f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            newReg = -1;
62f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        }
63f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
64f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        if (newReg < 0) {
65f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            throw new RuntimeException("no mapping specified for register");
66f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        }
67f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
68f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        return registerSpec.withReg(newReg);
69f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    }
70f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
71f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    /**
72f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * Returns the new-namespace mapping for the specified
73f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * old-namespace register, or -1 if one exists
74f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     *
75f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * @param oldReg &gt;=0; old-namespace register
76f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * @return new-namespace register or -1 if none.
77f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     */
78f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    public int oldToNew(int oldReg) {
79f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        if(oldReg >= oldToNew.size()) {
80f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            return -1;
81f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        }
82f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        return oldToNew.get(oldReg);
83f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    }
84f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
85f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    /** {@inheritDoc} */
86f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    public String toHuman() {
87f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        StringBuilder sb = new StringBuilder();
88f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
89f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        sb.append("Old\tNew\n");
90f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        int sz = oldToNew.size();
91f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        for(int i = 0; i < sz; i++) {
92f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            sb.append(i);
93f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            sb.append('\t');
94f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            sb.append(oldToNew.get(i));
95f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            sb.append('\n');
96f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        }
97f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
98f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        sb.append("new reg count:");
99f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
100f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        sb.append(runningCountNewRegisters);
101f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        sb.append('\n');
102f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
103f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        return sb.toString();
104f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    }
105f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
106f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    /**
107f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * adds a mapping to the mapper. If oldReg has already been mapped,
108f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * overwrites previous mapping with new mapping.
109f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     *
110f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * @param oldReg >=0
111f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * @param newReg >=0
112f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * @param category width of reg (1 or 2)
113f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     */
114f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    public void addMapping(int oldReg, int newReg, int category) {
115f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        if (oldReg >= oldToNew.size()) {
116f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            // expand the array as necessary
117f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            for (int i = oldReg - oldToNew.size(); i >= 0; i--) {
118f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project                oldToNew.add(-1);
119f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            }
120f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        }
121f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        oldToNew.set(oldReg, newReg);
122f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
123f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        if (runningCountNewRegisters < (newReg + category)) {
124f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            runningCountNewRegisters = newReg + category;
125f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        }
126f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    }
127f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project}
128