1 f2887b9d 2009-08-11 rsc #include <u.h>
2 f2887b9d 2009-08-11 rsc #include <libc.h>
3 f2887b9d 2009-08-11 rsc #include <draw.h>
4 f2887b9d 2009-08-11 rsc #include <cursor.h>
5 f2887b9d 2009-08-11 rsc #include <event.h>
6 f2887b9d 2009-08-11 rsc #include <bio.h>
7 f2887b9d 2009-08-11 rsc #include <plumb.h>
8 f2887b9d 2009-08-11 rsc #include <ctype.h>
9 f2887b9d 2009-08-11 rsc #include <keyboard.h>
10 f2887b9d 2009-08-11 rsc #include <thread.h>
11 f2887b9d 2009-08-11 rsc #include "page.h"
13 f2887b9d 2009-08-11 rsc typedef struct Cached Cached;
14 f2887b9d 2009-08-11 rsc struct Cached
16 f2887b9d 2009-08-11 rsc Document *doc;
23 f2887b9d 2009-08-11 rsc static Cached cache[5];
24 f2887b9d 2009-08-11 rsc static int rabusy;
26 f2887b9d 2009-08-11 rsc static Image*
27 f2887b9d 2009-08-11 rsc questionmark(void)
29 f2887b9d 2009-08-11 rsc static Image *im;
33 f2887b9d 2009-08-11 rsc im = xallocimage(display, Rect(0,0,50,50), GREY1, 1, DBlack);
34 f2887b9d 2009-08-11 rsc if(im == nil)
36 f2887b9d 2009-08-11 rsc string(im, ZP, display->white, ZP, display->defaultfont, "?");
41 f2887b9d 2009-08-11 rsc cacheflush(void)
46 f2887b9d 2009-08-11 rsc for(i=0; i<nelem(cache); i++){
47 f2887b9d 2009-08-11 rsc c = &cache[i];
49 f2887b9d 2009-08-11 rsc freeimage(c->im);
51 f2887b9d 2009-08-11 rsc c->doc = nil;
55 f2887b9d 2009-08-11 rsc static Image*
56 f2887b9d 2009-08-11 rsc _cachedpage(Document *doc, int angle, int page, char *ra)
59 f2887b9d 2009-08-11 rsc Cached *c, old;
60 f2887b9d 2009-08-11 rsc Image *im, *tmp;
61 d9e047e5 2010-02-04 rsc int ppi = 100;
62 d9e047e5 2010-02-04 rsc PDFInfo *pdf;
65 f2887b9d 2009-08-11 rsc if((page < 0 || page >= doc->npage) && !doc->fwdonly)
68 d9e047e5 2010-02-04 rsc if (doc->type == Tpdf){
69 d9e047e5 2010-02-04 rsc pdf = (PDFInfo *) doc->extra;
70 d9e047e5 2010-02-04 rsc ppi = pdf->gs.ppi;
73 d9e047e5 2010-02-04 rsc if (doc->type == Tps){
74 d9e047e5 2010-02-04 rsc ps = (PSInfo *) doc->extra;
75 d9e047e5 2010-02-04 rsc ppi = ps->gs.ppi;
80 f2887b9d 2009-08-11 rsc for(i=0; i<nelem(cache); i++){
81 f2887b9d 2009-08-11 rsc c = &cache[i];
82 d9e047e5 2010-02-04 rsc if(c->doc == doc && c->angle == angle && c->page == page && c->ppi == ppi){
83 f2887b9d 2009-08-11 rsc if(chatty) fprint(2, "cache%s hit %d\n", ra, page);
86 f2887b9d 2009-08-11 rsc if(c->doc == nil)
90 f2887b9d 2009-08-11 rsc if(i >= nelem(cache))
91 f2887b9d 2009-08-11 rsc i = nelem(cache)-1;
92 f2887b9d 2009-08-11 rsc c = &cache[i];
94 f2887b9d 2009-08-11 rsc freeimage(c->im);
96 f2887b9d 2009-08-11 rsc c->doc = nil;
97 f2887b9d 2009-08-11 rsc c->page = -1;
100 f2887b9d 2009-08-11 rsc if(chatty) fprint(2, "cache%s load %d\n", ra, page);
101 f2887b9d 2009-08-11 rsc im = doc->drawpage(doc, page);
102 f2887b9d 2009-08-11 rsc if(im == nil){
103 f2887b9d 2009-08-11 rsc if(doc->fwdonly) /* end of file */
105 f2887b9d 2009-08-11 rsc im = questionmark();
106 f2887b9d 2009-08-11 rsc if(im == nil){
109 f2887b9d 2009-08-11 rsc cacheflush();
112 f2887b9d 2009-08-11 rsc fprint(2, "out of memory: %r\n");
113 f2887b9d 2009-08-11 rsc wexits("memory");
118 f2887b9d 2009-08-11 rsc if(im->r.min.x != 0 || im->r.min.y != 0){
119 f2887b9d 2009-08-11 rsc /* translate to 0,0 */
120 f2887b9d 2009-08-11 rsc tmp = xallocimage(display, Rect(0, 0, Dx(im->r), Dy(im->r)), im->chan, 0, DNofill);
121 f2887b9d 2009-08-11 rsc if(tmp == nil){
122 f2887b9d 2009-08-11 rsc freeimage(im);
125 f2887b9d 2009-08-11 rsc drawop(tmp, tmp->r, im, nil, im->r.min, S);
126 f2887b9d 2009-08-11 rsc freeimage(im);
130 f2887b9d 2009-08-11 rsc switch(angle){
132 f2887b9d 2009-08-11 rsc im = rot90(im);
138 f2887b9d 2009-08-11 rsc im = rot270(im);
141 f2887b9d 2009-08-11 rsc if(im == nil)
144 f2887b9d 2009-08-11 rsc c->doc = doc;
145 f2887b9d 2009-08-11 rsc c->page = page;
146 f2887b9d 2009-08-11 rsc c->angle = angle;
148 d9e047e5 2010-02-04 rsc c->ppi = ppi;
151 f2887b9d 2009-08-11 rsc if(chatty) fprint(2, "cache%s mtf %d @%d:", ra, c->page, i);
153 f2887b9d 2009-08-11 rsc memmove(cache+1, cache, (c-cache)*sizeof cache[0]);
154 f2887b9d 2009-08-11 rsc cache[0] = old;
156 f2887b9d 2009-08-11 rsc for(i=0; i<nelem(cache); i++)
157 f2887b9d 2009-08-11 rsc fprint(2, " %d", cache[i].page);
158 f2887b9d 2009-08-11 rsc fprint(2, "\n");
160 f2887b9d 2009-08-11 rsc if(chatty) fprint(2, "cache%s return %d %p\n", ra, old.page, old.im);
161 f2887b9d 2009-08-11 rsc return old.im;
165 f2887b9d 2009-08-11 rsc raproc(void *a)
170 f2887b9d 2009-08-11 rsc lockdisplay(display);
172 46606276 2019-11-14 crossd * If there is only one page in a fwdonly file, we may reach EOF
173 46606276 2019-11-14 crossd * while doing readahead and page will exit without showing anything.
175 46606276 2019-11-14 crossd if(!c->doc->fwdonly)
176 46606276 2019-11-14 crossd _cachedpage(c->doc, c->angle, c->page, "-ra");
178 f2887b9d 2009-08-11 rsc unlockdisplay(display);
180 f2887b9d 2009-08-11 rsc threadexits(0);
184 f2887b9d 2009-08-11 rsc cachedpage(Document *doc, int angle, int page)
186 f2887b9d 2009-08-11 rsc static int lastpage = -1;
191 f2887b9d 2009-08-11 rsc if(doc->npage < 1)
192 f2887b9d 2009-08-11 rsc return display->white;
194 f2887b9d 2009-08-11 rsc im = _cachedpage(doc, angle, page, "");
195 f2887b9d 2009-08-11 rsc if(im == nil)
198 f2887b9d 2009-08-11 rsc /* readahead */
200 f2887b9d 2009-08-11 rsc if(!rabusy){
201 f2887b9d 2009-08-11 rsc if(page == lastpage+1)
202 f2887b9d 2009-08-11 rsc ra = page+1;
203 f2887b9d 2009-08-11 rsc else if(page == lastpage-1)
204 f2887b9d 2009-08-11 rsc ra = page-1;
206 f2887b9d 2009-08-11 rsc lastpage = page;
207 f2887b9d 2009-08-11 rsc if(ra >= 0){
208 f2887b9d 2009-08-11 rsc c = emalloc(sizeof(*c));
209 f2887b9d 2009-08-11 rsc c->doc = doc;
210 f2887b9d 2009-08-11 rsc c->angle = angle;
211 f2887b9d 2009-08-11 rsc c->page = ra;
212 f2887b9d 2009-08-11 rsc c->im = nil;
214 f2887b9d 2009-08-11 rsc if(proccreate(raproc, c, mainstacksize) == -1)