changeset 5379:9fc7bcadcc1b

.ttf font loader - based on TOOLS/subfont-c
author arpi
date Thu, 28 Mar 2002 20:22:34 +0000
parents 1a7fd8f13ac0
children 8a01cde9cf39
files libvo/ttf_load.c
diffstat 1 files changed, 714 insertions(+), 0 deletions(-) [+]
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/libvo/ttf_load.c	Thu Mar 28 20:22:34 2002 +0000
@@ -0,0 +1,714 @@
+/*
+ * Renders antialiased fonts for mplayer using freetype library.
+ * Should work with TrueType, Type1 and any other font supported by libfreetype.
+ * Can generate font.desc for any encoding.
+ *
+ *
+ * Artur Zaprzala <zybi@fanthom.irc.pl>
+ *
+ */
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <iconv.h>
+#include <math.h>
+#include <string.h>
+#include <libgen.h>
+
+#ifndef OLD_FREETYPE2
+#include <ft2build.h>	
+#include FT_FREETYPE_H
+#include FT_GLYPH_H
+#else			/* freetype 2.0.1 */
+#include <freetype/freetype.h>
+#include <freetype/ftglyph.h>
+#endif
+
+#include "../config.h"
+#include "../mp_msg.h"
+
+#include "../bswap.h"
+
+
+#ifndef DEBUG
+#define DEBUG	0
+#endif
+
+#include "font_load.h"
+
+//// default values
+static char		*encoding = "iso-8859-1";	/* target encoding */
+static char		*charmap = "ucs-4";		/* font charmap encoding, I hope ucs-4 is always big endian */
+						/* gcc 2.1.3 doesn't support ucs-4le, but supports ucs-4 (==ucs-4be) */
+static float		ppem = 28;			/* font size in pixels */
+
+static double		radius = 2;			/* blur radius */
+static double		thickness = 2.5;		/* outline thickness */
+
+//char*		font_desc = "font.desc";
+//char*		font_desc = "/dev/stdout";
+
+//char		*outdir = ".";
+
+static font_desc_t *desc=NULL;
+
+//// constants
+static int const	colors = 256;
+static int const	maxcolor = 255;
+static unsigned const	base = 256;
+static unsigned const	first_char = 33;
+#define max_charset_size	60000
+//int const	max_charset_size = 256;
+static unsigned	charset_size = 0;
+
+////
+//static char		*command;
+//static char		*encoding_name;
+static char		*font_path=NULL;
+//char		*font_metrics;
+//static int		append_mode = 0;
+static int		unicode_desc = 0;
+static int	font_id=0;
+
+static unsigned char	*bbuffer, *abuffer;
+static int		width, height;
+static int		padding;
+static FT_ULong	charset[max_charset_size];		/* characters we want to render; Unicode */
+static FT_ULong	charcodes[max_charset_size];	/* character codes in 'encoding' */
+static iconv_t cd;					// iconv conversion descriptor
+
+
+#define eprintf(...)		mp_msg(MSGT_CPLAYER,MSGL_INFO, __VA_ARGS__)
+#define ERROR_(msg, ...)	(mp_msg(MSGT_CPLAYER,MSGL_ERR,"[font_load] error: " msg "\n", __VA_ARGS__), exit(1))
+#define WARNING_(msg, ...)	mp_msg(MSGT_CPLAYER,MSGL_WARN,"[font_load] warning: " msg "\n", __VA_ARGS__)
+#define ERROR(...)		ERROR_(__VA_ARGS__, NULL)
+#define WARNING(...)		WARNING_(__VA_ARGS__, NULL)
+
+
+#define f266ToInt(x)		(((x)+32)>>6)	// round fractional fixed point number to integer
+						// coordinates are in 26.6 pixels (i.e. 1/64th of pixels)
+#define f266CeilToInt(x)	(((x)+63)>>6)	// ceiling
+#define f266FloorToInt(x)	((x)>>6)	// floor
+#define f1616ToInt(x)		(((x)+0x8000)>>16)	// 16.16
+#define floatTof266(x)		((int)((x)*(1<<6)+0.5))
+
+#define ALIGN(x)		(((x)+7)&~7)	// 8 byte align
+
+
+
+static void paste_bitmap(FT_Bitmap *bitmap, int x, int y) {
+    int drow = x+y*width;
+    int srow = 0;
+    int sp, dp, w, h;
+    if (bitmap->pixel_mode==ft_pixel_mode_mono)
+	for (h = bitmap->rows; h>0; --h, drow+=width, srow+=bitmap->pitch)
+	    for (w = bitmap->width, sp=dp=0; w>0; --w, ++dp, ++sp)
+		    bbuffer[drow+dp] = (bitmap->buffer[srow+sp/8] & (0x80>>(sp%8))) ? 255:0;
+    else
+	for (h = bitmap->rows; h>0; --h, drow+=width, srow+=bitmap->pitch)
+	    for (w = bitmap->width, sp=dp=0; w>0; --w, ++dp, ++sp)
+		    bbuffer[drow+dp] = bitmap->buffer[srow+sp];
+}
+
+
+static void render() {
+    FT_Library	library;
+    FT_Face	face;
+    FT_Error	error;
+    FT_Glyph	*glyphs;
+    FT_BitmapGlyph glyph;
+    //FILE	*f;
+    int	const	load_flags = FT_LOAD_DEFAULT | FT_LOAD_NO_HINTING;
+    int		pen_x = 0, pen_xa;
+    int		ymin = INT_MAX, ymax = INT_MIN;
+    int		i, uni_charmap = 1;
+    int		baseline, space_advance = 20;
+    int		glyphs_count = 0;
+
+
+    /* initialize freetype */
+    error = FT_Init_FreeType(&library);
+    if (error) ERROR("Init_FreeType failed.");
+    error = FT_New_Face(library, font_path, 0, &face);
+    if (error) ERROR("New_Face failed. Maybe the font path `%s' is wrong.", font_path);
+
+    /*
+    if (font_metrics) {
+	error = FT_Attach_File(face, font_metrics);
+	if (error) WARNING("FT_Attach_File failed.");
+    }
+    */
+
+
+#if 0
+    /************************************************************/
+    eprintf("Font encodings:\n");
+    for (i = 0; i<face->num_charmaps; ++i)
+	eprintf("'%.4s'\n", (char*)&face->charmaps[i]->encoding);
+
+    //error = FT_Select_Charmap(face, ft_encoding_unicode);
+    //error = FT_Select_Charmap(face, ft_encoding_adobe_standard);
+    //error = FT_Select_Charmap(face, ft_encoding_adobe_custom);
+    //error = FT_Set_Charmap(face, face->charmaps[1]);
+    //if (error) WARNING("FT_Select_Charmap failed.");
+#endif
+
+
+#if 0
+    /************************************************************/
+    if (FT_HAS_GLYPH_NAMES(face)) {
+	int const max_gname = 128;
+	char gname[max_gname];
+	for (i = 0; i<face->num_glyphs; ++i) {
+	    FT_Get_Glyph_Name(face, i, gname, max_gname);
+	    eprintf("%02x `%s'\n", i, gname);
+	}
+
+    }
+#endif
+
+
+    if (face->charmap==NULL || face->charmap->encoding!=ft_encoding_unicode) {
+	WARNING("Unicode charmap not available for this font. Very bad!");
+	uni_charmap = 0;
+	error = FT_Set_Charmap(face, face->charmaps[0]);
+	if (error) WARNING("No charmaps! Strange.");
+    }
+
+
+
+    /* set size */
+    if (FT_IS_SCALABLE(face)) {
+	error = FT_Set_Char_Size(face, floatTof266(ppem), 0, 0, 0);
+	if (error) WARNING("FT_Set_Char_Size failed.");
+    } else {
+	int j = 0;
+	int jppem = face->available_sizes[0].height;
+	/* find closest size */
+	for (i = 0; i<face->num_fixed_sizes; ++i) {
+	    if (fabs(face->available_sizes[i].height - ppem) < abs(face->available_sizes[i].height - jppem)) {
+		j = i;
+		jppem = face->available_sizes[i].height;
+	    }
+	}
+	WARNING("Selected font is not scalable. Using ppem=%i.", face->available_sizes[j].height);
+	error = FT_Set_Pixel_Sizes(face, face->available_sizes[j].width, face->available_sizes[j].height);
+	if (error) WARNING("FT_Set_Pixel_Sizes failed.");
+    }
+
+
+    if (FT_IS_FIXED_WIDTH(face))
+	WARNING("Selected font is fixed-width.");
+
+
+    /* compute space advance */
+    error = FT_Load_Char(face, ' ', load_flags);
+    if (error) WARNING("spacewidth set to default.");
+    else space_advance = f266ToInt(face->glyph->advance.x);
+
+
+    /* create font.desc */
+    if(!font_id){ // first font
+        desc->spacewidth=2*padding + space_advance;
+        desc->charspace=-2*padding;
+        desc->height=f266ToInt(face->size->metrics.height);
+//	fprintf(f, "ascender %i\n",	f266CeilToInt(face->size->metrics.ascender));
+//	fprintf(f, "descender %i\n",	f266FloorToInt(face->size->metrics.descender));
+    }
+
+    // render glyphs, compute bitmap size and [characters] section
+    glyphs = (FT_Glyph*)malloc(charset_size*sizeof(FT_Glyph*));
+    for (i= 0; i<charset_size; ++i) {
+	FT_GlyphSlot	slot;
+	FT_ULong	character, code;
+	FT_UInt		glyph_index;
+	FT_BBox		bbox;
+
+	character = charset[i];
+	code = charcodes[i];
+
+	// get glyph index
+	if (character==0)
+	    glyph_index = 0;
+	else {
+	    glyph_index = FT_Get_Char_Index(face, uni_charmap ? character:code);
+	    if (glyph_index==0) {
+		WARNING("Glyph for char 0x%02x|U+%04X|%c not found.", code, character,
+			 code<' '||code>255 ? '.':code);
+		continue;
+	    }
+	}
+
+	// load glyph
+	error = FT_Load_Glyph(face, glyph_index, load_flags);
+	if (error) {
+	    WARNING("FT_Load_Glyph 0x%02x (char 0x%02x|U+%04X) failed.", glyph_index, code, character);
+	    continue;
+	}
+	slot = face->glyph;
+
+	// render glyph
+	if (slot->format != ft_glyph_format_bitmap) {
+	    error = FT_Render_Glyph(slot, ft_render_mode_normal);
+	    if (error) {
+		WARNING("FT_Render_Glyph 0x%04x (char 0x%02x|U+%04X) failed.", glyph_index, code, character);
+		continue;
+	    }
+	}
+
+	// extract glyph image
+	error = FT_Get_Glyph(slot, (FT_Glyph*)&glyph);
+	if (error) {
+	    WARNING("FT_Get_Glyph 0x%04x (char 0x%02x|U+%04X) failed.", glyph_index, code, character);
+	    continue;
+	}
+	glyphs[glyphs_count++] = (FT_Glyph)glyph;
+
+#ifdef NEW_DESC
+	// max height
+	if (glyph->bitmap.rows > height) height = glyph->bitmap.rows;
+
+	// advance pen
+	pen_xa = pen_x + glyph->bitmap.width + 2*padding;
+
+	// font.desc
+	fprintf(f, "0x%04x %i %i %i %i %i %i;\tU+%04X|%c\n", unicode_desc ? character:code,
+		pen_x,						// bitmap start
+		glyph->bitmap.width + 2*padding,		// bitmap width
+		glyph->bitmap.rows + 2*padding,			// bitmap height
+		glyph->left - padding,				// left bearing
+		glyph->top + padding,				// top bearing
+		f266ToInt(slot->advance.x),			// advance
+		character, code<' '||code>255 ? '.':code);
+#else
+	// max height
+	if (glyph->top > ymax) {
+	    ymax = glyph->top;
+	    //eprintf("%3i: ymax %i (%c)\n", code, ymax, code);
+	}
+	if (glyph->top - glyph->bitmap.rows < ymin) {
+	    ymin = glyph->top - glyph->bitmap.rows;
+	    //eprintf("%3i: ymin %i (%c)\n", code, ymin, code);
+	}
+
+	/* advance pen */
+	pen_xa = pen_x + f266ToInt(slot->advance.x) + 2*padding;
+
+	/* font.desc */
+//	fprintf(f, "0x%04x %i %i;\tU+%04X|%c\n", unicode_desc ? character:code,
+//		pen_x,						// bitmap start
+//		pen_xa-1,					// bitmap end
+//		character, code<' '||code>255 ? '.':code);
+	desc->start[unicode_desc ? character:code]=pen_x;
+	desc->width[unicode_desc ? character:code]=pen_xa-1-pen_x;
+	desc->font[unicode_desc ? character:code]=font_id;
+
+#endif
+	pen_x = ALIGN(pen_xa);
+    }
+
+
+    width = pen_x;
+    pen_x = 0;
+#ifdef NEW_DESC
+    if (height<=0) ERROR("Something went wrong. Use the source!");
+    height += 2*padding;
+#else
+    if (ymax<=ymin) ERROR("Something went wrong. Use the source!");
+    height = ymax - ymin + 2*padding;
+    baseline = ymax + padding;
+#endif
+
+    // end of font.desc
+    if (DEBUG) eprintf("bitmap size: %ix%i\n", width, height);
+//    fprintf(f, "# bitmap size: %ix%i\n", width, height);
+//    fclose(f);
+
+    bbuffer = (unsigned char*)malloc(width*height);
+    if (bbuffer==NULL) ERROR("malloc failed.");
+    memset(bbuffer, 0, width*height);
+
+
+    /* paste glyphs */
+    for (i= 0; i<glyphs_count; ++i) {
+	glyph = (FT_BitmapGlyph)glyphs[i];
+#ifdef NEW_DESC
+	paste_bitmap(&glyph->bitmap,
+	    pen_x + padding,
+	    padding);
+
+	/* advance pen */
+	pen_x += glyph->bitmap.width + 2*padding;
+#else
+	paste_bitmap(&glyph->bitmap,
+	    pen_x + padding + glyph->left,
+	    baseline - glyph->top);
+
+	/* advance pen */
+	pen_x += f1616ToInt(glyph->root.advance.x) + 2*padding;
+#endif
+	pen_x = ALIGN(pen_x);
+
+	FT_Done_Glyph((FT_Glyph)glyph);
+    }
+    free(glyphs);
+
+
+    error = FT_Done_FreeType(library);
+    if (error) ERROR("FT_Done_FreeType failed.");
+}
+
+
+/* decode from 'encoding' to unicode */
+static FT_ULong decode_char(char c) {
+    FT_ULong o;
+    char *inbuf = &c;
+    char *outbuf = (char*)&o;
+    int inbytesleft = 1;
+    int outbytesleft = sizeof(FT_ULong);
+
+    size_t count = iconv(cd, &inbuf, &inbytesleft, &outbuf, &outbytesleft);
+
+    /* convert unicode BigEndian -> MachineEndian */
+    o = be2me_32(o);
+
+    // if (count==-1) o = 0; // not OK, at least my iconv() returns E2BIG for all
+    if (outbytesleft!=0) o = 0;
+
+    /* we don't want control characters */
+    if (o>=0x7f && o<0xa0) o = 0;
+    return o;
+}
+
+
+static void prepare_charset() {
+    FILE *f;
+    FT_ULong i;
+
+    f = fopen(encoding, "r");		// try to read custom encoding
+    if (f==NULL) {
+	int count = 0;
+	// check if ucs-4 is available
+	cd = iconv_open(charmap, charmap);
+	if (cd==(iconv_t)-1) ERROR("iconv doesn't know %s encoding. Use the source!", charmap);
+	iconv_close(cd);
+
+	cd = iconv_open(charmap, encoding);
+	if (cd==(iconv_t)-1) ERROR("Unsupported encoding `%s', use iconv --list to list character sets known on your system.", encoding);
+
+	charset_size = 256 - first_char;
+	for (i = 0; i<charset_size; ++i) {
+	    charcodes[count] = i+first_char;
+	    charset[count] = decode_char(i+first_char);
+	    //eprintf("%04X U%04X\n", charcodes[count], charset[count]);
+	    if (charset[count]!=0) ++count;
+	}
+	charcodes[count] = charset[count] = 0; ++count;
+	charset_size = count;
+
+	iconv_close(cd);
+    } else {
+	unsigned int character, code;
+	int count;
+
+	eprintf("Reading custom encoding from file '%s'.\n", encoding);
+
+       	while ((count = fscanf(f, "%x%*[ \t]%x", &character, &code)) != EOF) {
+	    if (charset_size==max_charset_size) {
+		WARNING("There is no place for  more than %i characters. Use the source!", max_charset_size);
+		break;
+	    }
+	    if (count==0) ERROR("Unable to parse custom encoding file.");
+	    if (character<32) continue;	// skip control characters
+	    charset[charset_size] = character;
+	    charcodes[charset_size] = count==2 ? code : character;
+	    ++charset_size;
+	}
+	fclose(f);
+//	encoding = basename(encoding);
+    }
+    if (charset_size==0) ERROR("No characters to render!");
+}
+
+
+// general outline
+static void outline(
+	unsigned char *s,
+	unsigned char *t,
+	int width,
+	int height,
+	int *m,
+	int r,
+	int mwidth) {
+
+    int x, y;
+    for (y = 0; y<height; ++y) {
+	for (x = 0; x<width; ++x, ++s, ++t) {
+	    unsigned max = 0;
+	    unsigned *mrow = m + r;
+	    unsigned char *srow = s -r*width;
+	    int x1=(x<r)?-x:-r;
+	    int x2=(x+r>=width)?(width-x-1):r;
+	    int my;
+
+	    for (my = -r; my<=r; ++my, srow+= width, mrow+= mwidth) {
+		int mx;
+		if (y+my < 0) continue;
+		if (y+my >= height) break;
+
+		for (mx = x1; mx<=x2; ++mx) {
+		    unsigned v = srow[mx] * mrow[mx];
+		    if (v>max) max = v;
+		}
+	    }
+	    *t = (max + base/2) / base;
+	}
+    }
+}
+
+
+// 1 pixel outline
+static void outline1(
+	unsigned char *s,
+	unsigned char *t,
+	int width,
+	int height) {
+
+    int x, y, mx, my;
+
+    for (x = 0; x<width; ++x, ++s, ++t) *t = *s;
+    for (y = 1; y<height-1; ++y) {
+	*t++ = *s++;
+	for (x = 1; x<width-1; ++x, ++s, ++t) {
+	    unsigned v = (
+		    s[-1-width]+
+		    s[-1+width]+
+		    s[+1-width]+
+		    s[+1+width]
+		)/2 + (
+		    s[-1]+
+		    s[+1]+
+		    s[-width]+
+		    s[+width]+
+		    s[0]
+		);
+	    *t = v>maxcolor ? maxcolor : v;
+	}
+	*t++ = *s++;
+    }
+    for (x = 0; x<width; ++x, ++s, ++t) *t = *s;
+}
+
+
+// gaussian blur
+static void blur(
+	unsigned char *buffer,
+	unsigned char *tmp,
+	int width,
+	int height,
+	int *m,
+	int r,
+	int mwidth,
+	unsigned volume) {
+
+    int x, y;
+
+    unsigned char *s = buffer - r;
+    unsigned char *t = tmp;
+    for (y = 0; y<height; ++y) {
+	for (x = 0; x<width; ++x, ++s, ++t) {
+	    unsigned sum = 0;
+	    int x1 = (x<r) ? r-x:0;
+	    int x2 = (x+r>=width) ? (r+width-x):mwidth;
+	    int mx;
+	    for (mx = x1; mx<x2; ++mx)
+		sum+= s[mx] * m[mx];
+	    *t = (sum + volume/2) / volume;
+	    //*t = sum;
+	}
+    }
+    tmp -= r*width;
+    for (x = 0; x<width; ++x, ++tmp, ++buffer) {
+	s = tmp;
+	t = buffer;
+	for (y = 0; y<height; ++y, s+= width, t+= width) {
+	    unsigned sum = 0;
+	    int y1 = (y<r) ? r-y:0;
+	    int y2 = (y+r>=height) ? (r+height-y):mwidth;
+	    unsigned char *smy = s + y1*width;
+	    int my;
+	    for (my = y1; my<y2; ++my, smy+= width)
+		sum+= *smy * m[my];
+	    *t = (sum + volume/2) / volume;
+	}
+    }
+}
+
+
+// Gaussian matrix
+// Maybe for future use.
+static unsigned gmatrix(unsigned *m, int r, int w, double const A) {
+    unsigned volume = 0;		// volume under Gaussian area is exactly -pi*base/A
+    int mx, my;
+
+    for (my = 0; my<w; ++my) {
+	for (mx = 0; mx<w; ++mx) {
+	    m[mx+my*w] = (unsigned)(exp(A * ((mx-r)*(mx-r)+(my-r)*(my-r))) * base + .5);
+	    volume+= m[mx+my*w];
+	    if (DEBUG) eprintf("%3i ", m[mx+my*w]);
+	}
+	if (DEBUG) eprintf("\n");
+    }
+    if (DEBUG) {
+	eprintf("A= %f\n", A);
+	eprintf("volume: %i; exact: %.0f; volume/exact: %.6f\n\n", volume, -M_PI*base/A, volume/(-M_PI*base/A));
+    }
+    return volume;
+}
+
+
+static void alpha() {
+    int const g_r = ceil(radius);
+    int const o_r = ceil(thickness);
+    int const g_w = 2*g_r+1;		// matrix size
+    int const o_w = 2*o_r+1;		// matrix size
+    double const A = log(1.0/base)/(radius*radius*2);
+
+    int mx, my, i;
+    unsigned volume = 0;		// volume under Gaussian area is exactly -pi*base/A
+
+    unsigned *g = (unsigned*)malloc(g_w * sizeof(unsigned));
+    unsigned *om = (unsigned*)malloc(o_w*o_w * sizeof(unsigned));
+    if (g==NULL || om==NULL) ERROR("malloc failed.");
+
+    // gaussian curve
+    for (i = 0; i<g_w; ++i) {
+	g[i] = (unsigned)(exp(A * (i-g_r)*(i-g_r)) * base + .5);
+	volume+= g[i];
+	if (DEBUG) eprintf("%3i ", g[i]);
+    }
+    //volume *= volume;
+    if (DEBUG) eprintf("\n");
+
+    /* outline matrix */
+    for (my = 0; my<o_w; ++my) {
+	for (mx = 0; mx<o_w; ++mx) {
+	    // antialiased circle would be perfect here, but this one is good enough
+	    double d = thickness + 1 - sqrt((mx-o_r)*(mx-o_r)+(my-o_r)*(my-o_r));
+	    om[mx+my*o_w] = d>=1 ? base : d<=0 ? 0 : (d*base + .5);
+	    if (DEBUG) eprintf("%3i ", om[mx+my*o_w]);
+	}
+	if (DEBUG) eprintf("\n");
+    }
+    if (DEBUG) eprintf("\n");
+
+
+    if(thickness==1.0)
+      outline1(bbuffer, abuffer, width, height);	// FAST solid 1 pixel outline
+    else
+      outline(bbuffer, abuffer, width, height, om, o_r, o_w);	// solid outline
+
+    //outline(bbuffer, abuffer, width, height, gm, g_r, g_w);	// Gaussian outline
+
+    blur(abuffer, bbuffer, width, height, g, g_r, g_w, volume);
+
+    free(g);
+    free(om);
+}
+
+font_desc_t* read_font_desc(char* fname,float factor,int verbose){
+    int i=font_id;
+    int j;
+    
+    if(!font_id){
+	desc=malloc(sizeof(font_desc_t));if(!desc) return NULL;
+	memset(desc,0,sizeof(font_desc_t));
+	memset(desc->font,255,sizeof(short)*65536);
+    }
+
+    padding = ceil(radius) + ceil(thickness);
+    
+    font_path=fname;
+
+    prepare_charset();
+
+    render();
+    desc->pic_b[font_id]=malloc(sizeof(raw_file));
+    desc->pic_b[font_id]->bmp=malloc(width*height);
+    memcpy(desc->pic_b[font_id]->bmp,bbuffer,width*height);
+    desc->pic_b[font_id]->pal=NULL;
+    desc->pic_b[font_id]->w=width;
+    desc->pic_b[font_id]->h=height;
+    desc->pic_b[font_id]->c=256;
+//    write_bitmap(bbuffer, 'b');
+
+    abuffer = (unsigned char*)malloc(width*height);
+    if (abuffer==NULL) ERROR("malloc failed.");
+    alpha();
+//    write_bitmap(abuffer, 'a');
+    desc->pic_a[font_id]=malloc(sizeof(raw_file));
+    desc->pic_a[font_id]->bmp=abuffer;
+//    desc->pic_a[font_id]->bmp=malloc(width*height);
+//    memcpy(desc->pic_a[font_id]->bmp,abuffer,width*height);
+    desc->pic_a[font_id]->pal=NULL;
+    desc->pic_a[font_id]->w=width;
+    desc->pic_a[font_id]->h=height;
+    desc->pic_a[font_id]->c=256;
+
+    free(bbuffer);
+//    free(abuffer);
+
+    //if(factor!=1.0f)
+    {
+        // re-sample alpha
+        int f=factor*256.0f;
+        int size=desc->pic_a[i]->w*desc->pic_a[i]->h;
+        int j;
+        if(verbose) printf("font: resampling alpha by factor %5.3f (%d) ",factor,f);fflush(stdout);
+        for(j=0;j<size;j++){
+            int x=desc->pic_a[i]->bmp[j];	// alpha
+            int y=desc->pic_b[i]->bmp[j];	// bitmap
+
+#ifdef FAST_OSD
+	    x=(x<(255-f))?0:1;
+#else
+
+	    x=255-((x*f)>>8); // scale
+	    //if(x<0) x=0; else if(x>255) x=255;
+	    //x^=255; // invert
+
+	    if(x+y>255) x=255-y; // to avoid overflows
+	    
+	    //x=0;            
+            //x=((x*f*(255-y))>>16);
+            //x=((x*f*(255-y))>>16)+y;
+            //x=(x*f)>>8;if(x<y) x=y;
+
+            if(x<1) x=1; else
+            if(x>=252) x=0;
+#endif
+
+            desc->pic_a[i]->bmp[j]=x;
+//            desc->pic_b[i]->bmp[j]=0; // hack
+        }
+        if(verbose) printf("DONE!\n");
+    }
+    if(!desc->height) desc->height=desc->pic_a[i]->h;
+
+j='_';if(desc->font[j]<0) j='?';
+for(i=0;i<512;i++)
+  if(desc->font[i]<0){
+      desc->start[i]=desc->start[j];
+      desc->width[i]=desc->width[j];
+      desc->font[i]=desc->font[j];
+  }
+desc->font[' ']=-1;
+desc->width[' ']=desc->spacewidth;
+
+printf("Font %s loaded successfully!\n",fname);
+    
+    ++font_id;
+
+//    fflush(stderr);
+    return desc;
+}