嵌入式linux之go语言开发(十三)LittlevGL,漂亮的嵌入式GUI的go语言绑定

简介: 嵌入式linux之go语言开发(十三)LittlevGL,漂亮的嵌入式GUI的go语言绑定

LittleVgl,一款开源嵌入式图形用户界面库。https://littlevgl.cn/


使用简单小巧,界面也漂亮。很适合用在嵌入式上。在Stm32单片机上都能跑,只要实现了基础的函数很好移植。


看下stm32单片机下的效果:



业余时间移植到了嵌入式linux上,使用帧缓冲Frambuffer,显示效果还不错。


计划选用LittleVgl作为嵌入式的漂亮的UI。


其他GUI如QT,miniGUI,周立功的AWTK也都有了解过。界面UI是强大,但是就是有点儿庞大,在简单的嵌入式界面上不想花太多精力和时间。毕竟不是手机,不需要太多人机交互,要的是简单小巧,够用即可。


之前本想在Ubuntu14上玩儿一下AWTK,但是编译依赖一大堆,最后放弃了。查看官网才知道需要使用Ubuntu16以上系统且安装好几个不知道什么用途的依赖,后续有机会再尝试吧。他们能不能出个纯c代码的精简版?可能太强大和支持N多平台的缘故吧,代码依赖很多三方库,这点儿有点儿不让人喜欢,不够小巧。


如图:





但是,还是不够好用。


计划使用go加上面向对象的思想,对其进行进一步的封装,让其更简单好用一点儿。


如果使用上类似于python上的绑定那样简单的话,该是多么美好的事情,如:


$ python3
Python 3.6.5 (default, May  9 2018, 10:02:20) 
[GCC 4.2.1 Compatible Apple LLVM 9.1.0 (clang-902.0.39.1)] on darwin
Type "help", "copyright", "credits" or "license" for more information.
>>> 
>>> from pylvgl import Demo, lvgl
>>> d = Demo()
>>>
>>> b1 = lvgl.Btn(lvgl.scr_act())        
>>> b1.set_size(200,50)
>>> b1.align(b1.get_parent(), lvgl.ALIGN_IN_LEFT_MID, 0, 0)
>>>
>>> l1 = lvgl.Label(b1)
>>> l1.set_text('LittlevGL')
>>>
>>> lm = lvgl.Lmeter(lvgl.scr_act())
>>> lm.align(cal.get_parent(), lvgl.ALIGN_IN_LEFT_MID, 300, 0)
>>> lm.set_range(0,100)
>>> lm.set_value(10)
>>>
>>> style_lm1.line_width = 4
>>> style_lm1.body_main_color = 0x2200
>>> style_lm1.body_grad_color = 0x0022
>>> lm.set_style(style_lm1)
>>>
>>> lb2 = lvgl.Label(lm)
>>> lb2.set_text('10%')
>>> lb2.align(lm, lvgl.ALIGN_CENTER, 0, 0)



已经初步验证了,go的绑定可行,使用也变得简单啦


如:


package main
import (
  "golvgl/lvgl"
  "log"
  "time"
)
func main() {
  log.Println("Hello Go,LittlevGL")
  lb := lvgl.Label(lvgl.ScrAct(), nil)
  lb.SetText("hello world,go test 222")
  lb.Align(nil, lvgl.LV_ALIGN_CENTER, 0, 0)
  log.Println("Hello Go Over 222")
  for true {
    lvgl.TickInc(5)
    lvgl.TaskHandler()
    time.Sleep(5000)
  }
}


随后代码会放在 github上,希望感兴趣的有志之士能一块儿参与完善下封装。


https://github.com/yangyongzhen/golvgl.git


