From ff8ad6b4970c0ce68d1da1cc683e73b2ffdbefe3 Mon Sep 17 00:00:00 2001 From: Luigi Rizzo <rizzo@icir.org> Date: Mon, 30 Jun 2008 15:45:15 +0000 Subject: [PATCH] implement the 'freeze' function for incoming frames; fix a bug which caused a crash when a videodevice was specified after startgui=1 in the config file. This also involves a slightly different method to determine if the gui is active or not. git-svn-id: https://origsvn.digium.com/svn/asterisk/trunk@126572 65c4cc65-6c06-0410-ace0-fbb531ad65f3 --- channels/chan_oss.c | 5 +++++ channels/console_gui.c | 35 ++++++++++++++++++++++++----------- channels/console_video.c | 38 ++++++++++++++++++++++++++------------ channels/console_video.h | 1 + 4 files changed, 56 insertions(+), 23 deletions(-) diff --git a/channels/chan_oss.c b/channels/chan_oss.c index e665eb2e97..a3908a5048 100644 --- a/channels/chan_oss.c +++ b/channels/chan_oss.c @@ -1387,6 +1387,11 @@ static struct chan_oss_pvt *store_config(struct ast_config *cfg, char *ctg) system(cmd); ast_free(cmd); } + + /* if the config file requested to start the GUI, do it */ + if (get_gui_startup(o->env)) + console_video_start(o->env, NULL); + if (o == &oss_default) /* we are done with the default */ return NULL; diff --git a/channels/console_gui.c b/channels/console_gui.c index a228c3ce74..480f6f1ce3 100644 --- a/channels/console_gui.c +++ b/channels/console_gui.c @@ -50,7 +50,9 @@ handled differently according to their location: keystrokes are used as keypad functions, or as text input if we are in text-input mode. - drag on some keypad areas (sliders etc.) are mapped to the - corresponding functions; + corresponding functions (mute/unmute audio and video, + enable/disable Picture-in-Picture, freeze the incoming video, + dial numbers, pick up or hang up a call, ...) Configuration options control the appeareance of the gui: @@ -58,9 +60,8 @@ Configuration options control the appeareance of the gui: keypad_font = /tmp/font.png ; the font to use for output For future implementation, intresting features can be the following: -- freeze of the video coming from our remote party - save of the whole SDL window as a picture -- oudio output device switching +- audio output device switching The audio switching feature should allow changing the device or switching to a recorded message for audio sent to remote party. @@ -343,10 +344,10 @@ enum skin_area { associated with the audio device markers, clicking on these markers will change the source device for audio output */ +#endif + /* Keys related to video sources */ KEY_FREEZE = 220, /* freeze the incoming video */ KEY_CAPTURE = 221, /* capture the whole SDL window as a picture */ -#endif - /* video source switching key(s) */ KEY_PIP = 230, /*indexes between 231 and 239 have been reserved for the "keys" associated with the device thumbnails, clicking on these pictures @@ -389,12 +390,19 @@ static char *keypad_toggle(struct video_desc *env, int index) case KEY_SENDVIDEO: /* send or do not send video */ env->out.sendvideo = !env->out.sendvideo; break; + case KEY_PIP: /* enable or disable Picture in Picture */ env->out.picture_in_picture = !env->out.picture_in_picture; break; + case KEY_MUTE: /* send or do not send audio */ ast_cli_command(env->gui->outfd, "console mute toggle"); break; + + case KEY_FREEZE: /* freeze/unfreeze the incoming frames */ + env->frame_freeze = !env->frame_freeze; + break; + #ifdef notyet case KEY_AUTOANSWER: { struct chan_oss_pvt *o = find_desc(oss_active); @@ -737,6 +745,7 @@ static void handle_mousedown(struct video_desc *env, SDL_MouseButtonEvent button case KEY_AUTOANSWER: case KEY_SENDVIDEO: /* send or not send the video */ case KEY_PIP: /* activate/deactivate picture in picture mode */ + case KEY_FREEZE: /* freeze/unfreeze the incoming video */ keypad_toggle(env, index); break; @@ -746,8 +755,6 @@ static void handle_mousedown(struct video_desc *env, SDL_MouseButtonEvent button break; #ifdef notyet /* XXX for future implementations */ - case KEY_FREEZE: - break case KEY_CAPTURE: break; #endif @@ -1407,6 +1414,9 @@ static void sdl_setup(struct video_desc *env) if (set_win(gui->screen, &gui->win[WIN_REMOTE], dpy_fmt, env->rem_dpy.w, env->rem_dpy.h, x0-kp_w/2-BORDER-env->rem_dpy.w, y0)) goto no_sdl; + /* unfreeze incoming frames if set (to avoid showing nothing) */ + env->frame_freeze = 0; + if (set_win(gui->screen, &gui->win[WIN_LOCAL], dpy_fmt, env->loc_dpy.w, env->loc_dpy.h, x0+kp_w/2+BORDER, y0)) @@ -1496,6 +1506,7 @@ static int kp_match_area(const struct keypad_entry *e, int x, int y) struct _s_k { const char *s; int k; }; static struct _s_k gui_key_map[] = { + {"FREEZE", KEY_FREEZE}, {"PIP", KEY_PIP}, {"PICK_UP", KEY_PICK_UP }, {"PICKUP", KEY_PICK_UP }, @@ -1537,6 +1548,8 @@ static int gui_map_token(const char *s) * token circle xc yc diameter * token circle xc yc x1 y1 h # ellipse, main diameter and height * token rect x0 y0 x1 y1 h # rectangle with main side and eight + * token x0 y0 w h # horizontal rectangle (short format) + * # this is used e.g. for message boards * token is the token to be returned, either a character or a symbol * as KEY_* above * Return 1 on success, 0 on error. @@ -1578,10 +1591,10 @@ static int keypad_cfg_read(struct gui_info *gui, const char *val) else if (e.c == KEY_EDIT) r = gui->kp_edit; if (r) { - r->x = atoi(s2); - r->y = e.x0; - r->w = e.y0; - r->h = e.x1; + r->x = atoi(s2); /* this becomes x0 */ + r->y = e.x0; /* this becomes y0 */ + r->w = e.y0; /* this becomes w */ + r->h = e.x1; /* this becomes h */ break; } if (strcasecmp(s2, "circle")) /* invalid */ diff --git a/channels/console_video.c b/channels/console_video.c index dbe7903df1..a68a930f87 100644 --- a/channels/console_video.c +++ b/channels/console_video.c @@ -137,6 +137,11 @@ void console_video_uninit(struct video_desc *env) { } +int get_gui_startup(struct video_desc* env) +{ + return 0; /* no gui here */ +} + int console_video_formats = 0; #else /* defined(HAVE_FFMPEG) && defined(HAVE_SDL) */ @@ -248,11 +253,11 @@ struct video_desc { struct fbuf_t rem_dpy; /* display remote video, no buffer (it is in win[WIN_REMOTE].bmp) */ struct fbuf_t loc_dpy; /* display local source, no buffer (managed by SDL in bmp[1]) */ - /*display for sources in additional windows, - ideally infinite additional sources could be present - pratically we assume a maximum of 9 sources to show*/ + /* geometry of the thumbnails for all video sources. */ struct fbuf_t src_dpy[MAX_VIDEO_SOURCES]; /* no buffer allocated here */ - + + int frame_freeze; /* flag to freeze the incoming frame */ + /* local information for grabbers, codecs, gui */ struct gui_info *gui; struct video_dec_desc *in; /* remote video descriptor */ @@ -274,6 +279,14 @@ void fbuf_free(struct fbuf_t *b) b->pix_fmt = x.pix_fmt; } +/* return the status of env->stayopen to chan_oss, as the latter + * does not have access to fields of struct video_desc + */ +int get_gui_startup(struct video_desc* env) +{ + return env->stayopen; +} + #if 0 /* helper function to print the amount of memory used by the process. * Useful to track memory leaks, unfortunately this code is OS-specific @@ -432,6 +445,7 @@ static int video_out_uninit(struct video_desc *env) v->devices[i].status_index = 0; } v->picture_in_picture = 0; + env->frame_freeze = 0; return -1; } @@ -545,7 +559,7 @@ static int video_out_init(struct video_desc *env) void console_video_uninit(struct video_desc *env) { int i, t = 100; /* initial wait is shorter, than make it longer */ - if (env->stayopen == 0) { /* in a call */ + if (env->stayopen == 0) { /* gui opened by a call, do the shutdown */ env->shutdown = 1; for (i=0; env->shutdown && i < 10; i++) { if (env->owner) @@ -555,6 +569,7 @@ void console_video_uninit(struct video_desc *env) if (env->owner) ast_channel_lock(env->owner); } + env->vthread = NULL; } env->owner = NULL; /* this is unconditional */ } @@ -920,7 +935,8 @@ static void *video_thread(void *arg) while (v->dec_in_dpy) { struct fbuf_t *tmp = v->dec_in_dpy; /* store current pointer */ - if (v->d_callbacks->dec_run(v, tmp)) + /* decode the frame, but show it only if not frozen */ + if (v->d_callbacks->dec_run(v, tmp) && !env->frame_freeze) show_frame(env, WIN_REMOTE); tmp->used = 0; /* mark buffer as free */ tmp->ebit = 0; @@ -1045,7 +1061,7 @@ void console_video_start(struct video_desc *env, struct ast_channel *owner) if (env == NULL) /* video not initialized */ return; env->owner = owner; /* work even if no owner is specified */ - if (env->stayopen) + if (env->vthread) return; /* already initialized, nothing to do */ init_env(env); env->out.enc = map_config_video_format(env->codec_name); @@ -1073,8 +1089,6 @@ void console_video_start(struct video_desc *env, struct ast_channel *owner) ast_pthread_create_background(&env->vthread, NULL, video_thread, env); /* detach the thread to make sure memory is freed on termination */ pthread_detach(env->vthread); - if (env->owner == NULL) - env->stayopen = 1; /* manually opened so don't close on hangup */ } /* @@ -1204,6 +1218,7 @@ int console_video_cli(struct video_desc *env, const char *var, int fd) } else if (!strcasecmp(var, "fps")) { ast_cli(fd, "fps is [%d]\n", env->out.fps); } else if (!strcasecmp(var, "startgui")) { + env->stayopen = 1; console_video_start(env, NULL); } else if (!strcasecmp(var, "stopgui") && env->stayopen != 0) { env->stayopen = 0; @@ -1236,7 +1251,7 @@ int console_video_config(struct video_desc **penv, return 1; /* error */ } - /* set default values */ + /* set default values - 0's are already there */ env->out.device_primary = 0; env->out.device_secondary = 0; env->out.fps = 5; @@ -1244,7 +1259,6 @@ int console_video_config(struct video_desc **penv, env->out.sendvideo = 1; env->out.qmin = 3; env->out.device_num = 0; - env->out.picture_in_picture = 0; /* PiP mode intially disabled */ } CV_START(var, val); CV_F("videodevice", device_table_fill(env->out.devices, &env->out.device_num, val)); @@ -1255,7 +1269,7 @@ int console_video_config(struct video_desc **penv, CV_F("remote_size", video_geom(&env->rem_dpy, val)); CV_STR("keypad", env->keypad_file); CV_F("region", keypad_cfg_read(env->gui, val)); - CV_F("startgui", console_video_start(env, NULL)); // support enabling gui at startup + CV_UINT("startgui", env->stayopen); /* enable gui at startup */ CV_STR("keypad_font", env->keypad_font); CV_STR("sdl_videodriver", env->sdl_videodriver); CV_UINT("fps", env->out.fps); diff --git a/channels/console_video.h b/channels/console_video.h index fb1a488219..f88e5fa1d8 100644 --- a/channels/console_video.h +++ b/channels/console_video.h @@ -97,6 +97,7 @@ int console_video_cli(struct video_desc *env, const char *var, int fd); int console_video_config(struct video_desc **penv, const char *var, const char *val); void console_video_uninit(struct video_desc *env); void console_video_start(struct video_desc *env, struct ast_channel *owner); +int get_gui_startup(struct video_desc* env); /* console_board.c */ -- GitLab