freetype库的移植

简介: freetype库的移植

freetype库的移植

freetype依赖于libpng,而libpng又依赖zlib,所以我们本次的移植实际上是需要移植三个库,每个的移植操作是类似的,只不过顺序不能颠倒

1. 移植zlib

1.1 下载zlib库

下载地址: zlib库官网

1.2 将zlib移动到linux虚拟机上,并解压,解压后进入得到的目录

1. tar -xzf zlib-1.2.11.tar.gz
2. cd zlib-l.2.11/

1.3 使用configure来配置makefile

1. export CC=arm-linux-gnueabihf-gcc   # 你的编译工具,我的和你的可能不一样
2. ./configure --prefix=/home/hxd/tools/zlib  # prefix用于指定库安装的位置

1.4 使用make && make install 安装库

1. make
2. make install

安装成功后,会在你刚刚设置的安装目录下得到三个目录,分别是include/, lib/, share/

1.5 将得到的include/目录下的所有文件拷贝到你的编译工具链的默认寻找头文件目录的下,将得到的lib/目录下的所有文件拷贝到你的编译工具链的默认寻找库文件的目录下(这样做的目的是为了以后编译的时候不需要自己手动来添加路径,所以这一步也可以不做)

1. echo "main(){}" | arm-linux-gnueabihf-gcc -E -v -    # 该命令可以查看编译工具的默认查找路径,你自己任选一个合适的

include路径,我是选的我的交叉工具链的路径/usr/lib/gcc-cross/arm-linux-gnueabihf/9/include



c22d0edcfe0047709281e98c043bbb86.png

