理清了思路以后开始通过实例测试GF的使用过程,为了简化测试环境,这次使用的是QNX Momentics环境和QNX虚拟机环境,没有使用BeagleBoard作为运行目标,主要是怕直接在板子上运行会有其他因素影响测试。
在QNX momentics环境中创建了一个新的QNX C Project, 在main.c里先加上gf头文件的引用:
#include <gf/gf.h>
然后在main方法里加上第一步操作,就是通过fg_dev_attach()方法连接到设备上去,代码如下:
if (gf_dev_attach(&gdev, GF_DEVICE_INDEX(0), &gdev_info) != GF_ERR_OK) {
printf("gf_dev_attach() failed\n");
return (-1);
}
不管后面是否成功,先测试一下能否连接设备,很气馁地发现编译都不通过,方法gf_dev_attach的引用有问题。
不是已经将fg.h引入了吗?
在网上找了一轮,发现是链接配置有问题,需要在链接选项里加上gf库的引用,配置在项目属性的“QNX c/c++”一栏,在“Linker”页面的“Other Options”项中填写“-lgf”,如下图:
然后的测试就一路顺风顺水:
连接设备成功!
连接显示器成功!
连接显示层成功!
具体的代码在文章后面有,这里就不贴了。
到了显示表面的地方卡了,有点问题,首先是使用那个方法有点搞不清,看了文档才发现应该是调用
gf_surface_create_layer()方法创建显示表面,然后才调用gf_layer_set_surfaces()将显示表面和显示层连接起来。
然后就是一直无法创建显示表面,测试的时候总是报错。
最后从openqnx网站上找到答案,去年的贴子,和我的问题一模一样,贴子链接如下:
http://www.openqnx.com/chinese/viewtopic.php?t=2656
贴子中提到问题的原因是QNX虚拟机启动Photon界面后将资源占住了,我们的测试程序无法使用。
解决方法是从新启动QNX虚拟机,在登录界面选择“ShutDown”按钮,然后在关机选择中选择“Exit to text mode”如下图,就是退出到文本界面的意思:
然后要注意,退出到文本界面后login字样直接出现在登录背景图上,登录背景图没有清空,这时不要以为有什么问题,直接输入用户名,回车,密码,回车,就可以进入文本界面了。
进入文本界面后回到QNX Momentics上测试应用,果然可以解决问题,成功创建显示表面。
后面就真的是一路顺风了,最后很没追求地在QNX上显示了一个无聊的方框:
下面是完整代码,供参考:
#include <stdlib.h>
#include <stdio.h>
#include <gf/gf.h>
int main(int argc, char *argv[]) {
printf("Starting Graphics test\n");
gf_dev_t gdev;
gf_dev_info_t gdev_info;
gf_display_t display;
gf_display_info_t display_info;
gf_layer_t layer;
gf_surface_t surface;
gf_context_t context;
if (gf_dev_attach(&gdev, GF_DEVICE_INDEX(0), &gdev_info) != GF_ERR_OK) {
printf("gf_dev_attach() failed\n");
return (-1);
}
printf("device attached\n");
if (gf_display_attach(&display, gdev, 0, &display_info) != GF_ERR_OK) {
printf("gf_display_attach() failed\n");
}
printf("display attached\n");
if (display != NULL) {
unsigned main_layer_index = display_info.main_layer_index;
if (gf_layer_attach(&layer, display, main_layer_index,
GF_LAYER_ATTACH_PASSIVE) != GF_ERR_OK) {
printf("gf_layer_attach() failed\n");
return EXIT_SUCCESS;
}
} else {
printf("Display is null \n");
return EXIT_SUCCESS;
}
printf("Layer attached \n");
//gf_surface_t surface;
int width = display_info.xres;
int height = display_info.yres;
int create_surface_result = gf_surface_create_layer(&surface, &layer, 1, 0,
width, height, display_info.format, NULL, 0);
if (create_surface_result == GF_ERR_OK) {
gf_layer_set_surfaces(layer, &surface, 1);
printf("attach to the surface , Done\n");
} else {
printf("attach to the surface , failed. Error: %d\n",
create_surface_result);
return EXIT_SUCCESS;
}
if (gf_context_create(&context) != GF_ERR_OK) {
fprintf(stderr, "gf_context_create failed\n");
}
if (gf_context_set_surface(context, surface) != GF_ERR_OK) {
fprintf(stderr, "gf_context_set_surface failed\n");
}
gf_draw_begin(context);
gf_context_set_fgcolor(context, 0xFFFFFF);
gf_draw_rect(context, 10, 10, 50, 50);
gf_context_set_fgcolor(context, 0x000000);
gf_draw_rect(context, 20, 20, 40, 40);
gf_draw_flush(context);
gf_draw_end(context);
gf_dev_detach(gdev);
return EXIT_SUCCESS;
}