2003/11/03 07:36:40
[org.ibex.core.git] / src / org / xwt / translators / Freetype.java
1 package org.xwt.translators;
2 import org.xwt.*;
3 import org.xwt.util.*;
4 import java.io.*;
5 import java.util.zip.*;
6 import java.util.*;
7 import org.bouncycastle.util.encoders.Base64;
8
9 // FEATURE: use streams, not memoryfont's
10 // FEATURE: kerning pairs
11 public class Freetype {
12
13     public Freetype() { }
14
15     private static byte[] image = null;
16     private static final int FONT_RESERVED = 256*1024;
17
18     private org.xwt.mips.Interpreter vm = null;
19
20     private Res loadedStream = null;
21
22     public void loadFontByteStream(Res res) {
23         try {
24             Log.log(this, "loading font " + res.getDescriptiveName());
25             loadedStream = res;
26             InputStream is = res.getInputStream();
27             byte[] fontstream = InputStreamToByteArray.convert(is);
28             if (image == null) image = InputStreamToByteArray.convert(Main.builtin.getInputStream("freetype.mips"));
29             vm = new org.xwt.mips.Interpreter(image);
30             int baseAddr = vm.sbrk(FONT_RESERVED);
31             vm.copyout(fontstream, baseAddr, fontstream.length);
32             vm.setUserInfo(0, baseAddr);
33             vm.setUserInfo(1, fontstream.length);
34             vm.start(new String[]{ "freetype.mips" });
35             vm.execute();
36         } catch (Exception e) {
37             Log.log(this, e);
38         }
39     }
40
41     public synchronized void renderGlyph(Font.Glyph glyph) throws IOException {
42         int width = 0;
43         int height = 0;
44         byte[] data = null;
45         String key = glyph.font.res.getDescriptiveName() + ":" + glyph.c;
46         key = new String(Base64.encode(key.getBytes()));
47         File cacheFile = new java.io.File(System.getProperty("user.home") +
48                                           java.io.File.separatorChar + ".xwt" +
49                                           java.io.File.separatorChar + "caches" +
50                                           java.io.File.separatorChar + "glyphs" +
51                                           java.io.File.separatorChar +
52                                           key);
53         new java.io.File(cacheFile.getParent()).mkdirs();
54
55         if (cacheFile.exists()) {
56             DataInputStream dis = new DataInputStream(new FileInputStream(cacheFile));
57             width = dis.readInt();
58             height = dis.readInt();
59             glyph.font.max_ascent = dis.readInt();
60             glyph.font.max_descent = dis.readInt();
61             glyph.baseline = dis.readInt();
62             glyph.advance = dis.readInt();
63             data = new byte[width * height];
64             if (width != 0 && height != 0) dis.readFully(data);
65             
66         } else try {
67             System.out.println("cache miss!");
68             if (loadedStream != glyph.font.res) loadFontByteStream(glyph.font.res);
69             vm.setUserInfo(2, (int)glyph.c);
70             vm.setUserInfo(3, (int)glyph.c);
71             vm.setUserInfo(4, glyph.font.pointsize);
72             long start = System.currentTimeMillis();
73             vm.execute();
74             glyph.font.max_ascent = vm.getUserInfo(8);
75             glyph.font.max_descent = vm.getUserInfo(9);
76             glyph.baseline = vm.getUserInfo(10);
77             glyph.advance = vm.getUserInfo(11);
78             
79             width = vm.getUserInfo(6);
80             height = vm.getUserInfo(7);
81
82             data = new byte[width * height];
83             int addr = vm.getUserInfo(5);
84             vm.copyin(addr,data,width*height);
85             File tmp = new File(cacheFile.getCanonicalPath() + ".tmp");
86             DataOutputStream dis = new DataOutputStream(new FileOutputStream(tmp));
87             dis.writeInt(width);
88             dis.writeInt(height);
89             dis.writeInt(glyph.font.max_ascent);
90             dis.writeInt(glyph.font.max_descent);
91             dis.writeInt(glyph.baseline);
92             dis.writeInt(glyph.advance);
93             if (width != 0 && height != 0) dis.write(data, 0, data.length);
94             dis.close();
95             tmp.renameTo(cacheFile);
96
97         } catch (Exception e) {
98             Log.log(this, e);
99         }
100
101         if (width == 0 || height == 0) Log.log(Freetype.class, "warning glyph has zero width/height");
102         glyph.p = Platform.createAlphaOnlyPicture(data, width, height);
103     }
104 }