0


粤嵌Linux GEC6818开发板实现电子相册

前言

最近学校要求使用粤嵌的开发板实现电子相册,具体的功能要有点击特定的区域实现上一张、下一张、自动播放图片、黑屏退出应用程序、左右滑动切换图片相关功能。其中涉及到的知识点也比较多(文件IO、内存映射、触摸屏、bmp图片格式、进程、线程创建和同步、字符串操作等)。为理清思路和复习去年学的Linux C应用编程知识,特写下此文进行回顾和总结。

先看看效果

粤嵌Linux GEC6818开发板实现电子相册

整个工程文件和使用到的图片在下方链接

门牙会稍息 / 粤嵌GEC 6818开发板实现简易电子相册和音乐播放器 · GitCode

一:内存映射

存储映射 I/O(memory-mapped I/O)是一种基于内存区域的高级 I/O 操作,它能将一个文件映射到进程地址空间中的一块内存区域中,当从这段内存中读数据时,就相当于读文件中的数据(对文件进行 read 操作),将数据写入这段内存时,则相当于将数据直接写入文件中(对文件进行 write 操作)。用到的两个函数是mmap和munmap,函数原型如下:

简言之addr设置为NULL的话内核会自动找一内存空间,大小就是length,粤嵌的屏是800480的,所以length就是800480*4,4代表一个像素点由四字节构成(ARGB),port参数设为PROT_READ、PROT_WRITE就是可读可写,flags描述的是映射区的属性。

二:BMP格式图片

用hexdump查看一下bmp图片的数据(高度和宽度)

LCD屏显示BMP格式图片的函数

  1. /**
  2. *lcd屏显示bmp格式图片函数
  3. *@param pathname 图片名字
  4. *@param x0 图片在LCD上显示的x起点坐标
  5. *@param y0 图片在LCD上显示的y起点坐标
  6. *@return 函数返回值
  7. */
  8. int lcd_show_bmp(char *pathname, int x0, int y0)
  9. {
  10. //1、打开设备文件
  11. int lcd = open("/dev/fb0", O_RDWR);
  12. if(-1 == lcd)
  13. {
  14. printf("lcd open error!\n");
  15. return -2;
  16. }
  17. //2、内存映射
  18. char *ptr = (char *)mmap(NULL, 800*480*4, PROT_READ|PROT_WRITE,
  19. MAP_SHARED, lcd, 0);
  20. if(NULL == ptr)
  21. {
  22. printf("mmap error!\n");
  23. return -3;
  24. }
  25. //清除背景色
  26. //bzero(ptr, 800*480*4);
  27. //打开图片
  28. int bmp = open(pathname, O_RDWR);
  29. if(-1 == bmp)
  30. {
  31. printf("bmp open error!\n");
  32. return -2;
  33. }
  34. //读取图片相关信息
  35. int bmp_size, bmp_width, bmp_height;
  36. //大小
  37. lseek(bmp, 2, SEEK_SET);
  38. read(bmp, &bmp_size, sizeof(bmp_size));
  39. //宽度和高度
  40. lseek(bmp, 18, SEEK_SET);
  41. read(bmp, &bmp_width, sizeof(bmp_width));
  42. read(bmp, &bmp_height, sizeof(bmp_height));
  43. //读取图片的颜色数据
  44. lseek(bmp, 54, SEEK_SET);
  45. char *bmp_buf = malloc(bmp_size); //申请空间
  46. read(bmp, bmp_buf, bmp_width*bmp_height*3);
  47. //对颜色数据进行处理(上下颠倒以及数据混乱)
  48. int bmp_sum = 0;
  49. int lcd_sum = 0;
  50. for(int y=0; y+y0<480 && y<bmp_height; y++)
  51. {
  52. for(int x=0; x+x0<800 && x<bmp_width; x++)
  53. {
  54. bmp_sum = 3*(x+((bmp_height-1-y)*bmp_width));
  55. lcd_sum = 4*(800*(y+y0)+x+x0);
  56. //等号的坐标属于lcd屏幕, 等号的右边是bmp图像数据
  57. ptr[lcd_sum+0] = bmp_buf[bmp_sum+0]; //蓝色数据
  58. ptr[lcd_sum+1] = bmp_buf[bmp_sum+1]; //绿色数据
  59. ptr[lcd_sum+2] = bmp_buf[bmp_sum+2]; //红色数据
  60. ptr[lcd_sum+3] = bmp_buf[bmp_sum+3]; //透明度数据
  61. }
  62. //usleep(1);
  63. }
  64. //释放相关资源
  65. munmap(ptr, 800*480*4);
  66. free(bmp_buf);
  67. //3、关闭文件
  68. close(lcd);
  69. close(bmp);
  70. }