package lvgl
/*
#cgo CFLAGS: -Iinclude/lvgl
#include "lvgl.h"
#cgo LDFLAGS: -Llibs -llvgl
#include <stdlib.h>
extern void disp_init(void);
extern void fbdev_init(void);
extern void fbdev_flush(void);
void disp_init(){
  lv_disp_drv_t disp_drv;
    lv_disp_drv_init(&disp_drv);
    disp_drv.disp_flush = fbdev_flush;
    lv_disp_drv_register(&disp_drv);
}
*/
import "C"
import "unsafe"
// Lvobj ...
type Lvobj C.struct__lv_obj_t
const (
  LV_ALIGN_CENTER C.uchar = C.LV_ALIGN_CENTER
)
func init() {
  C.lv_init()
  C.fbdev_init()
  C.disp_init()
}
// ScrAct ...
func ScrAct() *Lvobj {
  return (*Lvobj)(unsafe.Pointer(C.lv_scr_act()))
}
// Label ...
func Label(par, copy *Lvobj) *Lvobj {
  var p1 *C.struct__lv_obj_t
  var p2 *C.struct__lv_obj_t
  p1 = (*C.struct__lv_obj_t)(unsafe.Pointer(par))
  p2 = (*C.struct__lv_obj_t)(unsafe.Pointer(copy))
  return (*Lvobj)(unsafe.Pointer(C.lv_label_create(p1, p2)))
}
// SetText ...
func (obj *Lvobj) SetText(str string) {
  C.lv_label_set_text((*C.struct__lv_obj_t)(unsafe.Pointer(obj)), C.CString(str))
}
// Align ...
func (obj *Lvobj) Align(base *Lvobj, align C.uchar, x int, y int) {
  var ba *C.struct__lv_obj_t
  ba = (*C.struct__lv_obj_t)(unsafe.Pointer(base))
  C.lv_obj_align((*C.struct__lv_obj_t)(unsafe.Pointer(obj)), ba, align, C.short(x), C.short(y))
}
// TickInc ...
func TickInc(tick int) {
  C.lv_tick_inc(C.uint(tick))
}
// TaskHandler ...
func TaskHandler() {
  C.lv_task_handler()
}


其中遇到的问题总结:




void * 对应go中的unsafe.Pointer


c结构体和 go之间的转换为:


var obj *Lvobj
var p *C.struct__lv_obj_t
p = (*C.struct__lv_obj_t)(unsafe.Pointer(obj))


byte[]和 c的数组之间转换为:



         


如果是struct, union和enum的话,需要加上如下前缀,struct_、union_和enum_,比如 C.struct_MSG。


可以用注释符//和/**/包围C代码


import “C” 和包含C代码之间是没有空行的


动态库的导入和编译选项通过LDFLAGS、CFLAGS/CXXFLAGS来设置


还可以用pkg-config #cgo pkg-config : xxxxname


编译宏定义指定#cgo CFLAGS: -DNDEBUG -DXXXX=2


const (
   LV_ALIGN_CENTER C.uchar = C.LV_ALIGN_CENTER
   //  error:LV_PROTECT_POS C.enum_ = C.enum_LV_PROTECT_POS
)


对于c中的enum,没有enum{} name,名字的,它的类型是什么呢?经过摸索发现,其实就是个int或byte


ok: LV_PROTECT_POS byte = C.LV_PROTECT_POS
package main
// #include <stdio.h>
// #include <stdlib.h>
/*
void print(char *str) {
    printf("%s\n", str);
}
*/
import "C"
import "unsafe"
func main() {
    s := "Hello Cgo"
    cs := C.CString(s)
    C.print(cs)
    C.free(unsafe.Pointer(cs))
}


c结构体的位域和go之间转换为:



var img_bubble_pattern C.lv_img_dsc_t
var img_bubble_pattern_map = []byte{
  0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53,
  0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53,
  0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53, 0x53}
const (
  LV_ALIGN_CENTER C.uchar = C.LV_ALIGN_CENTER
)
func init() {
  fmt.Println(img_bubble_pattern)
  fmt.Println(reflect.TypeOf(img_bubble_pattern))
  //fmt.Println(reflect.TypeOf(img_bubble_pattern.header.always_zero))
  bt := [4]byte{0, 1, 2, 3}
  img_bubble_pattern = C.lv_img_dsc_t{
    header:    C.lv_img_header_t{bt},
    data_size: 112320 * C.LV_COLOR_SIZE / 8,
    data:      (*C.uchar)(unsafe.Pointer(&img_bubble_pattern_map[0])),
  }
}


未完待续,敬请期待....



想完善其他的封装太难了,几度想放弃。上面的c结构体的位域和go之间转换问题解决了,又遇到其他问题,如:


panic: runtime error: cgo argument has Go pointer to Go pointer


大意是 Go调用C Code时,Go传递给C Code的Go指针所指的Go Memory中不能包含任何指向Go Memory的Pointer。


