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.util.IntList;
21f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
22f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project/**
23f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project * This class maps one register space into another, with
24f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project * each mapping built up individually and added via addMapping()
25f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project */
2699409883d9c4c0ffb49b070ce307bb33a9dfe9f1The Android Open Source Projectpublic class BasicRegisterMapper extends RegisterMapper {
27f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    /** indexed by old register, containing new name */
28f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    private IntList oldToNew;
29f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
3099409883d9c4c0ffb49b070ce307bb33a9dfe9f1The Android Open Source Project    /** running count of used registers in new namespace */
31f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    private int runningCountNewRegisters;
32f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
33f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    /**
3499409883d9c4c0ffb49b070ce307bb33a9dfe9f1The Android Open Source Project     * Creates a new OneToOneRegisterMapper.
35de75089fb7216d19e9c22cce4dc62a49513477d3Carl Shapiro     *
36f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * @param countOldRegisters the number of registers in the old name space
37f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     */
38f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    public BasicRegisterMapper(int countOldRegisters) {
39f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        oldToNew = new IntList(countOldRegisters);
40f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    }
41f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
42f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    /** {@inheritDoc} */
43f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    @Override
44f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    public int getNewRegisterCount() {
45f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        return runningCountNewRegisters;
46f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    }
47f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
48f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    /** {@inheritDoc} */
49f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    @Override
50f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    public RegisterSpec map(RegisterSpec registerSpec) {
51f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        if (registerSpec == null) {
52f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            return null;
53f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        }
54f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
55f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        int newReg;
56f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        try {
57f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            newReg = oldToNew.get(registerSpec.getReg());
58f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        } catch (IndexOutOfBoundsException ex) {
59f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            newReg = -1;
60f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        }
61f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
62f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        if (newReg < 0) {
63f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            throw new RuntimeException("no mapping specified for register");
64f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        }
65f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
66f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        return registerSpec.withReg(newReg);
67f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    }
68f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
69f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    /**
70f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     * Returns the new-namespace mapping for the specified
7199409883d9c4c0ffb49b070ce307bb33a9dfe9f1The Android Open Source Project     * old-namespace register, or -1 if one exists.
72f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     *
7399409883d9c4c0ffb49b070ce307bb33a9dfe9f1The Android Open Source Project     * @param oldReg {@code >= 0;} old-namespace register
7499409883d9c4c0ffb49b070ce307bb33a9dfe9f1The Android Open Source Project     * @return new-namespace register or -1 if none
75f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project     */
76f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    public int oldToNew(int oldReg) {
7799409883d9c4c0ffb49b070ce307bb33a9dfe9f1The Android Open Source Project        if (oldReg >= oldToNew.size()) {
78f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            return -1;
79f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        }
8099409883d9c4c0ffb49b070ce307bb33a9dfe9f1The Android Open Source Project
81f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        return oldToNew.get(oldReg);
82f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    }
83f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
84f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    /** {@inheritDoc} */
85f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    public String toHuman() {
86f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        StringBuilder sb = new StringBuilder();
87f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
88f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        sb.append("Old\tNew\n");
89f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        int sz = oldToNew.size();
9099409883d9c4c0ffb49b070ce307bb33a9dfe9f1The Android Open Source Project
9199409883d9c4c0ffb49b070ce307bb33a9dfe9f1The 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    /**
10799409883d9c4c0ffb49b070ce307bb33a9dfe9f1The 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     *
11099409883d9c4c0ffb49b070ce307bb33a9dfe9f1The Android Open Source Project     * @param oldReg {@code >= 0;} old register
11199409883d9c4c0ffb49b070ce307bb33a9dfe9f1The Android Open Source Project     * @param newReg {@code >= 0;} new register
11299409883d9c4c0ffb49b070ce307bb33a9dfe9f1The Android Open Source Project     * @param category {@code 1..2;} width of reg
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        }
12199409883d9c4c0ffb49b070ce307bb33a9dfe9f1The Android Open Source Project
122f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        oldToNew.set(oldReg, newReg);
123f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project
124f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        if (runningCountNewRegisters < (newReg + category)) {
125f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project            runningCountNewRegisters = newReg + category;
126f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project        }
127f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project    }
128f6c387128427e121477c1b32ad35cdcaa5101ba3The Android Open Source Project}
129