main() --> main(void)
[mplayer/greg.git] / libvo / font_load.c
blob2703c3b79d7952f144417afc5e2a642be7404e7d
1 #include "config.h"
3 #include <stdio.h>
4 #include <stdlib.h>
5 #include <string.h>
6 #include <sys/types.h>
7 #include <sys/stat.h>
8 #include <unistd.h>
10 #include "font_load.h"
11 #include "mp_msg.h"
13 raw_file* load_raw(char *name,int verbose){
14 int bpp;
15 raw_file* raw=malloc(sizeof(raw_file));
16 unsigned char head[32];
17 FILE *f=fopen(name,"rb");
18 if(!f) goto err_out; // can't open
19 if(fread(head,32,1,f)<1) goto err_out; // too small
20 if(memcmp(head,"mhwanh",6)) goto err_out; // not raw file
21 raw->w=head[8]*256+head[9];
22 raw->h=head[10]*256+head[11];
23 raw->c=head[12]*256+head[13];
24 if(raw->w == 0) // 2 bytes were not enough for the width... read 4 bytes from the end of the header
25 raw->w = ((head[28]*0x100 + head[29])*0x100 + head[30])*0x100 + head[31];
26 if(raw->c>256) goto err_out; // too many colors!?
27 mp_msg(MSGT_OSD, MSGL_DBG2, "RAW: %s %d x %d, %d colors\n",name,raw->w,raw->h,raw->c);
28 if(raw->c){
29 raw->pal=malloc(raw->c*3);
30 fread(raw->pal,3,raw->c,f);
31 bpp=1;
32 } else {
33 raw->pal=NULL;
34 bpp=3;
36 raw->bmp=malloc(raw->h*raw->w*bpp);
37 fread(raw->bmp,raw->h*raw->w*bpp,1,f);
38 fclose(f);
39 return raw;
41 err_out:
42 if (f)
43 fclose(f);
44 free(raw);
45 return NULL;
48 extern int sub_unicode;
50 font_desc_t* read_font_desc(const char* fname,float factor,int verbose){
51 unsigned char sor[1024];
52 unsigned char sor2[1024];
53 font_desc_t *desc;
54 FILE *f = NULL;
55 char *dn;
56 //struct stat fstate;
57 char section[64];
58 int i,j;
59 int chardb=0;
60 int fontdb=-1;
61 int version=0;
62 int first=1;
64 desc=malloc(sizeof(font_desc_t));if(!desc) goto fail_out;
65 memset(desc,0,sizeof(font_desc_t));
67 f=fopen(fname,"rt");if(!f){ mp_msg(MSGT_OSD, MSGL_V, "font: can't open file: %s\n",fname); goto fail_out;}
69 i = strlen (fname) - 9;
70 if ((dn = malloc(i+1))){
71 strncpy (dn, fname, i);
72 dn[i]='\0';
75 desc->fpath = dn; // search in the same dir as fonts.desc
77 // desc->fpath=get_path("font/");
78 // if (stat(desc->fpath, &fstate)!=0) desc->fpath=DATADIR"/font";
83 // set up some defaults, and erase table
84 desc->charspace=2;
85 desc->spacewidth=12;
86 desc->height=0;
87 for(i=0;i<65536;i++) desc->start[i]=desc->width[i]=desc->font[i]=-1;
89 section[0]=0;
91 while(fgets(sor,1020,f)){
92 unsigned char* p[8];
93 int pdb=0;
94 unsigned char *s=sor;
95 unsigned char *d=sor2;
96 int ec=' ';
97 int id=0;
98 sor[1020]=0;
100 /* skip files that look like: TTF (0x00, 0x01), PFM (0x00, 0x01), PFB
101 * (0x80, 0x01), PCF (0x01, 0x66), fon ("MZ"), gzipped (0x1f, 0x8b) */
103 if (first) {
104 if (!sor[0] || sor[1] == 1 || (sor[0] == 'M' && sor[1] == 'Z') || (sor[0] == 0x1f && sor[1] == 0x8b) || (sor[0] == 1 && sor[1] == 0x66)) {
105 mp_msg(MSGT_OSD, MSGL_ERR, "%s doesn't look like a bitmap font description, ignoring.\n", fname);
106 goto fail_out;
108 first = 0;
111 p[0]=d;++pdb;
112 while(1){
113 int c=*s++;
114 if(c==0 || c==13 || c==10) break;
115 if(!id){
116 if(c==39 || c==34){ id=c;continue;} // idezojel
117 if(c==';' || c=='#') break;
118 if(c==9) c=' ';
119 if(c==' '){
120 if(ec==' ') continue;
121 *d=0; ++d;
122 p[pdb]=d;++pdb;
123 if(pdb>=8) break;
124 continue;
126 } else {
127 if(id==c){ id=0;continue;} // idezojel
130 *d=c;d++;
131 ec=c;
133 if(d==sor2) continue; // skip empty lines
134 *d=0;
136 // printf("params=%d sor=%s\n",pdb,sor);
137 // for(i=0;i<pdb;i++) printf(" param %d = '%s'\n",i,p[i]);
139 if(pdb==1 && p[0][0]=='['){
140 int len=strlen(p[0]);
141 if(len && len<63 && p[0][len-1]==']'){
142 strcpy(section,p[0]);
143 mp_msg(MSGT_OSD, MSGL_DBG2, "font: Reading section: %s\n",section);
144 if(strcmp(section,"[files]")==0){
145 ++fontdb;
146 if(fontdb>=16){ mp_msg(MSGT_OSD, MSGL_ERR, "font: Too many bitmaps defined.\n");goto fail_out;}
148 continue;
152 if(strcmp(section,"[fpath]")==0){
153 if(pdb==1){
154 if (desc->fpath)
155 free (desc->fpath); // release previously allocated memory
156 desc->fpath=strdup(p[0]);
157 continue;
159 } else
161 #ifdef SYS_AMIGAOS4
162 #define FONT_PATH_SEP ""
163 #else
164 //! path seperator for font paths, may not be more than one character
165 #define FONT_PATH_SEP "/"
166 #endif
168 if(strcmp(section,"[files]")==0){
169 char *default_dir=MPLAYER_DATADIR FONT_PATH_SEP "font";
170 if(pdb==2 && strcmp(p[0],"alpha")==0){
171 char *cp;
172 if (!(cp=malloc(strlen(desc->fpath)+strlen(p[1])+2))) goto fail_out;
174 snprintf(cp,strlen(desc->fpath)+strlen(p[1])+2,"%s" FONT_PATH_SEP "%s",
175 desc->fpath,p[1]);
176 if(!((desc->pic_a[fontdb]=load_raw(cp,verbose)))){
177 free(cp);
178 if (!(cp=malloc(strlen(default_dir)+strlen(p[1])+2)))
179 goto fail_out;
180 snprintf(cp,strlen(default_dir)+strlen(p[1])+2,"%s" FONT_PATH_SEP "%s",
181 default_dir,p[1]);
182 if (!((desc->pic_a[fontdb]=load_raw(cp,verbose)))){
183 mp_msg(MSGT_OSD, MSGL_ERR, "Can't load font bitmap: %s\n",p[1]);
184 free(cp);
185 goto fail_out;
188 free(cp);
189 continue;
191 if(pdb==2 && strcmp(p[0],"bitmap")==0){
192 char *cp;
193 if (!(cp=malloc(strlen(desc->fpath)+strlen(p[1])+2))) goto fail_out;
195 snprintf(cp,strlen(desc->fpath)+strlen(p[1])+2,"%s" FONT_PATH_SEP "%s",
196 desc->fpath,p[1]);
197 if(!((desc->pic_b[fontdb]=load_raw(cp,verbose)))){
198 free(cp);
199 if (!(cp=malloc(strlen(default_dir)+strlen(p[1])+2)))
200 goto fail_out;
201 snprintf(cp,strlen(default_dir)+strlen(p[1])+2,"%s" FONT_PATH_SEP "%s",
202 default_dir,p[1]);
203 if (!((desc->pic_b[fontdb]=load_raw(cp,verbose)))){
204 mp_msg(MSGT_OSD, MSGL_ERR, "Can't load font bitmap: %s\n",p[1]);
205 free(cp);
206 goto fail_out;
209 free(cp);
210 continue;
212 } else
214 if(strcmp(section,"[info]")==0){
215 if(pdb==2 && strcmp(p[0],"name")==0){
216 desc->name=strdup(p[1]);
217 continue;
219 if(pdb==2 && strcmp(p[0],"descversion")==0){
220 version=atoi(p[1]);
221 continue;
223 if(pdb==2 && strcmp(p[0],"spacewidth")==0){
224 desc->spacewidth=atoi(p[1]);
225 continue;
227 if(pdb==2 && strcmp(p[0],"charspace")==0){
228 desc->charspace=atoi(p[1]);
229 continue;
231 if(pdb==2 && strcmp(p[0],"height")==0){
232 desc->height=atoi(p[1]);
233 continue;
235 } else
237 if(strcmp(section,"[characters]")==0){
238 if(pdb==3){
239 int chr=p[0][0];
240 int start=atoi(p[1]);
241 int end=atoi(p[2]);
242 if(sub_unicode && (chr>=0x80)) chr=(chr<<8)+p[0][1];
243 else if(strlen(p[0])!=1) chr=strtol(p[0],NULL,0);
244 if(end<start) {
245 mp_msg(MSGT_OSD, MSGL_WARN, "error in font desc: end<start for char '%c'\n",chr);
246 } else {
247 desc->start[chr]=start;
248 desc->width[chr]=end-start+1;
249 desc->font[chr]=fontdb;
250 // printf("char %d '%c' start=%d width=%d\n",chr,chr,desc->start[chr],desc->width[chr]);
251 ++chardb;
253 continue;
256 mp_msg(MSGT_OSD, MSGL_ERR, "Syntax error in font desc: %s",sor);
257 goto fail_out;
260 fclose(f);
261 f = NULL;
263 if (first == 1) {
264 mp_msg(MSGT_OSD, MSGL_ERR, "%s is empty or a directory, ignoring.\n", fname);
265 goto fail_out;
268 //printf("font: pos of U = %d\n",desc->start[218]);
270 for(i=0;i<=fontdb;i++){
271 if(!desc->pic_a[i] || !desc->pic_b[i]){
272 mp_msg(MSGT_OSD, MSGL_ERR, "font: Missing bitmap(s) for sub-font #%d\n",i);
273 goto fail_out;
275 //if(factor!=1.0f)
277 // re-sample alpha
278 int f=factor*256.0f;
279 int size=desc->pic_a[i]->w*desc->pic_a[i]->h;
280 int j;
281 mp_msg(MSGT_OSD, MSGL_DBG2, "font: resampling alpha by factor %5.3f (%d) ",factor,f);fflush(stdout);
282 for(j=0;j<size;j++){
283 int x=desc->pic_a[i]->bmp[j]; // alpha
284 int y=desc->pic_b[i]->bmp[j]; // bitmap
286 #ifdef FAST_OSD
287 x=(x<(255-f))?0:1;
288 #else
290 x=255-((x*f)>>8); // scale
291 //if(x<0) x=0; else if(x>255) x=255;
292 //x^=255; // invert
294 if(x+y>255) x=255-y; // to avoid overflows
296 //x=0;
297 //x=((x*f*(255-y))>>16);
298 //x=((x*f*(255-y))>>16)+y;
299 //x=(x*f)>>8;if(x<y) x=y;
301 if(x<1) x=1; else
302 if(x>=252) x=0;
303 #endif
305 desc->pic_a[i]->bmp[j]=x;
306 // desc->pic_b[i]->bmp[j]=0; // hack
308 mp_msg(MSGT_OSD, MSGL_DBG2, "DONE!\n");
310 if(!desc->height) desc->height=desc->pic_a[i]->h;
313 j='_';if(desc->font[j]<0) j='?';
314 for(i=0;i<65536;i++)
315 if(desc->font[i]<0){
316 desc->start[i]=desc->start[j];
317 desc->width[i]=desc->width[j];
318 desc->font[i]=desc->font[j];
320 desc->font[' ']=-1;
321 desc->width[' ']=desc->spacewidth;
323 mp_msg(MSGT_OSD, MSGL_V, "Bitmap font %s loaded successfully! (%d chars)\n",fname,chardb);
325 return desc;
327 fail_out:
328 if (f)
329 fclose(f);
330 if (desc->fpath)
331 free(desc->fpath);
332 if (desc->name)
333 free(desc->name);
334 if (desc)
335 free(desc);
336 return NULL;
339 #if 0
340 int main(void){
342 read_font_desc("high_arpi.desc",1);
345 #endif