要求:原操作系統代碼里只是支持了日語顯示,需要做的是實現對這個系統的漢字全形支持。 hzk16的介紹以及簡單的使用方法 HZK16字型檔是符合GB2312標準的16×16點陣字型檔,HZK16的GB2312-80支持的漢字有6763個,符號682個。其中一級漢字有3755個,按聲序排列,二級漢字有300 ...
要求:原操作系統代碼里只是支持了日語顯示,需要做的是實現對這個系統的漢字全形支持。
hzk16的介紹以及簡單的使用方法
HZK16字型檔是符合GB2312標準的16×16點陣字型檔,HZK16的GB2312-80支持的漢字有6763個,符號682個。其中一級漢字有3755個,按聲序排列,二級漢字有3008個,按偏旁部首排列。我們在一些應用場合根本用不到這麼多漢字字模,所以在應用時就可以只提取部分字體作為己用。
HZK16字型檔里的16×16漢字一共需要256個點來顯示,也就是說需要32個位元組才能達到顯示一個普通漢字的目的。
我們知道一個GB2312漢字是由兩個位元組編碼的,範圍為A1A1~FEFE。A1-A9為符號區,B0到F7為漢字區。每一個區有94個字元(註意:這隻是編碼的許可範圍,不一定都有字型對應,比如符號區就有很多編碼空白區域)。下麵以漢字“我”為例,介紹如何在HZK16文件中找到它對應的32個位元組的字模數據。
前面說到一個漢字占兩個位元組,這兩個中前一個位元組為該漢字的區號,後一個位元組為該字的位號。其中,每個區記錄94個漢字,位號為該字在該區中的位置。所以要找到“我”在hzk16庫中的位置就必須得到它的區碼和位碼。(為了區別使用了區碼和區號,其實是一個東西,別被我誤導了)
區碼:區號(漢字的第一個位元組)-0xa0 (因為漢字編碼是從0xa0區開始的,所以文件最前面就是從0xa0區開始,要算出相對區碼)
位碼:位號(漢字的第二個位元組)-0xa0
這樣我們就可以得到漢字在HZK16中的絕對偏移位置:
offset=(94*(區碼-1)+(位碼-1))*32
註解:1、區碼減1是因為數組是以0為開始而區號位號是以1為開始的
2、(94*(區號-1)+位號-1)是一個漢字字模占用的位元組數
3、最後乘以32是因為漢字型檔文應從該位置起的32位元組信息記錄該字的字模信息(前面提到一個漢字要有32個位元組顯示)
有了偏移地址就可以從HZK16中讀取漢字編碼了
實現思路:
- 瞭解HZK編碼,理解一下符合GB2312標準的中文點陣字型檔文件HZK16;
- 下載中文GB2312的二進位點陣文件;
- 將HZK16.fnt文件放入nihongo文件夾中;
- 修改主makefile文件和app_make.txt文件,將原來裝載nihongo.fnt的語句替換成裝載HZK16.fnt即可;
- 修改bootpack.c文件,將之前分配的裝載日語字體的記憶體擴大,載入字型檔的文件名;
- 在haribote/graphic.c中添加支持漢字的代碼,增加一個函數用於顯示漢字;
- 修改putfonts8_asc函數里if (task->langmode == 3)語句塊;
- 測試程式。
- 註意:日文的編碼是分為左半部分和右半部分,而我們使用的HZK16是分為上半部分和下半部分的。
這裡其他的地方比較弄,第5步將大小修改一下,我的是nihongo = (unsigned char *) memman_alloc_4k(memman, 55*94*32);
第6步,要註意,HZK16是上下兩部分,不同於日文的左右兩部分的結構。
代碼如下:
void putfont32(char *vram, int xsize, int x, int y, char c, char *font1, char *font2) { int i,k,j,f; char *p, d ; j=0; p=vram+(y+j)*xsize+x; j++; //上半部分 for(i=0;i<16;i++) { for(k=0;k<8;k++) { if(font1[i]&(0x80>>k)) { p[k+(i%2)*8]=c; } } if(i%2==0){ for(k=0;k<4;k++){ f=p[k]; p[k]=p[7-k]; p[7-k]=f; } }else{ for(k=0;k<4;k++){ f=p[k+8]; p[k+8]=p[15-k]; p[15-k]=f; } } /* for(k=0;k<8/2;k++) { f=p[k+(i%2)*8]; p[k+(i%2)*8]=p[8-1-k+(i%2)*8]; p[8-1-k+(i%2)*8]=f; }*/ if(i%2) { p=vram+(y+j)*xsize+x; j++; } } //下半部分 for(i=0;i<16;i++) { for(k=0;k<8;k++) { if(font2[i]&(0x80>>k)) { p[k+(i%2)*8]=c; } } if(i%2==0){ for(k=0;k<4;k++){ f=p[k]; p[k]=p[7-k]; p[7-k]=f; } }else{ for(k=0;k<4;k++){ f=p[k+8]; p[k+8]=p[15-k]; p[15-k]=f; } } /*for(k=0;k<8/2;k++) { f=p[k+(i%2)*8]; p[k+(i%2)*8]=p[8-1-k+(i%2)*8]; p[8-1-k+(i%2)*8]=f; }*/ if(i%2) { p=vram+(y+j)*xsize+x; j++; } } return; }
運行結果,我們在euc.txt中加入一些漢字。
參考資料:
1.https://www.cnblogs.com/wunaozai/p/3858473.html 30天操作系統支持中文。