比如以下样例代码举例:我本意想在go层读取图片二进制内容,传给给c层,调用c的显示图片的接口。


package main
/*
#include <stdio.h>
//模拟图片信息的定义
typedef struct {
int a;
int b;
unsigned char* data;
}st;
//模拟 c层的获取图片内容并显示图片的接口
void testc(void *p){
   char * p1;
   p1 = p;
   printf("a is:%d\n",p1[0]);
}
*/
import "C"
import (
  "fmt"
  "unsafe"
)
var img_bubble_pattern C.st
//模拟go层二进制图片数据
var dat = []byte{
  1, 2, 3,
}
//模拟封装的调用c层的图片显示接口
func test(p *C.st) {
  //fmt.Printf("%#v\n", p)
  C.testc(unsafe.Pointer(p))
}
func main() {
  fmt.Println("Hello, World!")
  fmt.Printf("%#v\n", img_bubble_pattern)
  //对结构体赋值
  img_bubble_pattern = C.st{
    a: 1, b: 2, data: (*C.uchar)(unsafe.Pointer(&dat[0])),
  }
  //调用
  test(&img_bubble_pattern)
}


var dat是从go应用层读取到的图片的二进制数据,调c的接口完成图片显示。c的显示图片的接口就是一个结构体指针。


但是如果调试这段代码就发现,报panic: runtime error: cgo argument has Go pointer to Go pointer,


这段代码有问题吗?肯定有问题的,但是实际的用途上没错,就想这么干,让cgo达到我这调用的目的,这问题该怎么破?


果然,还是有高人的,这个问题问到了go语言交流群里的群主苏州-狄,昵称曦晨。在这里表示感谢!


他的解决方案如下,把这个指针转换的问题换了一种方式,绕过去了,但也达到了目的。


package main
/*
#include <stdio.h>
//模拟图片信息的定义
typedef struct {
int a;
int b;
unsigned char* data;
}st;
//模拟 c层的获取图片内容并显示图片的接口
void testc(void *p){
  st* p1 = p;
   printf("a is:%d\n",p1->data[0]);
}
*/
import "C"
import (
  "fmt"
  "unsafe"
)
var img_bubble_pattern C.st
//模拟go层二进制图片数据
var dat = []byte{
  1, 2, 3,
}
//模拟封装的调用c层的图片显示接口
func test(p C.st) {
  //fmt.Printf("%#v\n", p)
  C.testc(unsafe.Pointer(&p))
}
func main() {
  fmt.Println("Hello, World!")
  fmt.Printf("%#v\n", img_bubble_pattern)
  //对结构体赋值
  img_bubble_pattern = C.st{
    a: 1, b: 2, data: (*C.uchar)(unsafe.Pointer(&dat[0])),
  }
  //调用
  // C.testc(unsafe.Pointer(&img_bubble_pattern))
  test(img_bubble_pattern)
}


对比两份代码发现,在go层不传指针了,改为结构体的值传递,解决了这个问题。


后续记,这个问题又有新解了。使用mattn大神封装的库,https://github.com/mattn/go-pointer


可以实现CGO和C之间的指针传递,且不再报错 cgo argument has Go pointer to Go pointer


如下:


C.testc(pointer.Save(p))


package main
/*
#include <stdio.h>
#include <stdlib.h>
//模拟图片信息的定义
typedef struct {
int a;
int b;
unsigned char* data;
}st;
//模拟 c层的获取图片内容并显示图片的接口
void testc(void *p){
  st* p1 = p;
   printf("a is:%d\n",p1->data[0]);
}
*/
import "C"
import (
  "fmt"
  "unsafe"
  "github.com/mattn/go-pointer"
)
var img_bubble_pattern C.st
//模拟go层二进制图片数据
var dat = []byte{
  1, 2, 3,
}
//模拟封装的调用c层的图片显示接口
// func test(p C.st) {
//  //fmt.Printf("%#v\n", p)
//  C.testc(unsafe.Pointer(&p))
// }
//模拟封装的调用c层的图片显示接口
func test(p *C.st) {
  //fmt.Printf("%#v\n", p)
  //C.pass_pointer(pointer.Save(p))
  //v := *(pointer.Restore(C.get_from_pointer()).(*C.st))
  C.testc(pointer.Save(p))
}
func main() {
  fmt.Println("Hello, World!")
  fmt.Printf("%#v\n", img_bubble_pattern)
  //对结构体赋值
  img_bubble_pattern = C.st{
    a: 1, b: 2, data: (*C.uchar)(unsafe.Pointer(&dat[0])),
  }
  //调用
  // C.testc(unsafe.Pointer(&img_bubble_pattern))
  test(&img_bubble_pattern)
}