三:input_event接收触摸屏上报值

用hexdump命令查看触摸屏上报值

此实验用到input子系统中的type、code(下面的图片来源Linux内核中的input.h文件)

获取触摸屏坐标返回值函数

  1. /**
  2. *获取触摸屏坐标返回值函数
  3. *param NULL
  4. *return NULL
  5. */
  6. void get_touch()
  7. {
  8. //打开设备文件
  9. int touch_fd = open("/dev/input/event0", O_RDONLY);
  10. if(-1 == touch_fd)
  11. {
  12. printf("event0 open error!\n");
  13. }
  14. while(1)
  15. {
  16. read(touch_fd, &ts, sizeof(ts));
  17. //获取X、Y坐标
  18. if(EV_ABS == ts.type) //判断是否为触摸屏事件
  19. {
  20. if(ABS_X == ts.code)//判断是否为x轴数据
  21. {
  22. ts_x = ts.value;
  23. flag_x = 1;
  24. }
  25. else if(ABS_Y == ts.code) //判断是否为y轴数据
  26. {
  27. ts_y = ts.value;
  28. flag_y = 1;
  29. }
  30. }
  31. if(EV_KEY == ts.type)
  32. {
  33. //刚触碰的坐标/长按时
  34. if(ts.code == BTN_TOUCH && ts.value == 1)
  35. {
  36. old_x = ts_x;
  37. first_press_flag = 1;
  38. }
  39. }
  40. if(flag_x == 1 && flag_y == 1 && first_press_flag == 1)
  41. {
  42. flag_x = 0;
  43. flag_y = 0;
  44. pthread_mutex_lock(&mutex);
  45. flag_x_y = 1;
  46. //黑色底板才需要执行如下操作
  47. ts_x = ts_x*800/1024;
  48. ts_y = ts_y*480/600;
  49. pthread_mutex_unlock(&mutex);
  50. pthread_cond_signal(&cond);
  51. //break;
  52. }
  53. if(EV_KEY == ts.type)
  54. {
  55. //松开
  56. if(ts.code == BTN_TOUCH && ts.value == 0)
  57. {
  58. //从左到右的滑动
  59. if(((ts_x > old_x) && (ts_x - old_x > 150) && (ts_x < 600)))
  60. {
  61. right_left_slide_flag = 1;
  62. slider_right = 1;
  63. old_x = 300;
  64. }
  65. //从右到左的滑动
  66. else if(((old_x > ts_x) && (old_x - ts_x > 150) && (old_x < 600)))
  67. {
  68. right_left_slide_flag = 1;
  69. slider_left = 1;
  70. old_x = 300;
  71. }
  72. }
  73. }
  74. }
  75. //4、关闭文件
  76. close(touch_fd);
  77. }

四:创建线程,处理触摸屏坐标数据

阻塞式 I/O 的优点在于能够提升 CPU 的处理效率,当自身条件不满足时,进入阻塞状态,交出 CPU资源,将 CPU 资源让给别人使用;而非阻塞式则是抓紧利用 CPU 资源,譬如不断地去轮训,这样就会导致该程序占用了非常高的 CPU 使用率!我这里是想获得触摸点坐标之后再做相关的操作,当没有按下触摸屏的时候,相关线程就会阻塞挂起,节约资源,线程同步中使用互斥锁和条件变量就可以实现。

