UEFI-edk2源码中默认只有英文和法文的字库,在UI界面上或者shell终端打印中文字符串,则无法显示。例如,上一篇博客中的TestoneApp.cpp中,增加一行带中文字符串的打印:

Print(L"Hello, world!\r\n");Print(L"UEFI-dek 你好..!\r\n");

实际上运行的效果如下图所示。

也就是中文没有显示出来,这是因为UEFI采用位图(点阵图像)的方式进行字符显示,而edk2源码中默认只有英文和法文字库的位图数据。

一、字符编码介绍

UEFI中采用UCS-2的编码方式,即固定两个字节存一个字符的编码,需要显示该字符到显示器时,会通过字符编码来查询它的位图数据,从而实现字符显示。edk2源码中默认采用SimpleFont格式的点阵字体,字体点阵数据位于

MdeModulePkg\Universal\Console\GraphicsConsoleDxe\ LaffStd.c

SimpleFont是一种点阵字体,有两种格式,窄体(8x19)和宽体(16x19),英文字符串用窄体显示,如下图就是英文字符的字体点阵数据存放的数组。

从点阵数据中可以看出,字符编码为0x0020的点阵数据为全0,因为0x20是空格的ascii码,空格什么也不用显示,所以为全0。字符编码为0x0021对应为感叹号的ascii字符,如下图所示,将它的数据展开,由1组成的位图显示正好是个感叹号。

二、增加中文字库

中文需要用到16x19的宽体才能够容纳显示,如何获取中文字库的点阵数据呢?这里我们参考《UEFI原理与编程》原著作者戴正华写的代码,源码链接:https://github.com/zhenghuadai/uefi-programming/tree/master/book/GUIbasics/font/SimpleFont。
源码中example.data文件中包含了字符编码从0x4e00到0x9fa4的所有点阵数据,基本囊括了我们后续要用到的显示中文字库。接下需要做的就是这个中文字库数据注册到UEFI的SimpleFontPackageList中。
源码main.c文件中写了完整的实现过程,基本与edk2源码目录下"MdeModulePkg/Universal/Console/GraphicsConsoleDxe/GraphicsConsole.c"文件写的过程一致,先新建一个SimplifiedFont的结构体,将结构体成员NumberOfWideGlyphs指向一段新申请的空间,然后将点阵数组中的数据拷贝到申请的空间中,最后将SimplifiedFont结构体添加到SimpleFontPackageList中。如下所示是戴正华所写的源码目录下main.c文件中的一段程序。

EFI_STATUS CreatesimpleFontPkg(EFI_WIDE_GLYPH* WideGlyph, UINT32 SizeInBytes)
{EFI_STATUS Status; EFI_HII_SIMPLE_FONT_PACKAGE_HDR *simpleFont; UINT8 *Package; // Locate HII Database Protocol EFI_HII_DATABASE_PROTOCOL *HiiDatabase = 0; Status = gBS->LocateProtocol (  &gEfiHiiDatabaseProtocolGuid,  NULL,  (VOID **) &HiiDatabase  ); UINT32    packageLen = sizeof (EFI_HII_SIMPLE_FONT_PACKAGE_HDR) + SizeInBytes + 4; Package = (UINT8*)AllocateZeroPool (packageLen); WriteUnaligned32((UINT32 *) Package,packageLen); simpleFont = (EFI_HII_SIMPLE_FONT_PACKAGE_HDR *) (Package + 4); simpleFont->Header.Length = (UINT32) (packageLen - 4); simpleFont->Header.Type = EFI_HII_PACKAGE_SIMPLE_FONTS; simpleFont->NumberOfNarrowGlyphs = 0;simpleFont->NumberOfWideGlyphs = (UINT16) (SizeInBytes / sizeof (EFI_WIDE_GLYPH)); UINT8 * Location = (UINT8 *) (&simpleFont->NumberOfWideGlyphs + 1); CopyMem (Location, WideGlyph, SizeInBytes); EFI_HII_HANDLE  simpleFontHiiHandle = HiiAddPackages ( &gSimpleFontPackageListGuid,NULL, Package, NULL ); if(simpleFontHiiHandle == NULL) {return (EFI_STATUS)-1;}FreePool (Package); return EFI_SUCCESS;
}

三、UEFI SHELL中显示中文

首先我们参考MdeModulePkg/Universal/Console/GraphicsConsoleDxe/GraphicsConsole.c中添加英文字库的方式,先在MdeModulePkg/Universal/Console/目录下新建一个HansFontDxe目录,这里取名HansFont表示它是一个中文字库。在该目录下新建一个HansFontDxe.inf描述文件,同时把戴正华写的源码目录下的main.c程序文件和example.data字库数据文件拷贝过来。HansFontDxe.inf文件内容如下