cp -rfd /home/hxd/tools/zlib/include/* /usr/lib/gcc-cross/arm-linux-gnueabihf/9/include  # 加上-d的目的是将链接文件也这样拷贝过去,节约空间

lib路径, 我是选的我的交叉工具链的路径/usr/lib/gcc-cross/arm-linux-gnueabihf/9

0d936092a9ae45ebbcb0c8a6b8d8eed4.png

cp -rf /home/hxd/tools/zlib/lib/* /usr/lib/gcc-cross/arm-linux-gnueabihf/9/

1.6 将编译好的zlib库移植到开发板

只需要将lib/目录下的所有文件拷贝到开发板的/usr/lib/或/lib/下

1. 先将zlib文件夹复制nfs的目录下,然后开发板挂载该nfs
2. rm -rf /usr/lib/libz* /lib/libz*   # 如果以前存在该库,可能会互相干扰,所以删除以前的库
3. cp -rfd /mnt/zlib/lib/* /usr/lib

2. 移植libpng库

2.1 下载libpng库

下载地址: 官网

2.2 将libpng移动到linux虚拟机上,并解压,解压后进入得到的目录

1. tar -xJf libpng-1.6.37.tar.xz
2. cd libpng-l.6.37

2.3 执行./configure

if 之前完成了1.5操作
  1. ./configure --host=arm-linux-gnueabihf --prefix=/home/hxd/tools/libpng
else
  1. export CFLAGS="$CFLAGS -I/home/hxd/tools/zlib/include"   # 与你的zlib安装路径有关
  2. export CPPFLAGS="$CPPFLAGS -I/home/hxd/tools/zlib/include"
  3. export LDFLAGS="$LDFLAGS -L/home/hxd/tools/zlib/lib"
  4. ./configure --host=arm-linux-gnueabihf --prefix=/home/hxd/tools/libpng

2.4 安装

make && make install

2.5 配置到编译链中,方便后面的编译使用(具体路径1.5有解释), 还需要将freetype2目录下的所有文件移动到freetype2同级目录

1. cp -rfd /home/hxd/tools/libpng/include/* /usr/lib/gcc-cross/arm-linux-gnueabihf/9/include  # 加上-d的目的是将链接文件也这样拷贝过去,节约空间
2. cp -rf /home/hxd/tools/libpng/lib/* /usr/lib/gcc-cross/arm-linux-gnueabihf/9/
3. cp -rf /usr/lib/gcc-cross/arm-linux-gnueabihf/9/include/freetype2/* /usr/lib/gcc-cross/arm-linux-gnueabihf/9/include/

2.6 移植到开发板

需要将lib/目录下的所有文件拷贝到开发板的/usr/lib/或/lib/下

1. 先将libpng文件夹复制nfs的目录下,然后开发板挂载该nfs
2. rm -rf /usr/lib/libpng* /lib/libpng*   # 如果以前存在该库,可能会互相干扰,所以删除以前的库
3. cp -rfd /mnt/libpng/lib/* /usr/lib

3. 移植freetype库

3.1 下载

下载地址: 官网

3.2 移动到linux虚拟机上,解压后,进入该目录

1. tar -xJf freetype-2.10.2.tar.xz
2. cd freetype-2.10.2

3.3 使用configure配置makefile

if 前面的1.5和2.5都配置了
  1. ./configure --prefix=/home/hxd/tools/freetype --host=arm-linux-gnueabihf
elif 前面都没配置
  1. export CFLAGS="$CFLAGS -I/home/hxd/tools/zlib/include -I/home/hxd/tools/libpng/include"   # 与你的zlib和libpng安装路径有关
  2. export CPPFLAGS="$CPPFLAGS -I/home/hxd/tools/zlib/include -I/home/hxd/tools/libpng/include"
  3. export LDFLAGS="$LDFLAGS -L/home/hxd/tools/zlib/lib -L/home/hxd/tools/libpng/lib"
  4. ./configure --prefix=/home/hxd/tools/freetype --host=arm-linux-gnueabihf

3.4 安装

make && make install

3.5 配置到交叉编译工具链里面,方便后面的编译使用(具体路径1.5有解释)

1. cp -rfd /home/hxd/tools/freetype/include/* /usr/lib/gcc-cross/arm-linux-gnueabihf/9/include  # 加上-d的目的是将链接文件也这样拷贝过去,节约空间
2. cp -rf /home/hxd/tools/freetype/lib/* /usr/lib/gcc-cross/arm-linux-gnueabihf/9/

3.6 移植到开发板上

1. 先将libpng文件夹复制nfs的目录下,然后开发板挂载该nfs
2. rm -rf /usr/lib/libfreetype* /lib/libfreetype*   # 如果以前存在该库,可能会互相干扰,所以删除以前的库
3. cp -rfd /mnt/libpng/lib/* /usr/lib

4. 使用

4.1 如果没有配置1.5,2.5,3.5

编译示例:

arm-linux-gnueabihf-gcc test.c -o test -lz -lm -lpng -freetype -I/home/hxd/tools/freetype/include/freetype2 -L/home/hxd/tools/freetype/lib -L/home/hxd/tools/zlib/lib -L/home/hxd/tools/libpng/lib

4.2 如果配置了

编译示例:

arm-linux-gnueabihf-gcc test.c -o test -lz -lm -lpng -freetype

5. 实例(源自韦哥的代码)

#include <sys/mman.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <linux/fb.h>
#include <fcntl.h>
#include <stdio.h>
#include <string.h>
#include <math.h>
#include <wchar.h>
#include <sys/ioctl.h>
#include <ft2build.h>
#include FT_FREETYPE_H
#include FT_GLYPH_H
int fd_fb;
struct fb_var_screeninfo var;   /* Current var */
struct fb_fix_screeninfo fix;   /* Current fix */
int screen_size;
unsigned char *fbmem;
unsigned int line_width;
unsigned int pixel_width;
/* color : 0x00RRGGBB */
void lcd_put_pixel(int x, int y, unsigned int color)
{
    unsigned char *pen_8 = fbmem+y*line_width+x*pixel_width;
    unsigned short *pen_16; 
    unsigned int *pen_32;   
    unsigned int red, green, blue;  
    pen_16 = (unsigned short *)pen_8;
    pen_32 = (unsigned int *)pen_8;
    switch (var.bits_per_pixel)
    {
        case 8:
        {
            *pen_8 = color;
            break;
        }
        case 16:
        {
            /* 565 */
            red   = (color >> 16) & 0xff;
            green = (color >> 8) & 0xff;
            blue  = (color >> 0) & 0xff;
            color = ((red >> 3) << 11) | ((green >> 2) << 5) | (blue >> 3);
            *pen_16 = color;
            break;
        }
        case 32:
        {
            *pen_32 = color;
            break;
        }
        default:
        {
            printf("can't surport %dbpp\n", var.bits_per_pixel);
            break;
        }
    }
}
/**********************************************************************
 * 函数名称: draw_bitmap
 * 功能描述: 根据bitmap位图,在LCD指定位置显示汉字
 * 输入参数: x坐标,y坐标,位图指针
 * 输出参数: 无
 * 返 回 值: 无
 * 修改日期        版本号     修改人        修改内容
 * -----------------------------------------------
 * 2020/05/12        V1.0     zh(angenao)         创建
 ***********************************************************************/ 
