|
1 /* This Source Code Form is subject to the terms of the Mozilla Public |
|
2 * License, v. 2.0. If a copy of the MPL was not distributed with this |
|
3 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */ |
|
4 |
|
5 package org.mozilla.gecko; |
|
6 |
|
7 import java.io.BufferedOutputStream; |
|
8 import java.io.BufferedReader; |
|
9 import java.io.DataOutputStream; |
|
10 import java.io.File; |
|
11 import java.io.FileOutputStream; |
|
12 import java.io.FileReader; |
|
13 import java.io.FileWriter; |
|
14 import java.io.IOException; |
|
15 import java.io.PrintWriter; |
|
16 import java.nio.IntBuffer; |
|
17 import java.util.HashMap; |
|
18 import java.util.List; |
|
19 import java.util.Map; |
|
20 |
|
21 import org.json.JSONException; |
|
22 import org.json.JSONObject; |
|
23 import org.mozilla.gecko.gfx.LayerView; |
|
24 import org.mozilla.gecko.gfx.PanningPerfAPI; |
|
25 import org.mozilla.gecko.util.GeckoEventListener; |
|
26 |
|
27 import android.app.Activity; |
|
28 import android.util.Log; |
|
29 import android.view.View; |
|
30 |
|
31 import com.jayway.android.robotium.solo.Solo; |
|
32 |
|
33 public class FennecNativeDriver implements Driver { |
|
34 private static final int FRAME_TIME_THRESHOLD = 25; // allow 25ms per frame (40fps) |
|
35 |
|
36 private Activity mActivity; |
|
37 private Solo mSolo; |
|
38 private String mRootPath; |
|
39 |
|
40 private static String mLogFile = null; |
|
41 private static LogLevel mLogLevel = LogLevel.INFO; |
|
42 |
|
43 public enum LogLevel { |
|
44 DEBUG(1), |
|
45 INFO(2), |
|
46 WARN(3), |
|
47 ERROR(4); |
|
48 |
|
49 private int mValue; |
|
50 LogLevel(int value) { |
|
51 mValue = value; |
|
52 } |
|
53 public boolean isEnabled(LogLevel configuredLevel) { |
|
54 return mValue >= configuredLevel.getValue(); |
|
55 } |
|
56 private int getValue() { |
|
57 return mValue; |
|
58 } |
|
59 } |
|
60 |
|
61 public FennecNativeDriver(Activity activity, Solo robocop, String rootPath) { |
|
62 mActivity = activity; |
|
63 mSolo = robocop; |
|
64 mRootPath = rootPath; |
|
65 } |
|
66 |
|
67 //Information on the location of the Gecko Frame. |
|
68 private boolean mGeckoInfo = false; |
|
69 private int mGeckoTop = 100; |
|
70 private int mGeckoLeft = 0; |
|
71 private int mGeckoHeight= 700; |
|
72 private int mGeckoWidth = 1024; |
|
73 |
|
74 private void getGeckoInfo() { |
|
75 View geckoLayout = mActivity.findViewById(R.id.gecko_layout); |
|
76 if (geckoLayout != null) { |
|
77 int[] pos = new int[2]; |
|
78 geckoLayout.getLocationOnScreen(pos); |
|
79 mGeckoTop = pos[1]; |
|
80 mGeckoLeft = pos[0]; |
|
81 mGeckoWidth = geckoLayout.getWidth(); |
|
82 mGeckoHeight = geckoLayout.getHeight(); |
|
83 mGeckoInfo = true; |
|
84 } else { |
|
85 throw new RoboCopException("Unable to find view gecko_layout"); |
|
86 } |
|
87 } |
|
88 |
|
89 public int getGeckoTop() { |
|
90 if (!mGeckoInfo) { |
|
91 getGeckoInfo(); |
|
92 } |
|
93 return mGeckoTop; |
|
94 } |
|
95 |
|
96 public int getGeckoLeft() { |
|
97 if (!mGeckoInfo) { |
|
98 getGeckoInfo(); |
|
99 } |
|
100 return mGeckoLeft; |
|
101 } |
|
102 |
|
103 public int getGeckoHeight() { |
|
104 if (!mGeckoInfo) { |
|
105 getGeckoInfo(); |
|
106 } |
|
107 return mGeckoHeight; |
|
108 } |
|
109 |
|
110 public int getGeckoWidth() { |
|
111 if (!mGeckoInfo) { |
|
112 getGeckoInfo(); |
|
113 } |
|
114 return mGeckoWidth; |
|
115 } |
|
116 |
|
117 /** Find the element with given id. |
|
118 * |
|
119 * @return An Element representing the view, or null if the view is not found. |
|
120 */ |
|
121 public Element findElement(Activity activity, int id) { |
|
122 return new FennecNativeElement(id, activity); |
|
123 } |
|
124 |
|
125 public void startFrameRecording() { |
|
126 PanningPerfAPI.startFrameTimeRecording(); |
|
127 } |
|
128 |
|
129 public int stopFrameRecording() { |
|
130 final List<Long> frames = PanningPerfAPI.stopFrameTimeRecording(); |
|
131 int badness = 0; |
|
132 for (int i = 1; i < frames.size(); i++) { |
|
133 long frameTime = frames.get(i) - frames.get(i - 1); |
|
134 int delay = (int)(frameTime - FRAME_TIME_THRESHOLD); |
|
135 // for each frame we miss, add the square of the delay. This |
|
136 // makes large delays much worse than small delays. |
|
137 if (delay > 0) { |
|
138 badness += delay * delay; |
|
139 } |
|
140 } |
|
141 |
|
142 // Don't do any averaging of the numbers because really we want to |
|
143 // know how bad the jank was at its worst |
|
144 return badness; |
|
145 } |
|
146 |
|
147 public void startCheckerboardRecording() { |
|
148 PanningPerfAPI.startCheckerboardRecording(); |
|
149 } |
|
150 |
|
151 public float stopCheckerboardRecording() { |
|
152 final List<Float> checkerboard = PanningPerfAPI.stopCheckerboardRecording(); |
|
153 float total = 0; |
|
154 for (float val : checkerboard) { |
|
155 total += val; |
|
156 } |
|
157 return total * 100.0f; |
|
158 } |
|
159 |
|
160 private LayerView getSurfaceView() { |
|
161 final LayerView layerView = mSolo.getView(LayerView.class, 0); |
|
162 |
|
163 if (layerView == null) { |
|
164 log(LogLevel.WARN, "getSurfaceView could not find LayerView"); |
|
165 for (final View v : mSolo.getViews()) { |
|
166 log(LogLevel.WARN, " View: " + v); |
|
167 } |
|
168 } |
|
169 return layerView; |
|
170 } |
|
171 |
|
172 public PaintedSurface getPaintedSurface() { |
|
173 final LayerView view = getSurfaceView(); |
|
174 if (view == null) { |
|
175 return null; |
|
176 } |
|
177 |
|
178 final IntBuffer pixelBuffer = view.getPixels(); |
|
179 |
|
180 // now we need to (1) flip the image, because GL likes to do things up-side-down, |
|
181 // and (2) rearrange the bits from AGBR-8888 to ARGB-8888. |
|
182 int w = view.getWidth(); |
|
183 int h = view.getHeight(); |
|
184 pixelBuffer.position(0); |
|
185 String mapFile = mRootPath + "/pixels.map"; |
|
186 |
|
187 FileOutputStream fos = null; |
|
188 BufferedOutputStream bos = null; |
|
189 DataOutputStream dos = null; |
|
190 try { |
|
191 fos = new FileOutputStream(mapFile); |
|
192 bos = new BufferedOutputStream(fos); |
|
193 dos = new DataOutputStream(bos); |
|
194 |
|
195 for (int y = h - 1; y >= 0; y--) { |
|
196 for (int x = 0; x < w; x++) { |
|
197 int agbr = pixelBuffer.get(); |
|
198 dos.writeInt((agbr & 0xFF00FF00) | ((agbr >> 16) & 0x000000FF) | ((agbr << 16) & 0x00FF0000)); |
|
199 } |
|
200 } |
|
201 } catch (IOException e) { |
|
202 throw new RoboCopException("exception with pixel writer on file: " + mapFile); |
|
203 } finally { |
|
204 try { |
|
205 if (dos != null) { |
|
206 dos.flush(); |
|
207 dos.close(); |
|
208 } |
|
209 // closing dos automatically closes bos |
|
210 if (fos != null) { |
|
211 fos.flush(); |
|
212 fos.close(); |
|
213 } |
|
214 } catch (IOException e) { |
|
215 log(LogLevel.ERROR, e); |
|
216 throw new RoboCopException("exception closing pixel writer on file: " + mapFile); |
|
217 } |
|
218 } |
|
219 return new PaintedSurface(mapFile, w, h); |
|
220 } |
|
221 |
|
222 public int mHeight=0; |
|
223 public int mScrollHeight=0; |
|
224 public int mPageHeight=10; |
|
225 |
|
226 public int getScrollHeight() { |
|
227 return mScrollHeight; |
|
228 } |
|
229 public int getPageHeight() { |
|
230 return mPageHeight; |
|
231 } |
|
232 public int getHeight() { |
|
233 return mHeight; |
|
234 } |
|
235 |
|
236 public void setupScrollHandling() { |
|
237 GeckoAppShell.registerEventListener("robocop:scroll", new GeckoEventListener() { |
|
238 @Override |
|
239 public void handleMessage(final String event, final JSONObject message) { |
|
240 try { |
|
241 mScrollHeight = message.getInt("y"); |
|
242 mHeight = message.getInt("cheight"); |
|
243 // We don't want a height of 0. That means it's a bad response. |
|
244 if (mHeight > 0) { |
|
245 mPageHeight = message.getInt("height"); |
|
246 } |
|
247 } catch (JSONException e) { |
|
248 FennecNativeDriver.log(FennecNativeDriver.LogLevel.WARN, |
|
249 "WARNING: ScrollReceived, but message does not contain " + |
|
250 "expected fields: " + e); |
|
251 } |
|
252 } |
|
253 }); |
|
254 } |
|
255 |
|
256 /** |
|
257 * Takes a filename, loads the file, and returns a string version of the entire file. |
|
258 */ |
|
259 public static String getFile(String filename) |
|
260 { |
|
261 StringBuilder text = new StringBuilder(); |
|
262 |
|
263 BufferedReader br = null; |
|
264 try { |
|
265 br = new BufferedReader(new FileReader(filename)); |
|
266 String line; |
|
267 |
|
268 while ((line = br.readLine()) != null) { |
|
269 text.append(line); |
|
270 text.append('\n'); |
|
271 } |
|
272 } catch (IOException e) { |
|
273 log(LogLevel.ERROR, e); |
|
274 } finally { |
|
275 try { |
|
276 br.close(); |
|
277 } catch (IOException e) { |
|
278 } |
|
279 } |
|
280 return text.toString(); |
|
281 } |
|
282 |
|
283 /** |
|
284 * Takes a string of "key=value" pairs split by \n and creates a hash table. |
|
285 */ |
|
286 public static Map<String, String> convertTextToTable(String data) |
|
287 { |
|
288 HashMap<String, String> retVal = new HashMap<String, String>(); |
|
289 |
|
290 String[] lines = data.split("\n"); |
|
291 for (int i = 0; i < lines.length; i++) { |
|
292 String[] parts = lines[i].split("=", 2); |
|
293 retVal.put(parts[0].trim(), parts[1].trim()); |
|
294 } |
|
295 return retVal; |
|
296 } |
|
297 |
|
298 public static void logAllStackTraces(LogLevel level) { |
|
299 StringBuffer sb = new StringBuffer(); |
|
300 sb.append("Dumping ALL the threads!\n"); |
|
301 Map<Thread, StackTraceElement[]> allStacks = Thread.getAllStackTraces(); |
|
302 for (Thread t : allStacks.keySet()) { |
|
303 sb.append(t.toString()).append('\n'); |
|
304 for (StackTraceElement ste : allStacks.get(t)) { |
|
305 sb.append(ste.toString()).append('\n'); |
|
306 } |
|
307 sb.append('\n'); |
|
308 } |
|
309 log(level, sb.toString()); |
|
310 } |
|
311 |
|
312 /** |
|
313 * Set the filename used for logging. If the file already exists, delete it |
|
314 * as a safe-guard against accidentally appending to an old log file. |
|
315 */ |
|
316 public static void setLogFile(String filename) { |
|
317 mLogFile = filename; |
|
318 File file = new File(mLogFile); |
|
319 if (file.exists()) { |
|
320 file.delete(); |
|
321 } |
|
322 } |
|
323 |
|
324 public static void setLogLevel(LogLevel level) { |
|
325 mLogLevel = level; |
|
326 } |
|
327 |
|
328 public static void log(LogLevel level, String message) { |
|
329 log(level, message, null); |
|
330 } |
|
331 |
|
332 public static void log(LogLevel level, Throwable t) { |
|
333 log(level, null, t); |
|
334 } |
|
335 |
|
336 public static void log(LogLevel level, String message, Throwable t) { |
|
337 if (mLogFile == null) { |
|
338 assert(false); |
|
339 } |
|
340 |
|
341 if (level.isEnabled(mLogLevel)) { |
|
342 PrintWriter pw = null; |
|
343 try { |
|
344 pw = new PrintWriter(new FileWriter(mLogFile, true)); |
|
345 if (message != null) { |
|
346 pw.println(message); |
|
347 } |
|
348 if (t != null) { |
|
349 t.printStackTrace(pw); |
|
350 } |
|
351 } catch (IOException ioe) { |
|
352 Log.e("Robocop", "exception with file writer on: " + mLogFile); |
|
353 } finally { |
|
354 pw.close(); |
|
355 } |
|
356 // PrintWriter doesn't throw IOE but sets an error flag instead, |
|
357 // so check for that |
|
358 if (pw.checkError()) { |
|
359 Log.e("Robocop", "exception with file writer on: " + mLogFile); |
|
360 } |
|
361 } |
|
362 |
|
363 if (level == LogLevel.INFO) { |
|
364 Log.i("Robocop", message, t); |
|
365 } else if (level == LogLevel.DEBUG) { |
|
366 Log.d("Robocop", message, t); |
|
367 } else if (level == LogLevel.WARN) { |
|
368 Log.w("Robocop", message, t); |
|
369 } else if (level == LogLevel.ERROR) { |
|
370 Log.e("Robocop", message, t); |
|
371 } |
|
372 } |
|
373 } |