https://mp.weixin.qq.com/s/WUdaksFyjmT_c_jEx59VcQ

相关文章
|
16天前
|
Go 数据安全/隐私保护 开发者
Go语言开发
【10月更文挑战第26天】Go语言开发
32 3
|
17天前
|
Java 程序员 Go
Go语言的开发
【10月更文挑战第25天】Go语言的开发
26 3
|
29天前
|
Linux API 开发工具
FFmpeg开发笔记(五十九)Linux编译ijkplayer的Android平台so库
ijkplayer是由B站研发的移动端播放器,基于FFmpeg 3.4,支持Android和iOS。其源码托管于GitHub,截至2024年9月15日,获得了3.24万星标和0.81万分支,尽管已停止更新6年。本文档介绍了如何在Linux环境下编译ijkplayer的so库,以便在较新的开发环境中使用。首先需安装编译工具并调整/tmp分区大小,接着下载并安装Android SDK和NDK,最后下载ijkplayer源码并编译。详细步骤包括环境准备、工具安装及库编译等。更多FFmpeg开发知识可参考相关书籍。
81 0
FFmpeg开发笔记(五十九)Linux编译ijkplayer的Android平台so库
|
3月前
|
JSON 中间件 Go
go语言后端开发学习(四) —— 在go项目中使用Zap日志库
本文详细介绍了如何在Go项目中集成并配置Zap日志库。首先通过`go get -u go.uber.org/zap`命令安装Zap,接着展示了`Logger`与`Sugared Logger`两种日志记录器的基本用法。随后深入探讨了Zap的高级配置,包括如何将日志输出至文件、调整时间格式、记录调用者信息以及日志分割等。最后,文章演示了如何在gin框架中集成Zap,通过自定义中间件实现了日志记录和异常恢复功能。通过这些步骤,读者可以掌握Zap在实际项目中的应用与定制方法
130 1
go语言后端开发学习(四) —— 在go项目中使用Zap日志库
|
2月前
|
存储 Linux 开发工具
如何进行Linux内核开发【ChatGPT】
如何进行Linux内核开发【ChatGPT】
|
3月前
|
Java Linux API
Linux设备驱动开发详解2
Linux设备驱动开发详解
43 6
|
3月前
|
消息中间件 算法 Unix
Linux设备驱动开发详解1
Linux设备驱动开发详解
49 5
|
3月前
|
自然语言处理 前端开发 Linux
在Linux中,什么是 GUI?
在Linux中,什么是 GUI?
|
3月前
|
编解码 安全 Linux
基于arm64架构国产操作系统|Linux下的RTMP|RTSP低延时直播播放器开发探究
这段内容讲述了国产操作系统背景下,大牛直播SDK针对国产操作系统与Linux平台发布的RTMP/RTSP直播播放SDK。此SDK支持arm64架构,基于X协议输出视频,采用PulseAudio和Alsa Lib处理音频,具备实时静音、快照、缓冲时间设定等功能,并支持H.265编码格式。此外,提供了示例代码展示如何实现多实例播放器的创建与管理,包括窗口布局调整、事件监听、视频分辨率变化和实时快照回调等关键功能。这一技术实现有助于提高直播服务的稳定性和响应速度,适应国产操作系统在各行业中的应用需求。
109 3
|
3月前
|
算法 NoSQL 中间件
go语言后端开发学习(六) ——基于雪花算法生成用户ID
本文介绍了分布式ID生成中的Snowflake(雪花)算法。为解决用户ID安全性与唯一性问题,Snowflake算法生成的ID具备全局唯一性、递增性、高可用性和高性能性等特点。64位ID由符号位(固定为0)、41位时间戳、10位标识位(含数据中心与机器ID)及12位序列号组成。面对ID重复风险,可通过预分配、动态或统一分配标识位解决。Go语言实现示例展示了如何使用第三方包`sonyflake`生成ID,确保不同节点产生的ID始终唯一。
go语言后端开发学习(六) ——基于雪花算法生成用户ID