void
draw_bitmap( FT_Bitmap*  bitmap,
             FT_Int      x,
             FT_Int      y,
             unsigned int color)
{
    FT_Int  i, j, p, q;
    FT_Int  x_max = x + bitmap->width;
    FT_Int  y_max = y + bitmap->rows;
    //printf("x = %d, y = %d\n", x, y);
    for ( j = y, q = 0; j < y_max; j++, q++ )
    {
        for ( i = x, p = 0; i < x_max; i++, p++ )
        {
            if ( i < 0      || j < 0       ||
                i >= var.xres || j >= var.yres )
            continue;
            //image[j][i] |= bitmap->buffer[q * bitmap->width + p];
            if (bitmap->buffer[q * bitmap->width + p]) lcd_put_pixel(i, j, color);
        }
    }
}
int compute_string_bbox(FT_Face face, wchar_t *wstr, FT_BBox  *abbox)
{
    int i;
    int error;
    FT_BBox bbox;
    FT_BBox glyph_bbox;
    FT_Vector pen;
    FT_Glyph  glyph;
    FT_GlyphSlot slot = face->glyph;
    /* 初始化 */
    bbox.xMin = bbox.yMin = 32000;
    bbox.xMax = bbox.yMax = -32000;
    /* 指定原点为(0, 0) */
    pen.x = 0;
    pen.y = 0;
    /* 计算每个字符的bounding box */
    /* 先translate, 再load char, 就可以得到它的外框了 */
    for (i = 0; i < wcslen(wstr); i++)
    {
        /* 转换:transformation */
        FT_Set_Transform(face, 0, &pen);
        /* 加载位图: load glyph image into the slot (erase previous one) */
        error = FT_Load_Char(face, wstr[i], FT_LOAD_RENDER);
        if (error)
        {
            printf("FT_Load_Char error\n");
            return -1;
        }
        /* 取出glyph */
        error = FT_Get_Glyph(face->glyph, &glyph);
        if (error)
        {
            printf("FT_Get_Glyph error!\n");
            return -1;
        }
        /* 从glyph得到外框: bbox */
        FT_Glyph_Get_CBox(glyph, FT_GLYPH_BBOX_TRUNCATE, &glyph_bbox);
        /* 更新外框 */
        if ( glyph_bbox.xMin < bbox.xMin )
            bbox.xMin = glyph_bbox.xMin;
        if ( glyph_bbox.yMin < bbox.yMin )
            bbox.yMin = glyph_bbox.yMin;
        if ( glyph_bbox.xMax > bbox.xMax )
            bbox.xMax = glyph_bbox.xMax;
        if ( glyph_bbox.yMax > bbox.yMax )
            bbox.yMax = glyph_bbox.yMax;
        /* 计算下一个字符的原点: increment pen position */
        pen.x += slot->advance.x;
        pen.y += slot->advance.y;
    }
    /* return string bbox */
    *abbox = bbox;
}
int display_string(FT_Face face, wchar_t *wstr, int lcd_x, int lcd_y, unsigned int color, int angle)
{
    int i;
    int error;
    FT_BBox bbox;
    FT_Vector pen;
    FT_Glyph  glyph;
    FT_GlyphSlot slot = face->glyph;
    FT_Matrix     matrix;
    /* 把LCD坐标转换为笛卡尔坐标 */
    int x = lcd_x;
    int y = var.yres - lcd_y;
    /* 计算外框 */
    compute_string_bbox(face, wstr, &bbox);
    /* 反推原点 */
    pen.x = (x - bbox.xMin) * 64; /* 单位: 1/64像素 */
    pen.y = (y - bbox.yMax) * 64; /* 单位: 1/64像素 */
    /* 旋转角度 */
    /* set up matrix */
    matrix.xx = (FT_Fixed)( cos( angle ) * 0x10000L );
    matrix.xy = (FT_Fixed)(-sin( angle ) * 0x10000L );
    matrix.yx = (FT_Fixed)( sin( angle ) * 0x10000L );
    matrix.yy = (FT_Fixed)( cos( angle ) * 0x10000L );
    /* 处理每个字符 */
    for (i = 0; i < wcslen(wstr); i++)
    {
        /* 转换:transformation */
        FT_Set_Transform(face, &matrix, &pen);
        /* 加载位图: load glyph image into the slot (erase previous one) */
        error = FT_Load_Char(face, wstr[i], FT_LOAD_RENDER);
        if (error)
        {
            printf("FT_Load_Char error\n");
            return -1;
        }
        /* 在LCD上绘制: 使用LCD坐标 */
        draw_bitmap( &slot->bitmap,
                        slot->bitmap_left,
                        var.yres - slot->bitmap_top, color);
        /* 计算下一个字符的原点: increment pen position */
        pen.x += slot->advance.x;
        pen.y += slot->advance.y;
    }
    return 0;
}
int main(int argc, char **argv)
{
    wchar_t *wstr = L"你好世界!Hello World!";
    FT_Library    library;
    FT_Face       face;
    int error;
    FT_BBox bbox;
    int font_size = 24;
    int angle = 0;
    int lcd_x, lcd_y;
    if (argc < 4)
    {
        printf("Usage : %s <font_file> <lcd_x> <lcd_y> [font_size] [angle]\n", argv[0]);
        return -1;
    }
    lcd_x = strtoul(argv[2], NULL, 0);      
    lcd_y = strtoul(argv[3], NULL, 0);      
    if (argc == 5)
        font_size = strtoul(argv[4], NULL, 0);
    if (argc == 6)
    {
        font_size = strtoul(argv[4], NULL, 0);
        angle = atoi(argv[5]);    
    }
    fd_fb = open("/dev/fb0", O_RDWR);
    if (fd_fb < 0)
    {
        printf("can't open /dev/fb0\n");
        return -1;
    }
    if (ioctl(fd_fb, FBIOGET_VSCREENINFO, &var))
    {
        printf("can't get var\n");
        return -1;
    }
    if (ioctl(fd_fb, FBIOGET_FSCREENINFO, &fix))
    {
        printf("can't get fix\n");
        return -1;
    }
    line_width  = var.xres * var.bits_per_pixel / 8;
    pixel_width = var.bits_per_pixel / 8;
    screen_size = var.xres * var.yres * var.bits_per_pixel / 8;
    fbmem = (unsigned char *)mmap(NULL , screen_size, PROT_READ | PROT_WRITE, MAP_SHARED, fd_fb, 0);
    if (fbmem == (unsigned char *)-1)
    {
        printf("can't mmap\n");
        return -1;
    }
    /* 清屏: 全部设为黑色 */
    memset(fbmem, 0, screen_size);
    error = FT_Init_FreeType( &library );              /* initialize library */
    error = FT_New_Face( library, argv[1], 0, &face ); /* create face object */
    FT_Set_Pixel_Sizes(face, font_size, 0);
    printf("font_size: %d\n", font_size);
    display_string(face, L"你好世界!Hello World!", lcd_x, lcd_y, 0x112233, angle);
    display_string(face, L"你好世界!Hello World!", lcd_x, lcd_y+font_size, 0x222233, angle);
    display_string(face, L"你好世界!Hello World!", lcd_x, lcd_y+font_size*2, 0x332233, angle);
    display_string(face, L"你好世界!Hello World!", lcd_x, lcd_y+font_size*3, 0x442233, angle);
    return 0;   
}