[Defines]INF_VERSION                    = 0x00010006BASE_NAME                      = HansFontDxeFILE_GUID                      = 4ea97c46-7491-4dfd-b442-74798713ce5fVERSION_STRING                 = 0.1MODULE_TYPE                    = UEFI_DRIVER ENTRY_POINT                    = LoadFont [Sources]main.c[Packages]MdePkg/MdePkg.decMdeModulePkg/MdeModulePkg.dec[LibraryClasses]UefiDriverEntryPointUefiLibHiiLib  BaseMemoryLibUefiHiiServicesLib[Protocols]gEfiPciIoProtocolGuidgEfiGraphicsOutputProtocolGuidgEfiHiiDatabaseProtocolGuidgEfiHiiImageProtocolGuidgEfiHiiConfigRoutingProtocolGuidgEfiHiiStringProtocolGuidgEfiSimplePointerProtocolGuid[BuildOptions]GCC:*_*_IA32_CC_FLAGS = -O2 MSFT:*_*_*_CC_FLAGS = -DNEFI_SHELL_FILE_PROTOCOL -DUSE_SIMPLE_STDIO /GL- /wd4804 /wd4201 /Oi- /FAs

"MODULE_TYPE = UEFI_DRIVER "指明了这个目录下的工程为一个UEFI驱动模块。
"ENTRY_POINT = LoadFont"指明了驱动的入口函数,该函数位于main.c中,它会调用CreatesimpleFontPkg()函数,从而注册中文字库。
接下来参考"MdeModulePkg/Universal/Console/GraphicsConsoleDxe/GraphicsConsoleDxe.inf"文件的引用方式,将"MdeModulePkg/Universal/Console/HansFontDxe/HansFontDxe.inf"文件应用添加到EmulatorPkg/EmulatorPkg.dsc和EmulatorPkg/EmulatorPkg.fdf中,如下图所示:


其中添加到Emulator.fdf文件中的*.inf模块文件描述,编译时会把该模块编译的最终的efi固件中,如果该模块是驱动类型的话,在固件初始化的时候,会自动调用驱动模块的ENTRY_POINT指定的函数。
现在重新回到TestoneApp.cpp程序中来,重新编译运行Emulator工程,进入到shell后,再次运行TestoneApp.efi程序的话,就能显示中文字符了,结果如下图所示:

四、UEFI设置界面中显示中文

默认状态下UEFI的设置界面如下图所示:

搜索对应的英文字符串可以在源码中找出主界面下显示字符串的定义在"MdeModulePkg/Application/UiApp/FrontPageStrings.uni"文件中,里面由英文和法文的字符串定义显示,我们可以把里面的法文字符串删掉,替换成中文的(不删掉法文,直接添加中文定义也可以)。更改比较简单,如下图所示:

添加完中文字符串后,需要到EmulatorPkg/EmulatorPkg.dsc工程文件中添加语言设置,设置下可选语言与默认语言,我们这里只保留中文和英文语言的选择即可。

更改完成后,再次编译运行EmulatorPkg工程,进入到UEFI配置界面,显示效果如下图所示:

可以看到,中文显示并不完整,只显示了左边一半,同时,部分菜单栏也没有翻译过来。中文字符显示不完整的问题,应该与UiApp中计算显示位置与显示长度的计算问题有关,这里最快解决办法就是在中文字符串后面加同等数量的空格就行,比如STR_RESET_STRING,把它对应的中文字符串从"复位“改为"复位 ",比较长的字符串需要换行显示的也是这样更改,示例程序如下:

...
#string STR_LANGUAGE_SELECT_HELP       #language en-US  "This is the option one adjusts to change the language for the current system"#language zh-Hans "这是一个更改                当前系统语言              的选项    "
#string STR_MISSING_STRING             #language en-US  "Missing String"#language zh-Hans  "字符串缺失     "
#string STR_EMPTY_STRING               #language en-US  ""#language zh-Hans  ""
#string STR_RESET_STRING               #language en-US  "Reset"#language zh-Hans  "复位  "
#string STR_RESET_STRING_HELP          #language en-US  "Reset the current setting."#language zh-Hans  "复位当前的设置      "
...

部分菜单栏没有显示出中文,是英文它们的翻译对应字符串并不在FrontPageString.uni,继续在edk2工程中搜索对应的英文字符串,然后添加它的中文字符串即可,比如"Device Manager"字符串就位于"MdeModulePkg/Library/DeviceManagerUiLib/DeviceManagerStrings.uni"文件中,更改示例如下:

中文字符串添加完成后,再次编译运行EmulatorPkg工程,最终UEFI配置界面显示如下图所示:

可以看到,UEFI配置界面增加了中文显示。

