/* wmget - A background download manager as a Window Maker dock app Copyright (c) 2001-2003 Aaron Trickey Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. ******************************************************************** server.c - Manage the dock app display, accept & spawn jobs When ``wmget --dock'' is invoked, main() calls server(), defined below. This initializes the dock app window, the shared memory segment, and the job input queue, and then enters the main loop whereby it accepts X events (such as redraws or clicks), accepts job requests (from request(), in request.c), and monitors shared memory, updating the display as necessary. It forks off children to handle accepted jobs; see retrieve.c. */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include "wmget.h" #include "wmget.xpm" #include "dockapp/dockapp.h" static ServerConfig config; /*********************************************************************** * Text Drawing * The various CHAR_* consts locate and dimension the chars on the xpm. * Call init_font() to set up the CHAR_X and CHAR_Y tables, then * draw_string() to put text on the xpm. */ static const int CHAR_WIDTH = 6; static const int CHAR_HEIGHT = 7; static const int CHAR_UCALPHA_X = 1; static const int CHAR_UCALPHA_Y = 85; static const int CHAR_LCALPHA_X = 1; static const int CHAR_LCALPHA_Y = 95; static const int CHAR_SYMNUM_X = 1; static const int CHAR_SYMNUM_Y = 105; static int CHAR_X[128]; static int CHAR_Y[128]; static void init_font (void) { int i; int *cx, *cy; for (i = 0, cx = CHAR_X, cy = CHAR_Y; i < 128; ++i, ++cx, ++cy) { if (i > 'z') { *cx = CHAR_SYMNUM_X; /* 1st SYMNUM is the space */ *cy = CHAR_SYMNUM_Y; } else if (i >= 'a') { *cx = CHAR_LCALPHA_X + CHAR_WIDTH * (i - 'a'); *cy = CHAR_LCALPHA_Y; } else if (i > 'Z') { *cx = CHAR_SYMNUM_X; *cy = CHAR_SYMNUM_Y; } else if (i >= 'A') { *cx = CHAR_UCALPHA_X + CHAR_WIDTH * (i - 'A'); *cy = CHAR_UCALPHA_Y; } else if (i > '9') { *cx = CHAR_SYMNUM_X; *cy = CHAR_SYMNUM_Y; } else if (i >= ' ') { *cx = CHAR_SYMNUM_X + CHAR_WIDTH * (i - ' '); *cy = CHAR_SYMNUM_Y; } else { *cx = CHAR_SYMNUM_X; *cy = CHAR_SYMNUM_Y; } } } static void draw_string (const char *str, int x, int y) { for ( ; *str; ++str) { dockapp_overlay_pixmap ( CHAR_X[(int)*str], CHAR_Y[(int)*str], x, y, CHAR_WIDTH, CHAR_HEIGHT); x += CHAR_WIDTH; } } /*********************************************************************** * Button Widgets */ static const int BTN_PAUSE_X = 128; static const int BTN_STOP_X = 147; static const int BTN_Y = 37; static const int BTN_WIDTH = 19; static const int BTN_HEIGHT = 9; /*********************************************************************** * Progress Bars */ /* Coords and dimensions refer to the pbars, excluding the borders * which make up the ``ditches'' */ static const int PBAR_Y[4] = { 5, 20, 35, 50, }; static const int PBAR_X = 3; /* These are the graphics for the bars themselves. */ static const int PBAR_FULL_X = 67; static const int PBAR_FULL_Y = 37; static const int PBAR_EMPTY_X = 67; static const int PBAR_EMPTY_Y = 47; static const int PBAR_LENGTH = 58; static const int PBAR_HEIGHT = 9; static int bar_selected = -1; static void draw_pbar (int trough_x, int trough_y, int value, int max) { int width = ((unsigned long) PBAR_LENGTH * value) / max; dockapp_copy_pixmap ( PBAR_FULL_X, PBAR_FULL_Y, trough_x, trough_y, width, PBAR_HEIGHT); dockapp_copy_pixmap ( PBAR_EMPTY_X, PBAR_EMPTY_Y, trough_x + width, trough_y, PBAR_LENGTH - width, PBAR_HEIGHT); } static const char *const DEFAULT_TEXT[] = { " wmget", "", "", "", }; static void draw_pbars (void) { int i; for (i = 0; i < 4; ++i) { Job *j = &shmem->jobs[i]; if (j->status == J_EMPTY) { draw_pbar (PBAR_X, PBAR_Y[i], 0, 1); draw_string (DEFAULT_TEXT[i], PBAR_X + 1, PBAR_Y[i] + 1); continue; } if (i == bar_selected) { /* percentage (or error) + stop button */ draw_pbar (PBAR_X, PBAR_Y[i], 0, 1); if (j->status != J_FAILED) { char pct[4]; sprintf (pct, "%02lu%%", 100L * j->progress / j->prog_max); draw_string (pct, PBAR_X + 1, PBAR_Y[i] + 1); } else { char err[9]; strncpy (err, j->error, 8); err[8] = '\0'; draw_string (err, PBAR_X + 1, PBAR_Y[i] + 1); } dockapp_copy_pixmap ( BTN_STOP_X, BTN_Y, PBAR_X + PBAR_LENGTH - BTN_WIDTH, PBAR_Y[i], BTN_WIDTH, BTN_HEIGHT); } else { /* name + scrollbar, or error */ draw_pbar (PBAR_X, PBAR_Y[i], j->progress, j->prog_max); draw_string (j->options.display, PBAR_X + 1, PBAR_Y[i] + 1); } } } /*********************************************************************** * Shared memory segment: global pointer, constructor */ Shmem *shmem = 0; static int init_shmem (void) { int shmid; int i; if ((shmid = shmget (IPC_PRIVATE, sizeof *shmem, SHM_R | SHM_W)) < 0) { error_sys ("could not allocate shared memory segment [shmget()]"); return 1; } if ((shmem = shmat (shmid, 0, 0)) == (void *) -1) { error_sys ("could not attach shared memory segment [shmat()]"); return 1; } for (i = 0; i < 4; ++i) { shmem->jobs[i].status = J_EMPTY; shmem->jobs[i].options.display[0] = 0; shmem->jobs[i].progress = 0; shmem->jobs[i].prog_max = 0; } return 0; } /*********************************************************************** * start_job(): Spawn a new process and call retrieve() in there. * Note that `j' must be in shared memory. */ static int start_job (Job *j) { int f; j->prog_max = 1; j->progress = 0; j->status = J_INIT; j->stop_request = 0; f = fork (); if (f < 0) { error_sys ("could not create child process [fork()]"); return 1; } else if (f == 0) { /* child */ retrieve (j); if (j->status == J_FAILED) { /* Sleep until user acks the error. */ while (!j->stop_request) { struct timespec sleeptime = { 0, 100000000L }; nanosleep (&sleeptime, NULL); } } j->status = J_EMPTY; exit (0); } return 0; } /*********************************************************************** * The Job Queue. Okay, this is a little cheesy right now. */ static Job *job_queue[MAX_QUEUED_JOBS] = { 0 }; static size_t job_queue_depth = 0; static job_id_t next_job_id = 1; /* Job id 0 is never valid */ /*********************************************************************** * process_queue(): If a job has finished, pull it from its slot. * If a slot is open, pull the next job from the queue. */ static int process_queue (void) { size_t i; for (i = 0; i < MAX_ACTIVE_JOBS; ++i) { switch (shmem->jobs[i].status) { default: /* job occupying slot */ continue; case J_EMPTY: /* aha. see if there is anything queued up */ if (!job_queue_depth) continue; shmem->jobs[i] = *job_queue[--job_queue_depth]; free (job_queue[job_queue_depth]); debug ("Pulled new active job %lu off queue", shmem->jobs[i].job_id); if (start_job (&shmem->jobs[i])) return 1; continue; } } return 0; } /*********************************************************************** * cancel_job(): Cancel a job. If it's running, stop it; if it's * in the queue, dequeue it; if it's nowhere, do nothing. */ static int cancel_job (job_id_t job_id) { Job *j; Job **jp; /* First search the active jobs. */ for (j = shmem->jobs; j < shmem->jobs + MAX_ACTIVE_JOBS; ++j) { if (j->job_id == job_id) { switch (j->status) { case J_EMPTY: case J_STOPPING: case J_COMPLETE: /* Job has already completed. */ return 0; case J_FAILED: /* Job has already failed; this simply clears it * out. */ j->status = J_COMPLETE; return 0; default: /* just to keep the compiler warnings at bay */ break; } ++j->stop_request; /* Request job termination. */ return 0; } } /* Okay, now search the pending queue. */ for (jp = job_queue; jp < job_queue + job_queue_depth; ++jp) { if (*jp && (*jp)->job_id == job_id) { /* Simply delete it from the queue. */ --job_queue_depth; memmove (jp, jp + 1, ((job_queue + job_queue_depth) - jp) * sizeof (Job *)); return 0; } } /* Job not found. This is not an error, as we assume that the * job has already died off. */ return 0; } /*********************************************************************** * client_*(): Issue a response back to the client. */ static void client_error (FILE *fp, const char *fmt, ...) { va_list ap; va_start (ap, fmt); fprintf (fp, RESPONSE_ERROR " "); vfprintf (fp, fmt, ap); va_end (ap); } static void client_job_accepted (FILE *fp, job_id_t job_id) { fprintf (fp, RESPONSE_JOB_ACCEPTED " %lu\n", job_id); } static void client_job_canceled (FILE *fp, job_id_t job_id) { fprintf (fp, RESPONSE_JOB_CANCELED " %lu\n", job_id); } static void client_list_header (FILE *fp) { debug ("client_list_header()"); fprintf (fp, RESPONSE_LIST_COMING "\n"); } static void client_list_job (FILE *fp, Job *j) { const char *status; switch (j->status) { default: status = "UNKNOWN: Internal Error!"; break; case J_INIT: status = "INIT: Waiting to start"; break; case J_RUNNING: status = "RUNNING: Currently retrieving"; break; case J_PAUSED: status = "PAUSED: Download suspended"; break; case J_STOPPING: status = "STOPPING: Got stop request"; break; case J_COMPLETE: status = "COMPLETE: Download complete"; break; case J_FAILED: status = j->error; break; } fprintf (fp, "Job %lu [%9s]: %lu/%lu %s\n%s => %s\n\n", j->job_id, j->options.display, j->progress, j->prog_max, status, j->source_url, j->options.save_to); } static int insert_job (Job *j) { if (job_queue_depth >= MAX_QUEUED_JOBS) { error ("Job queue full"); free (j); return 1; } j->job_id = next_job_id++; debug ("Accepted job..."); debug_dump_job (j); job_queue[job_queue_depth++] = j; return 0; } static Job *init_job (Request *req, FILE *errfp) { struct stat st; const char *base_first; const char *base_last; size_t base_sz; Job *j = malloc (sizeof (Job)); if (!j) { client_error (errfp, "Dockapp out of memory!"); return 0; } STRCPY_TO_ARRAY (j->source_url, req->source_url); j->status = J_INIT; j->progress = j->prog_max = 0; j->stop_request = 0; j->options = config.job_defaults; /* Copy over any applicable options---except save_to and display, * which merit special consideration below. */ if (req->overwrite != -1) j->options.overwrite = req->overwrite; if (req->continue_from != -1) j->options.continue_from = req->continue_from; if (req->proxy) STRCPY_TO_ARRAY (j->options.proxy, req->proxy); if (req->follow != -1) j->options.follow = req->follow; if (req->user_agent) STRCPY_TO_ARRAY (j->options.user_agent, req->user_agent); if (req->use_ascii != -1) j->options.use_ascii = req->use_ascii; if (req->referer) STRCPY_TO_ARRAY (j->options.referer, req->referer); if (req->include != -1) j->options.include = req->include; if (req->interface) STRCPY_TO_ARRAY (j->options.interface, req->interface); if (req->proxy_auth) STRCPY_TO_ARRAY (j->options.proxy_auth, req->proxy_auth); if (req->auth) STRCPY_TO_ARRAY (j->options.auth, req->auth); /* Extract the "base name" (last slash-delimited component) of the * source URL for future use. */ base_last = j->source_url + strlen (j->source_url) - 1; while (*base_last == '/' && base_last > j->source_url) --base_last; base_first = base_last; while (*base_first != '/' && base_first > j->source_url) --base_first; base_sz = base_last - base_first; ++base_first; /* get it past that initial slash */ if (base_sz == 0) { /* Uh, oh... invalid source_url anyway... give up. */ client_error (errfp, "Invalid URL '%s'", j->source_url); goto RETURN_NULL; } debug ("baselen %d", base_sz); /* If no display-name was provided, use the basename. */ if (req->display) { STRCPY_TO_ARRAY (j->options.display, req->display); } else { size_t n = base_sz; if (n > sizeof j->options.display - 1) n = sizeof j->options.display - 1; strncpy (j->options.display, base_first, n); j->options.display[n] = '\0'; debug ("display was empty... set it to %s", j->options.display); } /* If there was a save-to location provided, copy it into the job. * If it's a relative path, make it relative to the download * directory. If it wasn't given, just copy the download dir. */ if (req->save_to) { if (req->save_to[0] == '/') { debug ("Reqest contained absolute dir."); } else { STRCPY_TO_ARRAY (j->options.save_to, config.job_defaults.save_to); if (strlen (j->options.save_to) + strlen (req->save_to) + 2 > MAXPATHLEN) { client_error (errfp, "Download output pathname too long"); goto RETURN_NULL; } strcat (j->options.save_to, "/"); strcat (j->options.save_to, req->save_to); debug ("Resolved output to '%s'", j->options.save_to); } } else { STRCPY_TO_ARRAY (j->options.save_to, config.job_defaults.save_to); debug ("Defaulted output to '%s'", j->options.save_to); } /* Now we've got something... let's see what it is... */ if (stat (j->options.save_to, &st)) { if (errno == ENOENT) { /* Name of a file which doesn't exist... ready to save. */ debug ("Target does not exist."); return j; } error_sys ("could not stat(`%s')", j->options.save_to); client_error (errfp, "Failed when checking pathname '%s'", j->options.save_to); goto RETURN_NULL; } /* If it's a directory name, append the basename from above and * re-stat. */ if (S_ISDIR (st.st_mode)) { int offset = strlen (j->options.save_to); debug ("Is a directory."); if (offset + base_sz + 2 > sizeof j->options.save_to) { client_error (errfp, "Save-to path too long!"); goto RETURN_NULL; } j->options.save_to[offset] = '/'; strncpy (j->options.save_to + offset + 1, base_first, base_sz); j->options.save_to[offset + 1 + base_sz] = '\0'; debug ("Extended to %s", j->options.save_to); if (stat (j->options.save_to, &st)) { if (errno == ENOENT) { return j; } error_sys ("could not stat(`%s')", j->options.save_to); client_error (errfp, "Failed when checking pathname '%s'", j->options.save_to); goto RETURN_NULL; } } /* If we're here, it's not a directory but it exists. */ debug ("%s Exists!", j->options.save_to); if (!j->options.overwrite && !j->options.continue_from) { client_error (errfp, "File '%s' exists and --overwrite not specified", j->options.save_to); goto RETURN_NULL; } /* For continuations, get the file length. If the file does not * exist, just disable continuation; this is not an error. * (Continuation may now be permanently enabled in the RC file.) */ if (j->options.continue_from) { if (S_ISREG (st.st_mode)) { j->options.continue_from = st.st_size; } else { j->options.continue_from = 0; } } /* Finally, check permissions */ if ((st.st_mode & S_IWOTH) || ((st.st_mode & S_IWGRP) && st.st_gid == getegid ()) || ((st.st_mode & S_IWUSR) && st.st_uid == geteuid ())) return j; client_error (errfp, "File '%s' exists and is not writable.\n", j->options.save_to); RETURN_NULL: free (j); return 0; } /*********************************************************************** * process_*(): Implementations of each server command. */ static void process_get ( FILE *fp, char *argnames[], char *argvalues[]) { char **an, **av; Request req; Job *job; debug ("process_get()"); /* Don't waste the user's time if we're full already... */ if (job_queue_depth >= MAX_QUEUED_JOBS) { client_error (fp, "Job queue full"); return; } /* Empty out the request object... */ clear_request (&req); /* And parse the args... */ for (an = argnames, av = argvalues; *an && *av; ++an, ++av) { if (strcasecmp (*an, ARG_GET_SOURCE_URL) == 0) { if (strlen (*av) > MAXURL) { client_error (fp, "Source URL too long"); return; } req.source_url = *av; } else if (strcasecmp (*an, ARG_GET_DISPLAY) == 0) { req.display = *av; } else if (strcasecmp (*an, ARG_GET_SAVE_TO) == 0) { req.save_to = *av; } else if (strcasecmp (*an, ARG_GET_CONTINUE_FROM) == 0) { char *end; req.continue_from = strtoul (*av, &end, 0); if (*end) { client_error (fp, ARG_GET_CONTINUE_FROM ": must be an integer"); return; } } else if (strcasecmp (*an, ARG_GET_OVERWRITE) == 0) { req.overwrite = 1; } else if (strcasecmp (*an, ARG_GET_PROXY) == 0) { req.proxy = *av; } else if (strcasecmp (*an, ARG_GET_FOLLOW) == 0) { req.follow = atoi (*av); } else if (strcasecmp (*an, ARG_GET_UA) == 0) { req.user_agent = *av; } else if (strcasecmp (*an, ARG_GET_USE_ASCII) == 0) { req.use_ascii = 1; } else if (strcasecmp (*an, ARG_GET_REFERER) == 0) { req.referer = *av; } else if (strcasecmp (*an, ARG_GET_INCLUDE) == 0) { req.include = 1; } else if (strcasecmp (*an, ARG_GET_INTERFACE) == 0) { req.interface = *av; } else if (strcasecmp (*an, ARG_GET_PROXY_AUTH) == 0) { req.proxy_auth = *av; } else if (strcasecmp (*an, ARG_GET_AUTH) == 0) { req.auth = *av; } else { client_error (fp, "Unknown parameter '%s'", *an); return; } } job = init_job (&req, fp); if (!job) return; if (insert_job (job)) { client_error (fp, "Invalid job parameters"); free (job); } else { client_job_accepted (fp, job->job_id); } } static void process_cancel ( FILE *fp, char *argnames[], char *argvalues[]) { char **an, **av; job_id_t job_id = 0; /* job id 0 is never valid */ debug ("process_cancel()"); for (an = argnames, av = argvalues; *an && *av; ++an, ++av) { if (strcasecmp (*an, ARG_CANCEL_JOBID) == 0) { job_id = strtoul (*av, 0, 0); } else { client_error (fp, "Unknown parameter '%s'", *an); return; } } if (job_id == 0) { client_error (fp, CMD_CANCEL " requires the argument " ARG_CANCEL_JOBID); return; } if (cancel_job (job_id)) client_error (fp, "Cancel failed"); else client_job_canceled (fp, job_id); } void process_list (FILE *fp, char *argnames[], char *argvalues[]) { Job *j, **jp; (void)argnames; (void)argvalues; debug ("process_list()"); client_list_header (fp); /* First list the active jobs. */ for (j = shmem->jobs; j < shmem->jobs + MAX_ACTIVE_JOBS; ++j) if (j->status != J_EMPTY) client_list_job (fp, j); /* Then the waiting jobs. */ for (jp = job_queue; jp < job_queue + job_queue_depth; ++jp) client_list_job (fp, *jp); } /*********************************************************************** * process_request(): Accept a command and parameters, process it, * and reply. */ static void process_request (FILE *fp) { char command[MAXCMDLEN]; char *arg; int nargs; char *argnames[MAXCMDARGS + 1]; char *argvalues[MAXCMDARGS + 1]; /* A command line always comes first. */ if (!fgets (command, sizeof command - 1, fp)) { debug ("No command on pipe!"); return; } /* Arguments come after whitespace. Note that not all commands have * args. Each argument looks like this: PARAMETERNAME(VALUE). * Hey, don't tell me *none* of you have ever done AS/400 CL.... */ nargs = 0; arg = command + strcspn (command, " \t\r\n"); *arg++ = 0; arg += strspn (arg, " \t"); while (*arg && *arg != '\n' && *arg != '\r') { if (nargs > MAXCMDARGS - 1) { client_error (fp, "Too many arguments!"); return; } argnames[nargs] = arg; if (!(arg = strchr (arg, '('))) { client_error (fp, "Argument missing value"); return; } *arg++ = 0; argvalues[nargs] = arg; /* Arguments are terminated by a ), of course, but they may * also contain characters (such as )) quoted by \. */ while (*arg && *arg != ')') { if (*arg == '\\') { if (!arg[1]) { client_error (fp, "Argument missing closing ), " "ended with \\ by itself"); return; } /* strlen(arg+1)+1 = strlen(arg)-1+1 = strlen(arg) */ memmove (arg, arg + 1, strlen (arg)); } ++arg; } if (!arg) { client_error (fp, "Argument missing closing )"); return; } *arg++ = 0; arg += strspn (arg, " \t"); ++nargs; } argnames[nargs] = 0; argvalues[nargs] = 0; /* Got a valid command/argument set. Process it. */ if (strcasecmp (command, CMD_GET) == 0) process_get (fp, argnames, argvalues); else if (strcasecmp (command, CMD_CANCEL) == 0) process_cancel (fp, argnames, argvalues); else if (strcasecmp (command, CMD_LIST) == 0) process_list (fp, argnames, argvalues); else client_error (fp, "Unknown command"); } /*********************************************************************** * on_iq_ready(): invoked by the dockapp lib when there are connections * pending on the iq */ static dockapp_rv_t on_iq_ready (void *unused0, short unused1) { FILE *fp; (void)unused0; (void)unused1; debug ("on_iq_ready"); if ((fp = iq_server_accept ())) { process_request (fp); fclose (fp); } return dockapp_ok; } static int init_grim_reaper (void) { struct sigaction sa; sa.sa_handler = SIG_IGN; sigemptyset (&sa.sa_mask); sa.sa_flags = SA_NOCLDSTOP | SA_RESTART; /* Obsolete - sa.sa_restorer = 0; */ if (sigaction (SIGCHLD, &sa, 0)) { error_sys ("sigaction(SIGCHLD) failed"); return 1; } return 0; } static dockapp_rv_t on_click_pbar ( void *cbdata, int x_unused, int y_unused) { int which = (int)cbdata; (void)x_unused; (void)y_unused; debug ("got a click on pbar %d", which); if (bar_selected == which) { /* Selected bar gets deselected. */ bar_selected = -1; } else { bar_selected = which; } return dockapp_ok; } static dockapp_rv_t on_click_stop ( void *cbdata, int x_unused, int y_unused) { int which = (int)cbdata; (void)x_unused; (void)y_unused; debug ("got a click on stop %d", which); if (bar_selected == which) { /* got a stop request (only works on selected pbar) */ ++shmem->jobs[which].stop_request; } return dockapp_ok; } static dockapp_rv_t on_periodic_callback (void *cbdata) { (void)cbdata; if (process_queue ()) return dockapp_exit; draw_pbars (); return dockapp_ok; } static dockapp_rv_t on_got_selection (void *cbdata, const char *str) { Request req; Job *j; (void)cbdata; debug ("on_got_selection >> %s", str); if (strlen (str) > MAXURL) { error ("rejecting job submission: URL too long!"); return dockapp_ok; } clear_request (&req); req.source_url = str; j = init_job (&req, stderr); if (!j) { return dockapp_ok; } debug ("submitting job for [%s]...", j->source_url); if (insert_job (j)) { free (j); debug ("insert_job rejected it!"); } return dockapp_ok; } static dockapp_rv_t on_middle_click (void *cbdata, int x, int y) { (void)cbdata; (void)x; (void)y; debug ("on_middle_click"); dockapp_request_selection_string (on_got_selection, 0); return dockapp_ok; } /* This is the main routine for the dock app (the first instance * started) */ int server (int argc, char **argv) { int i; config_server (argc, argv, &config); if (init_grim_reaper ()) return 1; if (init_shmem ()) return 1; if (iq_server_init ()) return 1; init_font (); dockapp_init_gui ("wmget", argv, wmget_xpm); for (i = 0; i < 4; ++i) { dockapp_add_clickregion ( PBAR_X + PBAR_LENGTH - BTN_WIDTH, PBAR_Y[i], PBAR_LENGTH, PBAR_HEIGHT, Button1Mask, on_click_stop, (void *)i); dockapp_add_clickregion ( PBAR_X, PBAR_Y[i], PBAR_LENGTH, PBAR_HEIGHT, Button1Mask, on_click_pbar, (void *)i); } dockapp_add_clickregion ( 0, 0, 64, 64, Button2Mask, on_middle_click, 0); dockapp_add_pollfd (iq_get_listen_fd (), POLLIN, on_iq_ready, 0); dockapp_set_periodic_callback (400, on_periodic_callback, 0); /* Perform one initial refresh. */ on_periodic_callback (0); dockapp_run (); return 0; }