左右滑动线程处理函数:

  1. void *right_left_slide_func(void *arg)
  2. {
  3. printf("enter right_left_slide_func\r\n");
  4. while(1){
  5. if(right_left_slide_flag == 1){
  6. if(slider_left == 1){
  7. touch_flag--;
  8. if(touch_flag <= 0)
  9. touch_flag = BMP_MAX_NUMBER;
  10. lcd_show_bmp(bmp_path[touch_flag - 1], 0, 0);
  11. slider_left = 0;
  12. right_left_slide_flag == 0;
  13. }
  14. else if(slider_right == 1){
  15. touch_flag++;
  16. if(touch_flag > BMP_MAX_NUMBER)
  17. touch_flag = 1;
  18. lcd_show_bmp(bmp_path[touch_flag - 1], 0, 0);
  19. slider_right = 0;
  20. right_left_slide_flag == 0;
  21. }
  22. right_left_slide_flag == 0;
  23. }
  24. }
  25. }

右侧选项框线程处理函数:

  1. void *area_switch(void *arg)
  2. {
  3. printf("enter area_switch\r\n");
  4. while(1){
  5. pthread_mutex_lock(&mutex);
  6. while(flag_x_y == 0)
  7. pthread_cond_wait(&cond, &mutex);
  8. while(flag_x_y == 1){
  9. //上一张
  10. if((ts_y > 0 && ts_y < 120) && (ts_x > 600))
  11. {
  12. touch_flag--;
  13. if(touch_flag <= 0)
  14. touch_flag = BMP_MAX_NUMBER;
  15. lcd_show_bmp(bmp_path[touch_flag - 1], 0, 0);
  16. ts_x = 0;
  17. ts_y = 0;
  18. }
  19. //下一张
  20. else if((ts_y > 120 && ts_y < 240) && (ts_x > 600))
  21. {
  22. touch_flag++;
  23. if(touch_flag > BMP_MAX_NUMBER)
  24. touch_flag = 1;
  25. lcd_show_bmp(bmp_path[touch_flag - 1], 0, 0);
  26. ts_x = 0;
  27. ts_y = 0;
  28. }
  29. //幻灯片
  30. else if((ts_y > 240 && ts_y < 360) && (ts_x > 600))
  31. {
  32. printf("click slider photo\r\n");
  33. if(slider_flag == 0){
  34. slider_flag = 1;
  35. }
  36. else{
  37. slider_flag = 0;
  38. }
  39. printf("slider_flag = %d\r\n", slider_flag);
  40. ts_x = 0;
  41. ts_y = 0;
  42. }
  43. //息屏
  44. else if((ts_y > 360 && ts_y < 480) && (ts_x > 600))
  45. {
  46. char command[] = "kill -9 ";
  47. char str[10];
  48. sprintf(str, "%d", pid);
  49. strcat(command, str);
  50. printf("command = %s\r\n", command);
  51. //方式一:显示一张黑色的图片
  52. lcd_show_bmp("black.bmp", 0, 0);
  53. system(command);
  54. }
  55. else{
  56. flag_x_y = 0;
  57. break;
  58. }
  59. flag_x_y = 0;
  60. }
  61. pthread_mutex_unlock(&mutex);
  62. }
  63. }

自动播放图片线程处理函数:

  1. void *slider_func(void *arg)
  2. {
  3. printf("enter slider_func : %d\r\n", pthread_self());
  4. while(1){
  5. if(slider_flag == 1){
  6. printf("enter slider_func\r\n");
  7. touch_flag++;
  8. if(touch_flag > BMP_MAX_NUMBER)
  9. touch_flag = 1;
  10. lcd_show_bmp(bmp_path[touch_flag - 1], 0, 0);
  11. sleep(1);
  12. }
  13. }
  14. }

五:Main函数

