要求:原操做系統代碼裏只是支持了日語顯示,須要作的是實現對這個系統的漢字全角支持。html
hzk16的介紹以及簡單的使用方法數組
HZK16字庫是符合GB2312標準的16×16點陣字庫,HZK16的GB2312-80支持的漢字有6763個,符號682個。其中一級漢字有3755個,按聲序排列,二級漢字有3008個,按偏旁部首排列。咱們在一些應用場合根本用不到這麼多漢字字模,因此在應用時就能夠只提取部分字體做爲己用。app
HZK16字庫裏的16×16漢字一共須要256個點來顯示,也就是說須要32個字節才能達到顯示一個普通漢字的目的。函數
咱們知道一個GB2312漢字是由兩個字節編碼的,範圍爲A1A1~FEFE。A1-A9爲符號區,B0到F7爲漢字區。每個區有94個字符(注意:這只是編碼的許可範圍,不必定都有字型對應,好比符號區就有不少編碼空白區域)。下面以漢字「我」爲例,介紹如何在HZK16文件中找到它對應的32個字節的字模數據。測試
前面說到一個漢字佔兩個字節,這兩個中前一個字節爲該漢字的區號,後一個字節爲該字的位號。其中,每一個區記錄94個漢字,位號爲該字在該區中的位置。因此要找到「我」在hzk16庫中的位置就必須獲得它的區碼和位碼。(爲了區別使用了區碼和區號,實際上是一個東西,別被我誤導了)字體
區碼:區號(漢字的第一個字節)-0xa0 (由於漢字編碼是從0xa0區開始的,因此文件最前面就是從0xa0區開始,要算出相對區碼)編碼
位碼:位號(漢字的第二個字節)-0xa0spa
這樣咱們就能夠獲得漢字在HZK16中的絕對偏移位置:操作系統
offset=(94*(區碼-1)+(位碼-1))*32code
註解:一、區碼減1是由於數組是以0爲開始而區號位號是以1爲開始的
二、(94*(區號-1)+位號-1)是一個漢字字模佔用的字節數
三、最後乘以32是由於漢字庫文應從該位置起的32字節信息記錄該字的字模信息(前面提到一個漢字要有32個字節顯示)
有了偏移地址就能夠從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天操做系統支持中文。