3.UEFI-edk2 增加中文显示相关推荐

  1. 为Linux系统增加中文字体支持:解决显示问题的三种方法

    Linux 增加中文字体支持 在使用 Linux 操作系统的过程中,用户经常会遇到中文字体显示问题,导致一些界面上的文字不能正确显示.为了解决这个问题,我们可以通过以下方法来增加中文字体支持. 一.安 ...

  2. python画图中文显示_解决Linux系统中python matplotlib画图的中文显示问题

    最近想学习一些python数据分析的内容,就弄了个爬虫爬取了一些数据,并打算用Anaconda一套的工具(pandas, numpy, scipy, matplotlib, jupyter)等进行一些 ...

  3. 让英文版的LINUX 支持中文显示

    (以REDHAT为例) 1.首先查看系统中有没有安装中文支持软件包   rpm -qa |grep fonts-chinese 如果命令执行后能够查询到相应的包,那么说明已经安装过包:反之则没有安装 ...

  4. Xna支持中文显示方法归纳

    Xna不同于DirectX,因其内部并未提供类似于D3DFont的机制(据说之所以会这样做,也是考虑到与Xbox360兼容的缘故),使得显示中文变得极为不便. 虽然如此,要实现Xna下的中文显示依然存 ...

  5. python中文显示不出来_彻底解决Python里matplotlib不显示中文的问题

    在很长一段时间里用Python绘图,matplotlib都不能很好的显示中文,起初是认为我的pycharm里的设置问题,但是发现同样的问题在spyder里也同样的出现了,虽然有的地方可以用英文实在不行 ...

  6. Mac字体路径,解决matplotlib中文显示问题,超级好用~

    不需要用fc-list,找到字体路径,解决matplotlib中文显示问题. 1.找到字体路径 mac字体的路径一般都是在:/System/Library/Fonts 下 如果不放心可在:访达> ...

  7. web项目查询mysql中文显示为问号_MySQL 表中的中文显示为问号

    文章目录 1 问题:MySQL数据库中的中文显示为问号 2 解决思路 2.1 查看字符集 2.2 修改my.ini 2.3 重启mysql 2.4 验证字符集生效 3 其他 3.1 修改表的编码格式 ...

  8. python设置折线图标题字体为宋体_python中matplotlib画折线图实例(坐标轴数字、字符串混搭及标题中文显示)...

    最近在用python中的matplotlib画折线图,遇到了坐标轴 "数字+刻度" 混合显示.标题中文显示.批量处理等诸多问题.通过学习解决了,来记录下.如有错误或不足之处,望请指 ...

  9. DbVisualizer Pro 9.1.1连DB2数据库中文显示乱码的解决【一篇就够】

    一.修改安装目录下配置文件 安装目录/resources/dbvis-custom.prefs 增加配置如下: dbvis.grid.encode=true dbvis.grid.fromEncode ...

  10. python 画曲线(基本的坐标轴,刻度,网格,解决中文显示乱码)

    参考链接: 本文参考的一些链接: https://www.cnblogs.com/liutongqing/p/6985805.html https://blog.csdn.net/lyq_csdn/a ...

最新文章

  1. 【html、CSS、javascript-8】JavaScript作用域
  2. JavaSE入门学习51:多线程编程(二)
  3. vagrant boxes
  4. 如何只下载一个github项目的某一特定文件夹或文件
  5. python可以下载百度文库_百度文库随便下载,解除限制
  6. 初中文化能学编程吗_网页编程课程来了,确定不来pick一下!!!|科创辅学进行时...
  7. 力扣530. 二叉搜索树的最小绝对差(JavaScript)
  8. python获取app信息的库_Python学习教程:另辟蹊径,appium抓取app应用数据了解一下...
  9. elementui el-radio-group 分组排序问题
  10. weiPe系统启动盘还原
  11. 【码上实战】【立体匹配系列】经典SGM:(3)代价聚合
  12. selected和checked区别
  13. zencart bohase 模板
  14. 纳什效率系数与可决系数的差异
  15. 埃尔米特函数的计算(C++)
  16. poj1639 Picnic Planning 最小度数限制生成树
  17. 代表着团结幸福平安的中国结绳
  18. 干货长文《对抗中的主动防御》
  19. 增值电信业务经营许可证-仅限互联网信息服务-信息服务业务许可证如何办理,办理流程汇总
  20. STM32基础课程笔记

热门文章

  1. 视频批量消重 短视频解析去水印在线
  2. notepad linux版本,Notepad++ Linux版
  3. pgz-sbv-gf极狐低代码平台发布
  4. (十)统计表速查(标准正态分布、t分布、卡方分布)
  5. 2021微信公开课PRO:微信视频号首次公布运营规则,“点赞”表情成视频号年度表情
  6. 对voc2007数据集进行person的单类数据抽取
  7. Mac下librdkafka下载安装
  8. 【毕业设计8】基于STM32的红外测距系统
  9. 囧从流氓同事的河蟹喜酒归来......
  10. vue H5 唤醒app