[TOC]
一、Window模块
要使用ZBar进行条码识别的时候,输入的可以是视频流也可以是图像流。
在采用视频流进行输入的情况下,我们往往采取的方式是打开摄像头窗口进行扫码识别。在摄像头捕获到的视频信息中,ZBar需要对视频信息进行一系列的采集和处理,如逐帧捕获等等。Window模块会将处理后的信息交给其他模块进行解码等处理。
采用图像输入时,也需要Window模块的参与,这会在后续代码分析中提到。
这部分功能的实现并不是由Video模块实现,而是在ZBar打开窗口时,由Window模块进行实现的。这样的模块分离,使得整个项目的结构更加清晰,各模块之间的分工更加明确。
除此之外,根据ZBar的项目流程,可以看到,Window模块还有一个核心功能为将图像显示到用户指定的特定于平台的输出窗口。
ZBar项目的核心代码大部分都在zbar文件夹下,而zbar文件夹下的window.h和window.c则负责了这一模块的核心功能实现,而具体的调用实现则是在Processor处理模块以及其他api中完成。
本次代码分析将从window.h和window.c中展开。
二、代码分析
window数据结构解析
根据代码中的注释,给出数据结构中每个属性的补充说明。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71
| `
1. struct zbar_window_s { 2. errinfo_t err; /* 报错信息 */ 3. zbar_image_t *image; /* 上一张显示的图像 */ 4. /* 该图像信息的访问权限必须上锁 */
6. unsigned overlay; /* 用户设定的覆盖等级 */
8. uint32_t format; /* 输出格式 */ 9. unsigned width, height; /* 当前输出尺寸 */ 10. unsigned max_width, max_height;
12. uint32_t src_format; /* 当前输入格式 */ 13. unsigned src_width; /* 上一张显示图像的尺寸 */ 14. unsigned src_height;
16. unsigned dst_width; /* 转换目标的图像尺寸 */ 17. unsigned dst_height;
19. unsigned scale_num; /* 输出缩放尺度 */ 20. unsigned scale_den;
22. point_t scaled_offset; /* 输出位置和大小 */ 23. point_t scaled_size;
25. uint32_t *formats; /* 支持的格式(必须以0结尾) */
27. zbar_mutex_t imglock; /* 锁定当前显示的图像 */
29. void *display; 30. unsigned long xwin; 31. unsigned long time; /* 以毫秒为单位显示图像 */ 32. unsigned long time_avg; /* 帧间时间的平均值 */
34. window_state_t *state; /* 接口特定状态 */
36. /* 依赖接口的方法 */ 37. int (*init)(zbar_window_t*, zbar_image_t*, int); 38. int (*draw_image)(zbar_window_t*, zbar_image_t*); 39. int (*cleanup)(zbar_window_t*); 40. };
|
Window线程的上锁与解锁
在代码注释中提到,在渲染窗口时,必须保证渲染函数window.draw是线程安全的。ZBar对于线程安全方面的处理都是采用互斥锁和信号量实现。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53
| `
1. static inline int window_lock (zbar_window_t *w) 2. { 3. int rc = 0; 4. if((rc = _zbar_mutex_lock(&w->imglock))) { 5. err_capture(w, SEV_FATAL, ZBAR_ERR_LOCKING, __func__, 6. "unable to acquire lock"); 7. w->err.errnum = rc; 8. return(-1); 9. } 10. return(0); 11. }
13. static inline int window_unlock (zbar_window_t *w) 14. { 15. int rc = 0; 16. if((rc = _zbar_mutex_unlock(&w->imglock))) { 17. err_capture(w, SEV_FATAL, ZBAR_ERR_LOCKING, __func__, 18. "unable to release lock"); 19. w->err.errnum = rc; 20. return(-1); 21. } 22. return(0); 23. }
`

|
当有渲染窗口的线程获得了互斥锁,其他线程则必须等待,直到其释放锁。
而关于互斥锁的具体实现,则在mutex.c中体现,后续有机会再进行解析。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45
| `
1. zbar_window_t *zbar_window_create () 2. { 3. zbar_window_t *w = calloc(1, sizeof(zbar_window_t)); 4. if(!w) 5. return(NULL); 6. err_init(&w->err, ZBAR_MOD_WINDOW); 7. w->overlay = 1; 8. (void)_zbar_mutex_init(&w->imglock); 9. return(w); 10. }
12. void zbar_window_destroy (zbar_window_t *w) 13. { 14. /* detach */ 15. zbar_window_attach(w, NULL, 0); 16. err_cleanup(&w->err); 17. _zbar_mutex_destroy(&w->imglock); 18. free(w); 19. }
`

|
ZBar在创建一个窗口和销毁一个窗口时,同时新建和释放互斥锁。
在创建窗口时,首先为窗口申请内存,并为当前窗口新建互斥锁,当要渲染形成窗口界面时,则通过互斥锁实现线程互斥。
窗口固定
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54
| `
1. int zbar_window_attach (zbar_window_t *w, 2. void *display, 3. unsigned long drawable) 4. { 5. /* release image */ 6. zbar_window_draw(w, NULL); 7. if(w->cleanup) { 8. w->cleanup(w); 9. w->cleanup = NULL; 10. w->draw_image = NULL; 11. } 12. if(w->formats) { 13. free(w->formats); 14. w->formats = NULL; 15. } 16. w->src_format = 0; 17. w->src_width = w->src_height = 0; 18. w->scaled_size.x = w->scaled_size.y = 0; 19. w->dst_width = w->dst_height = 0; 20. w->max_width = w->max_height = 1 << 15; 21. w->scale_num = w->scale_den = 1; 22. return(_zbar_window_attach(w, display, drawable)); 23. }
`

|
该函数实现了窗口在平台上的关联和固定。
**这里对外界平台做一下说明:平台可以是 *X Windows *的任何“可绘制”窗口 或 Windows 的“HWND”。
在形式参数中传递 NULL 空值 指定从资源中分离,将显示进一步的输入。
ZBar对窗口销毁时,除了释放了互斥锁,同时还需要对窗口中的当前图像进行释放,并对属性进行重置,并设定为不可渲染。
窗口渲染
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36
| `1. int zbar_window_draw (zbar_window_t *w, 2. zbar_image_t *img) 3. { 4. if(window_lock(w)) 5. return(-1); 6. if(!w->draw_image) 7. img = NULL; 8. if(img) { 9. _zbar_image_refcnt(img, 1); 10. if(img->width != w->src_width || 11. img->height != w->src_height) 12. w->dst_width = 0; 13. } 14. if(w->image) 15. _zbar_image_refcnt(w->image, -1); 16. w->image = img; 17. return(window_unlock(w)); 18. }
|
该函数实现了将图像输出到特定窗口。
可以看到,如果执行该函数时,互斥锁已被占用,则不允许执行后续程序。
在将图像输出时,如果出现图像尺寸与窗口尺寸不一致的情况,ZBar会将输出的目标图像的宽置为0,从而实现强制适应。
三、总结
本次代码分析对Window模块的几个函数进行了简单说明,关于窗口重渲染以及图层的覆盖处理将在下次分析中给出。