目录
相关文章
|
6月前
|
Linux
linux 交叉编译libpng,libjpeg库
linux 交叉编译libpng,libjpeg库
98 1
|
XML 编解码 自然语言处理
不需要熟悉,但需要了解的libiconv库
但是很多老式的计算机还在使用当地的传统的字符编码方式。而一些程序,例如邮件程序和浏览器必须能在这些不同的用户编码之间作转换。其他的一些程序则内置支持Unicode,以顺利支持国际化的处理,但是仍然有在Unicode和其他的传统编码之间转换的需求。GNU的libiconv就是为这两种应用设计的编码转换库。
不需要熟悉,但需要了解的libiconv库
|
Ubuntu 编译器 Windows
zlib开发笔记(四):zlib库介绍、编译windows vs2015x64版本和工程模板
zlib开发笔记(四):zlib库介绍、编译windows vs2015x64版本和工程模板
zlib开发笔记(四):zlib库介绍、编译windows vs2015x64版本和工程模板
|
Ubuntu 算法 Linux
移植Zlib,Libpng,FreeType详细步骤
移植Zlib,Libpng,FreeType详细步骤
523 0
|
Linux
tslib库的移植
tslib库的移植
172 0
|
Shell 编译器 Linux
zlib-1.2.11库、libpng-1.6.36库编译及交叉编译 —— 附带shell编译脚本及源码
zlib-1.2.11库、libpng-1.6.36库编译及交叉编译 —— 附带shell编译脚本及源码
548 0
zlib-1.2.11库、libpng-1.6.36库编译及交叉编译 —— 附带shell编译脚本及源码
ffmpeg支持的扩展库有哪些
ffmpeg支持的扩展库有哪些
181 0
|
Ubuntu 编译器 C语言
zlib开发笔记(三):zlib库介绍、在ubuntu上进行arm平台交叉编译
zlib开发笔记(三):zlib库介绍、在ubuntu上进行arm平台交叉编译
zlib开发笔记(三):zlib库介绍、在ubuntu上进行arm平台交叉编译
|
Ubuntu Windows
zlib开发笔记(二):zlib库介绍、ubuntu平台编译和工程模板
zlib开发笔记(二):zlib库介绍、ubuntu平台编译和工程模板
zlib开发笔记(二):zlib库介绍、ubuntu平台编译和工程模板