diff options
-rw-r--r-- | src/libdraw/string.c | 11 | ||||
-rw-r--r-- | src/libdraw/stringwidth.c | 13 | ||||
-rw-r--r-- | src/libdraw/subfont.c | 14 |
3 files changed, 32 insertions, 6 deletions
diff --git a/src/libdraw/string.c b/src/libdraw/string.c index 4dfb27fa..4e876c17 100644 --- a/src/libdraw/string.c +++ b/src/libdraw/string.c @@ -65,6 +65,7 @@ _string(Image *dst, Point pt, Image *src, Point sp, Font *f, char *s, Rune *r, i char **sptr; Rune **rptr; Font *def; + Subfont *sf; if(s == nil){ s = ""; @@ -76,6 +77,7 @@ _string(Image *dst, Point pt, Image *src, Point sp, Font *f, char *s, Rune *r, i rptr = nil; }else rptr = &r; + sf = nil; while((*s || *r) && len){ max = Max; if(len < max) @@ -124,13 +126,18 @@ _string(Image *dst, Point pt, Image *src, Point sp, Font *f, char *s, Rune *r, i len -= n; } if(subfontname){ - if(_getsubfont(f->display, subfontname) == 0){ - def = f->display->defaultfont; + freesubfont(sf); + if((sf=_getsubfont(f->display, subfontname)) == 0){ + def = f->display ? f->display->defaultfont : nil; if(def && f!=def) f = def; else break; } + /* + * must not free sf until cachechars has found it in the cache + * and picked up its own reference. + */ } } return pt; diff --git a/src/libdraw/stringwidth.c b/src/libdraw/stringwidth.c index 8d8c6d08..522fbc01 100644 --- a/src/libdraw/stringwidth.c +++ b/src/libdraw/stringwidth.c @@ -13,6 +13,7 @@ _stringnwidth(Font *f, char *s, Rune *r, int len) Rune rune, **rptr; char *subfontname, **sptr; Font *def; + Subfont *sf; if(s == nil){ s = ""; @@ -30,6 +31,7 @@ _stringnwidth(Font *f, char *s, Rune *r, int len) if(len < max) max = len; n = 0; + sf = nil; while((l = cachechars(f, sptr, rptr, cbuf, max, &wid, &subfontname)) <= 0){ if(++n > 10){ if(*r) @@ -40,19 +42,26 @@ _stringnwidth(Font *f, char *s, Rune *r, int len) name = f->name; else name = "unnamed font"; + freesubfont(sf); fprint(2, "stringwidth: bad character set for rune 0x%.4ux in %s\n", rune, name); return twid; } if(subfontname){ - if(_getsubfont(f->display, subfontname) == 0){ - def = f->display->defaultfont; + freesubfont(sf); + if((sf=_getsubfont(f->display, subfontname)) == 0){ + def = f->display ? f->display->defaultfont : nil; if(def && f!=def) f = def; else break; } + /* + * must not free sf until cachechars has found it in the cache + * and picked up its own reference. + */ } } + freesubfont(sf); agefont(f); twid += wid; len -= l; diff --git a/src/libdraw/subfont.c b/src/libdraw/subfont.c index 61838b05..085afba3 100644 --- a/src/libdraw/subfont.c +++ b/src/libdraw/subfont.c @@ -5,13 +5,14 @@ Subfont* allocsubfont(char *name, int n, int height, int ascent, Fontchar *info, Image *i) { - Subfont *f; + Subfont *f, *cf; assert(height != 0 /* allocsubfont */); f = malloc(sizeof(Subfont)); if(f == 0) return 0; +fprint(2, "allocsubfont %p\n", f); f->n = n; f->height = height; f->ascent = ascent; @@ -19,9 +20,18 @@ allocsubfont(char *name, int n, int height, int ascent, Fontchar *info, Image *i f->bits = i; f->ref = 1; if(name){ + /* + * if already caching this subfont, leave older + * (and hopefully more widely used) copy in cache. + * this case should not happen -- we got called + * because cachechars needed this subfont and it + * wasn't in the cache. + */ f->name = strdup(name); - if(lookupsubfont(i->display, name) == 0) + if((cf=lookupsubfont(i->display, name)) == 0) installsubfont(name, f); + else + freesubfont(cf); /* drop ref we just picked up */ }else f->name = 0; return f; |