main函数就是一些线程、互斥锁、条件变量的创建和回收

  1. int main()
  2. {
  3. lcd_show_bmp("choice.bmp", 600, 0);
  4. lcd_show_bmp(bmp_path[touch_flag - 1], 0, 0);
  5. pthread_t tid, tid_area_switch, tid_right_left_slide;
  6. int ret = 0;
  7. pthread_create(&tid, NULL, slider_func, NULL);
  8. pthread_create(&tid_area_switch, NULL, area_switch, NULL);
  9. pthread_create(&tid_right_left_slide, NULL, right_left_slide_func, NULL);
  10. pthread_mutex_init(&mutex, NULL);
  11. pthread_cond_init(&cond, NULL);
  12. printf("main thread = %ld\r\n", pthread_self());
  13. pid = getpid();
  14. printf("pid = %d\r\n", pid);
  15. while(1){
  16. get_touch();
  17. }
  18. pthread_join(tid, NULL);
  19. pthread_join(tid_area_switch, NULL);
  20. pthread_join(tid_right_left_slide, NULL);
  21. pthread_cond_destroy(&cond);
  22. exit(0);
  23. }

六:完整代码

  1. #include <stdio.h>
  2. #include <fcntl.h>
  3. #include <sys/stat.h>
  4. #include <sys/types.h>
  5. #include <unistd.h>
  6. #include <string.h>
  7. #include <sys/mman.h>
  8. #include <stdlib.h>
  9. #include <linux/input.h>
  10. #include <pthread.h>
  11. #include <unistd.h>
  12. #define BMP_MAX_NUMBER 4
  13. char bmp_path[4][100] = {"1.1.bmp", "1.2.bmp", "1.3.bmp", "1.4.bmp"};
  14. int slider_flag = 0;
  15. pid_t pid;
  16. pthread_mutex_t mutex;
  17. pthread_cond_t cond;
  18. //自定义函数:LCD屏幕显示bmp图片
  19. //pathname:需要打开的图片路径
  20. //x0:存放图片显示的x轴起点
  21. //y0:存放图片显示的y轴起点
  22. int lcd_show_bmp(char *pathname, int x0, int y0)
  23. {
  24. //1、打开设备文件
  25. int lcd = open("/dev/fb0", O_RDWR);
  26. if(-1 == lcd)
  27. {
  28. printf("lcd open error!\n");
  29. return -2;
  30. }
  31. //2、内存映射
  32. char *ptr = (char *)mmap(NULL, 800*480*4, PROT_READ|PROT_WRITE,
  33. MAP_SHARED, lcd, 0);
  34. if(NULL == ptr)
  35. {
  36. printf("mmap error!\n");
  37. return -3;
  38. }
  39. //清除背景色
  40. //bzero(ptr, 800*480*4);
  41. //打开图片
  42. int bmp = open(pathname, O_RDWR);
  43. if(-1 == bmp)
  44. {
  45. printf("bmp open error!\n");
  46. return -2;
  47. }
  48. //读取图片相关信息
  49. int bmp_size, bmp_width, bmp_height;
  50. //大小
  51. lseek(bmp, 2, SEEK_SET);
  52. read(bmp, &bmp_size, sizeof(bmp_size));
  53. //宽度和高度
  54. lseek(bmp, 18, SEEK_SET);
  55. read(bmp, &bmp_width, sizeof(bmp_width));
  56. read(bmp, &bmp_height, sizeof(bmp_height));
  57. //读取图片的颜色数据
  58. lseek(bmp, 54, SEEK_SET);
  59. char *bmp_buf = malloc(bmp_size); //申请空间
  60. read(bmp, bmp_buf, bmp_width*bmp_height*3);
  61. //对颜色数据进行处理(上下颠倒以及数据混乱)
  62. int bmp_sum = 0;
  63. int lcd_sum = 0;
  64. for(int y=0; y+y0<480 && y<bmp_height; y++)
  65. {
  66. for(int x=0; x+x0<800 && x<bmp_width; x++)
  67. {
  68. bmp_sum = 3*(x+((bmp_height-1-y)*bmp_width));
  69. lcd_sum = 4*(800*(y+y0)+x+x0);
  70. //等号的坐标属于lcd屏幕, 等号的右边是bmp图像数据
  71. ptr[lcd_sum+0] = bmp_buf[bmp_sum+0]; //蓝色数据
  72. ptr[lcd_sum+1] = bmp_buf[bmp_sum+1]; //绿色数据
  73. ptr[lcd_sum+2] = bmp_buf[bmp_sum+2]; //红色数据
  74. ptr[lcd_sum+3] = bmp_buf[bmp_sum+3]; //透明度数据
  75. }
  76. //usleep(1);
  77. }
  78. munmap(ptr, 800*480*4);
  79. free(bmp_buf);
  80. //3、关闭文件
  81. close(lcd);
  82. close(bmp);
  83. }
  84. int touch_flag = 1;
  85. int ts_x, ts_y;
  86. int old_x = 300;
  87. int slider_left = 0;
  88. int slider_right = 0;
  89. int flag_x_y = 0;
  90. int right_left_slide_flag = 0;
  91. struct input_event ts;
  92. int flag_x = 0, flag_y = 0, first_press_flag = 0;
  93. //获取触摸屏坐标
  94. //自定义函数:获取触摸屏的坐标
  95. void get_touch()
  96. {
  97. //1、打开设备文件
  98. int touch_fd = open("/dev/input/event0", O_RDONLY);
  99. if(-1 == touch_fd)
  100. {
  101. printf("event0 open error!\n");
  102. }
  103. while(1)
  104. {
  105. read(touch_fd, &ts, sizeof(ts));
  106. //3、分析数据
  107. if(EV_ABS == ts.type) //判断是否为触摸屏事件
  108. {
  109. if(ABS_X == ts.code)//判断是否为x轴数据
  110. {
  111. ts_x = ts.value;
  112. flag_x = 1;
  113. }
  114. else if(ABS_Y == ts.code) //判断是否为y轴数据
  115. {
  116. ts_y = ts.value;
  117. flag_y = 1;
  118. }
  119. }
  120. if(EV_KEY == ts.type)
  121. {
  122. //刚触碰的坐标/长按时
  123. if(ts.code == BTN_TOUCH && ts.value == 1)
  124. {
  125. old_x = ts_x;
  126. first_press_flag = 1;
  127. }
  128. }
  129. if(flag_x == 1 && flag_y == 1 && first_press_flag == 1)
  130. {
  131. flag_x = 0;
  132. flag_y = 0;
  133. pthread_mutex_lock(&mutex);
  134. flag_x_y = 1;
  135. //黑色底板才需要执行如下操作
  136. ts_x = ts_x*800/1024;
  137. ts_y = ts_y*480/600;
  138. pthread_mutex_unlock(&mutex);
  139. pthread_cond_signal(&cond);
  140. //break;
  141. }
  142. if(EV_KEY == ts.type)
  143. {
  144. //松开
  145. if(ts.code == BTN_TOUCH && ts.value == 0)
  146. {
  147. //从左到右的滑动
  148. if(((ts_x > old_x) && (ts_x - old_x > 150) && (ts_x < 600)))
  149. {
  150. right_left_slide_flag = 1;
  151. slider_right = 1;
  152. old_x = 300;
  153. }
  154. //从右到左的滑动
  155. else if(((old_x > ts_x) && (old_x - ts_x > 150) && (old_x < 600)))
  156. {
  157. right_left_slide_flag = 1;
  158. slider_left = 1;
  159. old_x = 300;
  160. }
  161. }
  162. }
  163. }
  164. //4、关闭文件
  165. close(touch_fd);
  166. }
  167. void *right_left_slide_func(void *arg)
  168. {
  169. printf("enter right_left_slide_func\r\n");
  170. while(1){
  171. if(right_left_slide_flag == 1){
  172. if(slider_left == 1){
  173. touch_flag--;
  174. if(touch_flag <= 0)
  175. touch_flag = BMP_MAX_NUMBER;
  176. lcd_show_bmp(bmp_path[touch_flag - 1], 0, 0);
  177. slider_left = 0;
  178. right_left_slide_flag == 0;
  179. }
  180. else if(slider_right == 1){
  181. touch_flag++;
  182. if(touch_flag > BMP_MAX_NUMBER)
  183. touch_flag = 1;
  184. lcd_show_bmp(bmp_path[touch_flag - 1], 0, 0);
  185. slider_right = 0;
  186. right_left_slide_flag == 0;
  187. }
  188. right_left_slide_flag == 0;
  189. }
  190. }
  191. }
  192. void *area_switch(void *arg)
  193. {
  194. printf("enter area_switch\r\n");
  195. while(1){
  196. pthread_mutex_lock(&mutex);
  197. while(flag_x_y == 0)
  198. pthread_cond_wait(&cond, &mutex);
  199. while(flag_x_y == 1){
  200. //上一张
  201. if((ts_y > 0 && ts_y < 120) && (ts_x > 600))
  202. {
  203. touch_flag--;
  204. if(touch_flag <= 0)
  205. touch_flag = BMP_MAX_NUMBER;
  206. lcd_show_bmp(bmp_path[touch_flag - 1], 0, 0);
  207. ts_x = 0;
  208. ts_y = 0;
  209. }
  210. //下一张
  211. else if((ts_y > 120 && ts_y < 240) && (ts_x > 600))
  212. {
  213. touch_flag++;
  214. if(touch_flag > BMP_MAX_NUMBER)
  215. touch_flag = 1;
  216. lcd_show_bmp(bmp_path[touch_flag - 1], 0, 0);
  217. ts_x = 0;
  218. ts_y = 0;
  219. }
  220. //幻灯片
  221. else if((ts_y > 240 && ts_y < 360) && (ts_x > 600))
  222. {
  223. printf("click slider photo\r\n");
  224. if(slider_flag == 0){
  225. slider_flag = 1;
  226. }
  227. else{
  228. slider_flag = 0;
  229. }
  230. printf("slider_flag = %d\r\n", slider_flag);
  231. ts_x = 0;
  232. ts_y = 0;
  233. }
  234. //息屏
  235. else if((ts_y > 360 && ts_y < 480) && (ts_x > 600))
  236. {
  237. char command[] = "kill -9 ";
  238. char str[10];
  239. sprintf(str, "%d", pid);
  240. strcat(command, str);
  241. printf("command = %s\r\n", command);
  242. //方式一:显示一张黑色的图片
  243. lcd_show_bmp("black.bmp", 0, 0);
  244. system(command);
  245. }
  246. else{
  247. flag_x_y = 0;
  248. break;
  249. }
  250. flag_x_y = 0;
  251. }
  252. pthread_mutex_unlock(&mutex);
  253. }
  254. }
  255. void *slider_func(void *arg)
  256. {
  257. printf("enter slider_func : %d\r\n", pthread_self());
  258. while(1){
  259. if(slider_flag == 1){
  260. printf("enter slider_func\r\n");
  261. touch_flag++;
  262. if(touch_flag > BMP_MAX_NUMBER)
  263. touch_flag = 1;
  264. lcd_show_bmp(bmp_path[touch_flag - 1], 0, 0);
  265. sleep(1);
  266. }
  267. }
  268. }
  269. int main()
  270. {
  271. lcd_show_bmp("choice.bmp", 600, 0);
  272. lcd_show_bmp(bmp_path[touch_flag - 1], 0, 0);
  273. pthread_t tid, tid_area_switch, tid_right_left_slide;
  274. int ret = 0;
  275. pthread_create(&tid, NULL, slider_func, NULL);
  276. pthread_create(&tid_area_switch, NULL, area_switch, NULL);
  277. pthread_create(&tid_right_left_slide, NULL, right_left_slide_func, NULL);
  278. pthread_mutex_init(&mutex, NULL);
  279. pthread_cond_init(&cond, NULL);
  280. printf("main thread = %ld\r\n", pthread_self());
  281. pid = getpid();
  282. printf("pid = %d\r\n", pid);
  283. while(1){
  284. get_touch();
  285. }
  286. pthread_join(tid, NULL);
  287. pthread_join(tid_area_switch, NULL);
  288. pthread_join(tid_right_left_slide, NULL);
  289. pthread_cond_destroy(&cond);
  290. exit(0);
  291. }

总结

以上就是本文的全部内容, 希望能够帮助到你。

标签: linux 学习 arm开发

本文转载自: https://blog.csdn.net/weixin_56646002/article/details/130232201
版权归原作者 门牙会稍息 所有, 如有侵权,请联系我们删除。

“粤嵌Linux GEC6818开发板实现电子相册”的评论:

还没有评论