1/*
2 * ProGuard -- shrinking, optimization, obfuscation, and preverification
3 *             of Java bytecode.
4 *
5 * Copyright (c) 2002-2014 Eric Lafortune (eric@graphics.cornell.edu)
6 *
7 * This program is free software; you can redistribute it and/or modify it
8 * under the terms of the GNU General Public License as published by the Free
9 * Software Foundation; either version 2 of the License, or (at your option)
10 * any later version.
11 *
12 * This program is distributed in the hope that it will be useful, but WITHOUT
13 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
14 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
15 * more details.
16 *
17 * You should have received a copy of the GNU General Public License along
18 * with this program; if not, write to the Free Software Foundation, Inc.,
19 * 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
20 */
21package proguard.obfuscate;
22
23import java.io.*;
24
25
26/**
27 * This class can parse mapping files and invoke a processor for each of the
28 * mapping entries.
29 *
30 * @author Eric Lafortune
31 */
32public class MappingReader
33{
34    private final File mappingFile;
35
36
37    public MappingReader(File mappingFile)
38    {
39        this.mappingFile = mappingFile;
40    }
41
42
43    /**
44     * Reads the mapping file, presenting all of the encountered mapping entries
45     * to the given processor.
46     */
47    public void pump(MappingProcessor mappingProcessor) throws IOException
48    {
49        LineNumberReader reader = new LineNumberReader(
50                                  new BufferedReader(
51                                  new FileReader(mappingFile)));
52        try
53        {
54            String className = null;
55
56            // Read the subsequent class mappings and class member mappings.
57            while (true)
58            {
59                String line = reader.readLine();
60
61                if (line == null)
62                {
63                    break;
64                }
65
66                line = line.trim();
67
68                // The distinction between a class mapping and a class
69                // member mapping is the initial whitespace.
70                if (line.endsWith(":"))
71                {
72                    // Process the class mapping and remember the class's
73                    // old name.
74                    className = processClassMapping(line, mappingProcessor);
75                }
76                else if (className != null)
77                {
78                    // Process the class member mapping, in the context of the
79                    // current old class name.
80                    processClassMemberMapping(className, line, mappingProcessor);
81                }
82            }
83        }
84        catch (IOException ex)
85        {
86            throw new IOException("Can't process mapping file (" + ex.getMessage() + ")");
87        }
88        finally
89        {
90            try
91            {
92                reader.close();
93            }
94            catch (IOException ex)
95            {
96                // This shouldn't happen.
97            }
98        }
99    }
100
101
102    /**
103     * Parses the given line with a class mapping and processes the
104     * results with the given mapping processor. Returns the old class name,
105     * or null if any subsequent class member lines can be ignored.
106     */
107    private String processClassMapping(String           line,
108                                       MappingProcessor mappingProcessor)
109    {
110        // See if we can parse "___ -> ___:", containing the original
111        // class name and the new class name.
112
113        int arrowIndex = line.indexOf("->");
114        if (arrowIndex < 0)
115        {
116            return null;
117        }
118
119        int colonIndex = line.indexOf(':', arrowIndex + 2);
120        if (colonIndex < 0)
121        {
122            return null;
123        }
124
125        // Extract the elements.
126        String className    = line.substring(0, arrowIndex).trim();
127        String newClassName = line.substring(arrowIndex + 2, colonIndex).trim();
128
129        // Process this class name mapping.
130        boolean interested = mappingProcessor.processClassMapping(className, newClassName);
131
132        return interested ? className : null;
133    }
134
135
136    /**
137     * Parses the given line with a class member mapping and processes the
138     * results with the given mapping processor.
139     */
140    private void processClassMemberMapping(String           className,
141                                           String           line,
142                                           MappingProcessor mappingProcessor)
143    {
144        // See if we can parse "___:___:___ ___(___) -> ___",
145        // containing the optional line numbers, the return type, the original
146        // field/method name, optional arguments, and the new field/method name.
147
148        int colonIndex1    =                           line.indexOf(':');
149        int colonIndex2    = colonIndex1    < 0 ? -1 : line.indexOf(':', colonIndex1    + 1);
150        int spaceIndex     =                           line.indexOf(' ', colonIndex2    + 2);
151        int argumentIndex1 =                           line.indexOf('(', spaceIndex     + 1);
152        int argumentIndex2 = argumentIndex1 < 0 ? -1 : line.indexOf(')', argumentIndex1 + 1);
153        int arrowIndex     =                           line.indexOf("->", Math.max(spaceIndex, argumentIndex2) + 1);
154
155        if (spaceIndex < 0 ||
156            arrowIndex < 0)
157        {
158            return;
159        }
160
161        // Extract the elements.
162        String type    = line.substring(colonIndex2 + 1, spaceIndex).trim();
163        String name    = line.substring(spaceIndex + 1, argumentIndex1 >= 0 ? argumentIndex1 : arrowIndex).trim();
164        String newName = line.substring(arrowIndex + 2).trim();
165
166        // Process this class member mapping.
167        if (type.length()    > 0 &&
168            name.length()    > 0 &&
169            newName.length() > 0)
170        {
171            // Is it a field or a method?
172            if (argumentIndex2 < 0)
173            {
174                mappingProcessor.processFieldMapping(className, type, name, newName);
175            }
176            else
177            {
178                int firstLineNumber = 0;
179                int lastLineNumber  = 0;
180
181                if (colonIndex2 > 0)
182                {
183                    firstLineNumber = Integer.parseInt(line.substring(0, colonIndex1).trim());
184                    lastLineNumber  = Integer.parseInt(line.substring(colonIndex1 + 1, colonIndex2).trim());
185                }
186
187                String arguments = line.substring(argumentIndex1 + 1, argumentIndex2).trim();
188
189                mappingProcessor.processMethodMapping(className,
190                                                      firstLineNumber,
191                                                      lastLineNumber,
192                                                      type,
193                                                      name,
194                                                      arguments,
195                                                      newName);
196            }
197        }
198    }
199}
200