Merge branch 'helmut' into helmutexp
authorHelmut Grohne <helmut@subdivi.de>
Wed, 29 Jul 2009 20:14:26 +0000 (22:14 +0200)
committerHelmut Grohne <helmut@subdivi.de>
Wed, 29 Jul 2009 20:14:26 +0000 (22:14 +0200)
Conflicts:
uzbl.c

1  2 
uzbl.c
uzbl.h

diff --combined uzbl.c
--- 1/uzbl.c
--- 2/uzbl.c
+++ b/uzbl.c
@@@ -53,9 -53,6 +53,9 @@@
  #include <errno.h>
  #include <fcntl.h>
  #include <signal.h>
 +#include <poll.h>
 +#include <sys/uio.h>
 +#include <sys/ioctl.h>
  #include <assert.h>
  #include "uzbl.h"
  #include "config.h"
@@@ -67,13 -64,13 +67,13 @@@ cons
  GOptionEntry entries[] =
  {
      { "uri",      'u', 0, G_OPTION_ARG_STRING, &uzbl.state.uri,
 -        "Uri to load at startup (equivalent to 'set uri = URI')", "URI" },
 +        "Uri to load at startup (equivalent to 'uzbl <uri>' or 'set uri = URI' after uzbl has launched)", "URI" },
      { "verbose",  'v', 0, G_OPTION_ARG_NONE,   &uzbl.state.verbose,
          "Whether to print all messages or just errors.", NULL },
      { "name",     'n', 0, G_OPTION_ARG_STRING, &uzbl.state.instance_name,
          "Name of the current instance (defaults to Xorg window id)", "NAME" },
      { "config",   'c', 0, G_OPTION_ARG_STRING, &uzbl.state.config_file,
 -        "Config file (this is pretty much equivalent to uzbl < FILE )", "FILE" },
 +        "Path to config file or '-' for stdin", "FILE" },
      { "socket",   's', 0, G_OPTION_ARG_INT, &uzbl.state.socket_id,
          "Socket ID", "SOCKET" },
      { "geometry", 'g', 0, G_OPTION_ARG_STRING, &uzbl.gui.geometry,
@@@ -108,92 -105,86 +108,92 @@@ typedef const struct 
  #define PTR_C_INT(var,    fun) { .ptr.i = (int*)&(var), .type = TYPE_INT, .dump = 0, .writeable = 0, .func = fun }
  #define PTR_C_FLOAT(var,  fun) { .ptr.f = &(var), .type = TYPE_FLOAT, .dump = 0, .writeable = 0, .func = fun }
  
- const struct {
+ const struct var_name_to_ptr_t {
      const char *name;
      uzbl_cmdprop cp;
  } var_name_to_ptr[] = {
 -/*    variable name         pointer to variable in code                  dump callback function    */
 +/*    variable name            pointer to variable in code                     dump callback function    */
  /*  ---------------------------------------------------------------------------------------------- */
 -    { "uri",                 PTR_V_STR(uzbl.state.uri,                    1,   cmd_load_uri)},
 -    { "verbose",             PTR_V_INT(uzbl.state.verbose,                1,   NULL)},
 -    { "mode",                PTR_V_INT(uzbl.behave.mode,                  0,   NULL)},
 -    { "inject_html",         PTR_V_STR(uzbl.behave.inject_html,           0,   cmd_inject_html)},
 -    { "base_url",            PTR_V_STR(uzbl.behave.base_url,              1,   NULL)},
 -    { "html_endmarker",      PTR_V_STR(uzbl.behave.html_endmarker,        1,   NULL)},
 -    { "html_mode_timeout",   PTR_V_INT(uzbl.behave.html_timeout,          1,   NULL)},
 -    { "keycmd",              PTR_V_STR(uzbl.state.keycmd,                 1,   set_keycmd)},
 -    { "status_message",      PTR_V_STR(uzbl.gui.sbar.msg,                 1,   update_title)},
 -    { "show_status",         PTR_V_INT(uzbl.behave.show_status,           1,   cmd_set_status)},
 -    { "status_top",          PTR_V_INT(uzbl.behave.status_top,            1,   move_statusbar)},
 -    { "status_format",       PTR_V_STR(uzbl.behave.status_format,         1,   update_title)},
 -    { "status_pbar_done",    PTR_V_STR(uzbl.gui.sbar.progress_s,          1,   update_title)},
 -    { "status_pbar_pending", PTR_V_STR(uzbl.gui.sbar.progress_u,          1,   update_title)},
 -    { "status_pbar_width",   PTR_V_INT(uzbl.gui.sbar.progress_w,          1,   update_title)},
 -    { "status_background",   PTR_V_STR(uzbl.behave.status_background,     1,   update_title)},
 -    { "insert_indicator",    PTR_V_STR(uzbl.behave.insert_indicator,      1,   update_indicator)},
 -    { "command_indicator",   PTR_V_STR(uzbl.behave.cmd_indicator,         1,   update_indicator)},
 -    { "title_format_long",   PTR_V_STR(uzbl.behave.title_format_long,     1,   update_title)},
 -    { "title_format_short",  PTR_V_STR(uzbl.behave.title_format_short,    1,   update_title)},
 -    { "icon",                PTR_V_STR(uzbl.gui.icon,                     1,   set_icon)},
 -    { "insert_mode",         PTR_V_INT(uzbl.behave.insert_mode,           1,   set_mode_indicator)},
 -    { "always_insert_mode",  PTR_V_INT(uzbl.behave.always_insert_mode,    1,   cmd_always_insert_mode)},
 -    { "reset_command_mode",  PTR_V_INT(uzbl.behave.reset_command_mode,    1,   NULL)},
 -    { "modkey",              PTR_V_STR(uzbl.behave.modkey,                1,   cmd_modkey)},
 -    { "load_finish_handler", PTR_V_STR(uzbl.behave.load_finish_handler,   1,   NULL)},
 -    { "load_start_handler",  PTR_V_STR(uzbl.behave.load_start_handler,    1,   NULL)},
 -    { "load_commit_handler", PTR_V_STR(uzbl.behave.load_commit_handler,   1,   NULL)},
 -    { "history_handler",     PTR_V_STR(uzbl.behave.history_handler,       1,   NULL)},
 -    { "download_handler",    PTR_V_STR(uzbl.behave.download_handler,      1,   NULL)},
 -    { "cookie_handler",      PTR_V_STR(uzbl.behave.cookie_handler,        1,   cmd_cookie_handler)},
 -    { "new_window",          PTR_V_STR(uzbl.behave.new_window,            1,   cmd_new_window)},
 -    { "fifo_dir",            PTR_V_STR(uzbl.behave.fifo_dir,              1,   cmd_fifo_dir)},
 -    { "socket_dir",          PTR_V_STR(uzbl.behave.socket_dir,            1,   cmd_socket_dir)},
 -    { "http_debug",          PTR_V_INT(uzbl.behave.http_debug,            1,   cmd_http_debug)},
 -    { "shell_cmd",           PTR_V_STR(uzbl.behave.shell_cmd,             1,   NULL)},
 -    { "proxy_url",           PTR_V_STR(uzbl.net.proxy_url,                1,   set_proxy_url)},
 -    { "max_conns",           PTR_V_INT(uzbl.net.max_conns,                1,   cmd_max_conns)},
 -    { "max_conns_host",      PTR_V_INT(uzbl.net.max_conns_host,           1,   cmd_max_conns_host)},
 -    { "useragent",           PTR_V_STR(uzbl.net.useragent,                1,   cmd_useragent)},
 +    { "uri",                    PTR_V_STR(uzbl.state.uri,                       1,   cmd_load_uri)},
 +    { "verbose",                PTR_V_INT(uzbl.state.verbose,                   1,   NULL)},
 +    { "mode",                   PTR_V_INT(uzbl.behave.mode,                     0,   NULL)},
 +    { "inject_html",            PTR_V_STR(uzbl.behave.inject_html,              0,   cmd_inject_html)},
 +    { "base_url",               PTR_V_STR(uzbl.behave.base_url,                 1,   NULL)},
 +    { "html_endmarker",         PTR_V_STR(uzbl.behave.html_endmarker,           1,   NULL)},
 +    { "html_mode_timeout",      PTR_V_INT(uzbl.behave.html_timeout,             1,   NULL)},
 +    { "keycmd",                 PTR_V_STR(uzbl.state.keycmd,                    1,   set_keycmd)},
 +    { "status_message",         PTR_V_STR(uzbl.gui.sbar.msg,                    1,   update_title)},
 +    { "show_status",            PTR_V_INT(uzbl.behave.show_status,              1,   cmd_set_status)},
 +    { "status_top",             PTR_V_INT(uzbl.behave.status_top,               1,   move_statusbar)},
 +    { "status_format",          PTR_V_STR(uzbl.behave.status_format,            1,   update_title)},
 +    { "status_pbar_done",       PTR_V_STR(uzbl.gui.sbar.progress_s,             1,   update_title)},
 +    { "status_pbar_pending",    PTR_V_STR(uzbl.gui.sbar.progress_u,             1,   update_title)},
 +    { "status_pbar_width",      PTR_V_INT(uzbl.gui.sbar.progress_w,             1,   update_title)},
 +    { "status_background",      PTR_V_STR(uzbl.behave.status_background,        1,   update_title)},
 +    { "insert_indicator",       PTR_V_STR(uzbl.behave.insert_indicator,         1,   update_indicator)},
 +    { "command_indicator",      PTR_V_STR(uzbl.behave.cmd_indicator,            1,   update_indicator)},
 +    { "title_format_long",      PTR_V_STR(uzbl.behave.title_format_long,        1,   update_title)},
 +    { "title_format_short",     PTR_V_STR(uzbl.behave.title_format_short,       1,   update_title)},
 +    { "icon",                   PTR_V_STR(uzbl.gui.icon,                        1,   set_icon)},
 +    { "insert_mode",            PTR_V_INT(uzbl.behave.insert_mode,              1,   set_mode_indicator)},
 +    { "always_insert_mode",     PTR_V_INT(uzbl.behave.always_insert_mode,       1,   cmd_always_insert_mode)},
 +    { "reset_command_mode",     PTR_V_INT(uzbl.behave.reset_command_mode,       1,   NULL)},
 +    { "modkey",                 PTR_V_STR(uzbl.behave.modkey,                   1,   cmd_modkey)},
 +    { "load_finish_handler",    PTR_V_STR(uzbl.behave.load_finish_handler,      1,   NULL)},
 +    { "load_start_handler",     PTR_V_STR(uzbl.behave.load_start_handler,       1,   NULL)},
 +    { "load_commit_handler",    PTR_V_STR(uzbl.behave.load_commit_handler,      1,   NULL)},
 +    { "history_handler",        PTR_V_STR(uzbl.behave.history_handler,          1,   NULL)},
 +    { "download_handler",       PTR_V_STR(uzbl.behave.download_handler,         1,   NULL)},
 +    { "cookie_handler",         PTR_V_STR(uzbl.behave.cookie_handler,           1,   cmd_cookie_handler)},
 +    { "new_window",             PTR_V_STR(uzbl.behave.new_window,               1,   cmd_new_window)},
 +    { "fifo_dir",               PTR_V_STR(uzbl.behave.fifo_dir,                 1,   cmd_fifo_dir)},
 +    { "socket_dir",             PTR_V_STR(uzbl.behave.socket_dir,               1,   cmd_socket_dir)},
 +    { "http_debug",             PTR_V_INT(uzbl.behave.http_debug,               1,   cmd_http_debug)},
 +    { "shell_cmd",              PTR_V_STR(uzbl.behave.shell_cmd,                1,   NULL)},
 +    { "proxy_url",              PTR_V_STR(uzbl.net.proxy_url,                   1,   set_proxy_url)},
 +    { "max_conns",              PTR_V_INT(uzbl.net.max_conns,                   1,   cmd_max_conns)},
 +    { "max_conns_host",         PTR_V_INT(uzbl.net.max_conns_host,              1,   cmd_max_conns_host)},
 +    { "useragent",              PTR_V_STR(uzbl.net.useragent,                   1,   cmd_useragent)},
  
      /* exported WebKitWebSettings properties */
 -    { "zoom_level",          PTR_V_FLOAT(uzbl.behave.zoom_level,          1,   cmd_zoom_level)},
 -    { "font_size",           PTR_V_INT(uzbl.behave.font_size,             1,   cmd_font_size)},
 -    { "monospace_size",      PTR_V_INT(uzbl.behave.monospace_size,        1,   cmd_font_size)},
 -    { "minimum_font_size",   PTR_V_INT(uzbl.behave.minimum_font_size,     1,   cmd_minimum_font_size)},
 -    { "disable_plugins",     PTR_V_INT(uzbl.behave.disable_plugins,       1,   cmd_disable_plugins)},
 -    { "disable_scripts",     PTR_V_INT(uzbl.behave.disable_scripts,       1,   cmd_disable_scripts)},
 -    { "autoload_images",     PTR_V_INT(uzbl.behave.autoload_img,          1,   cmd_autoload_img)},
 -    { "autoshrink_images",   PTR_V_INT(uzbl.behave.autoshrink_img,        1,   cmd_autoshrink_img)},
 -    { "enable_spellcheck",   PTR_V_INT(uzbl.behave.enable_spellcheck,     1,   cmd_enable_spellcheck)},
 -    { "enable_private",      PTR_V_INT(uzbl.behave.enable_private,        1,   cmd_enable_private)},
 -    { "print_backgrounds",   PTR_V_INT(uzbl.behave.print_bg,              1,   cmd_print_bg)},
 -    { "stylesheet_uri",      PTR_V_STR(uzbl.behave.style_uri,             1,   cmd_style_uri)},
 -    { "resizable_text_areas",PTR_V_INT(uzbl.behave.resizable_txt,         1,   cmd_resizable_txt)},
 -    { "default_encoding",    PTR_V_STR(uzbl.behave.default_encoding,      1,   cmd_default_encoding)},
 -    { "enforce_96_dpi",      PTR_V_INT(uzbl.behave.enforce_96dpi,         1,   cmd_enforce_96dpi)},
 -    { "caret_browsing",      PTR_V_INT(uzbl.behave.caret_browsing,        1,   cmd_caret_browsing)},
 +    { "zoom_level",             PTR_V_FLOAT(uzbl.behave.zoom_level,             1,   cmd_zoom_level)},
 +    { "font_size",              PTR_V_INT(uzbl.behave.font_size,                1,   cmd_font_size)},
 +    { "default_font_family",    PTR_V_STR(uzbl.behave.default_font_family,      1,   cmd_default_font_family)},
 +    { "monospace_font_family",  PTR_V_STR(uzbl.behave.monospace_font_family,    1,   cmd_monospace_font_family)},
 +    { "cursive_font_family",    PTR_V_STR(uzbl.behave.cursive_font_family,      1,   cmd_cursive_font_family)},
 +    { "sans_serif_font_family", PTR_V_STR(uzbl.behave.sans_serif_font_family,   1,   cmd_sans_serif_font_family)},
 +    { "serif_font_family",      PTR_V_STR(uzbl.behave.serif_font_family,        1,   cmd_serif_font_family)},
 +    { "fantasy_font_family",    PTR_V_STR(uzbl.behave.fantasy_font_family,      1,   cmd_fantasy_font_family)},
 +    { "monospace_size",         PTR_V_INT(uzbl.behave.monospace_size,           1,   cmd_font_size)},
 +    { "minimum_font_size",      PTR_V_INT(uzbl.behave.minimum_font_size,        1,   cmd_minimum_font_size)},
 +    { "disable_plugins",        PTR_V_INT(uzbl.behave.disable_plugins,          1,   cmd_disable_plugins)},
 +    { "disable_scripts",        PTR_V_INT(uzbl.behave.disable_scripts,          1,   cmd_disable_scripts)},
 +    { "autoload_images",        PTR_V_INT(uzbl.behave.autoload_img,             1,   cmd_autoload_img)},
 +    { "autoshrink_images",      PTR_V_INT(uzbl.behave.autoshrink_img,           1,   cmd_autoshrink_img)},
 +    { "enable_spellcheck",      PTR_V_INT(uzbl.behave.enable_spellcheck,        1,   cmd_enable_spellcheck)},
 +    { "enable_private",         PTR_V_INT(uzbl.behave.enable_private,           1,   cmd_enable_private)},
 +    { "print_backgrounds",      PTR_V_INT(uzbl.behave.print_bg,                 1,   cmd_print_bg)},
 +    { "stylesheet_uri",         PTR_V_STR(uzbl.behave.style_uri,                1,   cmd_style_uri)},
 +    { "resizable_text_areas",   PTR_V_INT(uzbl.behave.resizable_txt,            1,   cmd_resizable_txt)},
 +    { "default_encoding",       PTR_V_STR(uzbl.behave.default_encoding,         1,   cmd_default_encoding)},
 +    { "enforce_96_dpi",         PTR_V_INT(uzbl.behave.enforce_96dpi,            1,   cmd_enforce_96dpi)},
 +    { "caret_browsing",         PTR_V_INT(uzbl.behave.caret_browsing,           1,   cmd_caret_browsing)},
  
    /* constants (not dumpable or writeable) */
 -    { "WEBKIT_MAJOR",        PTR_C_INT(uzbl.info.webkit_major,                 NULL)},
 -    { "WEBKIT_MINOR",        PTR_C_INT(uzbl.info.webkit_minor,                 NULL)},
 -    { "WEBKIT_MICRO",        PTR_C_INT(uzbl.info.webkit_micro,                 NULL)},
 -    { "ARCH_UZBL",           PTR_C_STR(uzbl.info.arch,                         NULL)},
 -    { "COMMIT",              PTR_C_STR(uzbl.info.commit,                       NULL)},
 -    { "LOAD_PROGRESS",       PTR_C_INT(uzbl.gui.sbar.load_progress,            NULL)},
 -    { "LOAD_PROGRESSBAR",    PTR_C_STR(uzbl.gui.sbar.progress_bar,             NULL)},
 -    { "TITLE",               PTR_C_STR(uzbl.gui.main_title,                    NULL)},
 -    { "SELECTED_URI",        PTR_C_STR(uzbl.state.selected_url,                NULL)},
 -    { "MODE",                PTR_C_STR(uzbl.gui.sbar.mode_indicator,           NULL)},
 -    { "NAME",                PTR_C_STR(uzbl.state.instance_name,               NULL)},
 -
 -    { NULL,                  {.ptr.s = NULL, .type = TYPE_INT, .dump = 0, .writeable = 0, .func = NULL}}
 +    { "WEBKIT_MAJOR",           PTR_C_INT(uzbl.info.webkit_major,                    NULL)},
 +    { "WEBKIT_MINOR",           PTR_C_INT(uzbl.info.webkit_minor,                    NULL)},
 +    { "WEBKIT_MICRO",           PTR_C_INT(uzbl.info.webkit_micro,                    NULL)},
 +    { "ARCH_UZBL",              PTR_C_STR(uzbl.info.arch,                            NULL)},
 +    { "COMMIT",                 PTR_C_STR(uzbl.info.commit,                          NULL)},
 +    { "LOAD_PROGRESS",          PTR_C_INT(uzbl.gui.sbar.load_progress,               NULL)},
 +    { "LOAD_PROGRESSBAR",       PTR_C_STR(uzbl.gui.sbar.progress_bar,                NULL)},
 +    { "TITLE",                  PTR_C_STR(uzbl.gui.main_title,                       NULL)},
 +    { "SELECTED_URI",           PTR_C_STR(uzbl.state.selected_url,                   NULL)},
 +    { "MODE",                   PTR_C_STR(uzbl.gui.sbar.mode_indicator,              NULL)},
 +    { "NAME",                   PTR_C_STR(uzbl.state.instance_name,                  NULL)},
 +
 +    { NULL,                     {.ptr.s = NULL, .type = TYPE_INT, .dump = 0, .writeable = 0, .func = NULL}}
- }, *n2v_p = var_name_to_ptr;
+ };
  
  
  const struct {
  /* construct a hash from the var_name_to_ptr array for quick access */
  void
  make_var_to_name_hash() {
+     struct var_name_to_ptr_t *n2v_p = &var_name_to_ptr;
      uzbl.comm.proto_var = g_hash_table_new(g_str_hash, g_str_equal);
      while(n2v_p->name) {
          g_hash_table_insert(uzbl.comm.proto_var, n2v_p->name, (gpointer) &n2v_p->cp);
  }
  
  /* --- UTILITY FUNCTIONS --- */
- enum {EXP_ERR, EXP_SIMPLE_VAR, EXP_BRACED_VAR, EXP_EXPR, EXP_JS, EXP_ESCAPE};
- guint
+ enum exp_type {EXP_ERR, EXP_SIMPLE_VAR, EXP_BRACED_VAR, EXP_EXPR, EXP_JS, EXP_ESCAPE};
+ enum exp_type
  get_exp_type(const gchar *s) {
      /* variables */
      if(*(s+1) == '(')
@@@ -257,7 -249,7 +258,7 @@@ return EXP_ERR
  gchar *
  expand(const char *s, guint recurse) {
      uzbl_cmdprop *c;
-     guint etype;
+     enum exp_type etype;
      char *end_simple_var = "^°!\"§$%&/()=?'`'+~*'#-.:,;@<>| \\{}[]¹²³¼½";
      char *ret = NULL;
      char *vend = NULL;
@@@ -412,7 -404,7 +413,7 @@@ str_replace (const char* search, const 
  }
  
  GArray*
- read_file_by_line (gchar *path) {
+ read_file_by_line (const gchar *path) {
      GIOChannel *chan = NULL;
      gchar *readbuf = NULL;
      gsize len;
@@@ -561,7 -553,7 +562,7 @@@ mime_policy_cb(WebKitWebView *web_view
      return TRUE;
  }
  
- WebKitWebView*
+ /*@null@*/ WebKitWebView*
  create_web_view_cb (WebKitWebView  *web_view, WebKitWebFrame *frame, gpointer user_data) {
      (void) web_view;
      (void) frame;
@@@ -797,7 -789,7 +798,7 @@@ VIEWFUNC(go_forward
  #undef VIEWFUNC
  
  /* -- command to callback/function map for things we cannot attach to any signals */
- struct {char *key; CommandInfo value;} cmdlist[] =
+ struct {const char *key; CommandInfo value;} cmdlist[] =
  {   /* key                   function      no_split      */
      { "back",               {view_go_back, 0}              },
      { "forward",            {view_go_forward, 0}           },
      { "sync_spawn",         {spawn_sync, 0}                }, // needed for cookie handler
      { "sh",                 {spawn_sh, 0}                  },
      { "sync_sh",            {spawn_sh_sync, 0}             }, // needed for cookie handler
 +    { "talk_to_socket",     {talk_to_socket, 0}            },
      { "exit",               {close_uzbl, 0}                },
      { "search",             {search_forward_text, TRUE}    },
      { "search_reverse",     {search_reverse_text, TRUE}    },
@@@ -1308,7 -1299,7 +1309,7 @@@ run_command (const gchar *command, cons
      return result;
  }
  
- gchar**
+ /*@null@*/ gchar**
  split_quoted(const gchar* src, const gboolean unquote) {
      /* split on unquoted space, return array of strings;
         remove a layer of quotes and backslashes if unquote */
@@@ -1406,118 -1397,6 +1407,118 @@@ spawn_sh_sync(WebKitWebView *web_view, 
  }
  
  void
 +talk_to_socket(WebKitWebView *web_view, GArray *argv, GString *result) {
 +    (void)web_view; (void)result;
 +
 +    int fd, len;
 +    struct sockaddr_un sa;
 +    char* sockpath;
 +    ssize_t ret;
 +    struct pollfd pfd;
 +    struct iovec* iov;
 +    guint i;
 +
 +    if(uzbl.comm.sync_stdout) uzbl.comm.sync_stdout = strfree(uzbl.comm.sync_stdout);
 +
 +    /* This function could be optimised by storing a hash table of socket paths
 +       and associated connected file descriptors rather than closing and
 +       re-opening for every call. Also we could launch a script if socket connect
 +       fails. */
 +
 +    /* First element argv[0] is path to socket. Following elements are tokens to
 +       write to the socket. We write them as a single packet with each token
 +       separated by an ASCII nul (\0). */
 +    if(argv->len < 2) {
 +        g_printerr("talk_to_socket called with only %d args (need at least two).\n",
 +            (int)argv->len);
 +        return;
 +    }
 +
 +    /* copy socket path, null terminate result */
 +    sockpath = g_array_index(argv, char*, 0);
 +    g_strlcpy(sa.sun_path, sockpath, sizeof(sa.sun_path));
 +    sa.sun_family = AF_UNIX;
 +
 +    /* create socket file descriptor and connect it to path */
 +    fd = socket(AF_UNIX, SOCK_SEQPACKET, 0);
 +    if(fd == -1) {
 +        g_printerr("talk_to_socket: creating socket failed (%s)\n", strerror(errno));
 +        return;
 +    }
 +    if(connect(fd, (struct sockaddr*)&sa, sizeof(sa))) {
 +        g_printerr("talk_to_socket: connect failed (%s)\n", strerror(errno));
 +        close(fd);
 +        return;
 +    }
 +
 +    /* build request vector */
 +    iov = g_malloc(sizeof(struct iovec) * (argv->len - 1));
 +    if(!iov) {
 +        g_printerr("talk_to_socket: unable to allocated memory for token vector\n");
 +        close(fd);
 +        return;
 +    }
 +    for(i = 1; i < argv->len; ++i) {
 +        iov[i - 1].iov_base = g_array_index(argv, char*, i);
 +        iov[i - 1].iov_len = strlen(iov[i - 1].iov_base) + 1; /* string plus \0 */
 +    }
 +
 +    /* write request */
 +    ret = writev(fd, iov, argv->len - 1);
 +    g_free(iov);
 +    if(ret == -1) {
 +        g_printerr("talk_to_socket: write failed (%s)\n", strerror(errno));
 +        close(fd);
 +        return;
 +    }
 +
 +    /* wait for a response, with a 500ms timeout */
 +    pfd.fd = fd;
 +    pfd.events = POLLIN;
 +    while(1) {
 +        ret = poll(&pfd, 1, 500);
 +        if(ret == 1) break;
 +        if(ret == 0) errno = ETIMEDOUT;
 +        if(errno == EINTR) continue;
 +        g_printerr("talk_to_socket: poll failed while waiting for input (%s)\n",
 +            strerror(errno));
 +        close(fd);
 +        return;
 +    }
 +
 +    /* get length of response */
 +    if(ioctl(fd, FIONREAD, &len) == -1) {
 +        g_printerr("talk_to_socket: cannot find daemon response length, "
 +            "ioctl failed (%s)\n", strerror(errno));
 +        close(fd);
 +        return;
 +    }
 +
 +    /* if there is a response, read it */
 +    if(len) {
 +        uzbl.comm.sync_stdout = g_malloc(len + 1);
 +        if(!uzbl.comm.sync_stdout) {
 +            g_printerr("talk_to_socket: failed to allocate %d bytes\n", len);
 +            close(fd);
 +            return;
 +        }
 +        uzbl.comm.sync_stdout[len] = 0; /* ensure result is null terminated */
 +
 +        ret = read(fd, uzbl.comm.sync_stdout, len);
 +        if(ret == -1) {
 +            g_printerr("talk_to_socket: failed to read from socket (%s)\n",
 +                strerror(errno));
 +            close(fd);
 +            return;
 +        }
 +    }
 +
 +    /* clean up */
 +    close(fd);
 +    return;
 +}
 +
 +void
  parse_command(const char *cmd, const char *param, GString *result) {
      CommandInfo *c;
  
@@@ -1555,8 -1434,7 +1556,7 @@@ voi
  set_proxy_url() {
      SoupURI *suri;
  
-     if(*uzbl.net.proxy_url == ' '
-        || uzbl.net.proxy_url == NULL) {
+     if(uzbl.net.proxy_url == NULL || *uzbl.net.proxy_url == ' ') {
          soup_session_remove_feature_by_type(uzbl.net.soup_session,
                  (GType) SOUP_SESSION_PROXY_URI);
      }
@@@ -1640,42 -1518,6 +1640,42 @@@ cmd_font_size() 
  }
  
  void
 +cmd_default_font_family() {
 +    g_object_set (G_OBJECT(view_settings()), "default-font-family",
 +            uzbl.behave.default_font_family, NULL);
 +}
 +
 +void
 +cmd_monospace_font_family() {
 +    g_object_set (G_OBJECT(view_settings()), "monospace-font-family",
 +            uzbl.behave.monospace_font_family, NULL);
 +}
 +
 +void
 +cmd_sans_serif_font_family() {
 +    g_object_set (G_OBJECT(view_settings()), "sans_serif-font-family",
 +            uzbl.behave.sans_serif_font_family, NULL);
 +}
 +
 +void
 +cmd_serif_font_family() {
 +    g_object_set (G_OBJECT(view_settings()), "serif-font-family",
 +            uzbl.behave.serif_font_family, NULL);
 +}
 +
 +void
 +cmd_cursive_font_family() {
 +    g_object_set (G_OBJECT(view_settings()), "cursive-font-family",
 +            uzbl.behave.cursive_font_family, NULL);
 +}
 +
 +void
 +cmd_fantasy_font_family() {
 +    g_object_set (G_OBJECT(view_settings()), "fantasy-font-family",
 +            uzbl.behave.fantasy_font_family, NULL);
 +}
 +
 +void
  cmd_zoom_level() {
      webkit_web_view_set_zoom_level (uzbl.gui.web_view, uzbl.behave.zoom_level);
  }
@@@ -1854,7 -1696,7 +1854,7 @@@ move_statusbar() 
  }
  
  gboolean
- set_var_value(gchar *name, gchar *val) {
+ set_var_value(const gchar *name, gchar *val) {
      uzbl_cmdprop *c = NULL;
      char *endp = NULL;
      char *buf = NULL;
@@@ -1936,7 -1778,7 +1936,7 @@@ parse_cmd_line(const char *ctl_line, GS
      }
  }
  
- gchar*
+ /*@null@*/ gchar*
  build_stream_name(int type, const gchar* dir) {
      State *s = &uzbl.state;
      gchar *str = NULL;
@@@ -1977,7 -1819,7 +1977,7 @@@ control_fifo(GIOChannel *gio, GIOCondit
      return TRUE;
  }
  
- gchar*
+ /*@null@*/ gchar*
  init_fifo(gchar *dir) { /* return dir or, on error, free dir and return NULL */
      if (uzbl.comm.fifo_path) { /* get rid of the old fifo if one exists */
          if (unlink(uzbl.comm.fifo_path) == -1)
@@@ -2101,7 -1943,7 +2101,7 @@@ control_client_socket(GIOChannel *clien
      return TRUE;
  }
  
- gchar*
+ /*@null@*/ gchar*
  init_socket(gchar *dir) { /* return dir or, on error, free dir and return NULL */
      if (uzbl.comm.socket_path) { /* remove an existing socket should one exist */
          if (unlink(uzbl.comm.socket_path) == -1)
@@@ -2326,11 -2168,16 +2326,11 @@@ exec_paramcmd(const Action *act, const 
  }
  
  
 -GtkWidget*
 +void
  create_browser () {
      GUI *g = &uzbl.gui;
  
 -    GtkWidget* scrolled_window = gtk_scrolled_window_new (NULL, NULL);
 -    //main_window_ref = g_object_ref(scrolled_window);
 -    gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scrolled_window), GTK_POLICY_NEVER, GTK_POLICY_NEVER); //todo: some sort of display of position/total length. like what emacs does
 -
      g->web_view = WEBKIT_WEB_VIEW (webkit_web_view_new ());
 -    gtk_container_add (GTK_CONTAINER (scrolled_window), GTK_WIDGET (g->web_view));
  
      g_signal_connect (G_OBJECT (g->web_view), "notify::title", G_CALLBACK (title_change_cb), NULL);
      g_signal_connect (G_OBJECT (g->web_view), "load-progress-changed", G_CALLBACK (progress_change_cb), g->web_view);
      g_signal_connect (G_OBJECT (g->web_view), "download-requested", G_CALLBACK (download_cb), g->web_view);
      g_signal_connect (G_OBJECT (g->web_view), "create-web-view", G_CALLBACK (create_web_view_cb), g->web_view);
      g_signal_connect (G_OBJECT (g->web_view), "mime-type-policy-decision-requested", G_CALLBACK (mime_policy_cb), g->web_view);
 -
 -    return scrolled_window;
  }
  
  GtkWidget*
@@@ -2409,8 -2258,7 +2409,8 @@@ inject_handler_args(const gchar *actnam
      if ((g_strcmp0(actname, "spawn") == 0) ||
          (g_strcmp0(actname, "sh") == 0) ||
          (g_strcmp0(actname, "sync_spawn") == 0) ||
 -        (g_strcmp0(actname, "sync_sh") == 0)) {
 +        (g_strcmp0(actname, "sync_sh") == 0) ||
 +        (g_strcmp0(actname, "talk_to_socket") == 0)) {
          guint i;
          GString *a = g_string_new("");
          gchar **spawnparts = split_quoted(origargs, FALSE);
@@@ -2508,7 -2356,7 +2508,7 @@@ add_binding (const gchar *key, const gc
      g_strfreev(parts);
  }
  
- gchar*
+ /*@null@*/ gchar*
  get_xdg_var (XDG_Var xdg) {
      const gchar* actual_value = getenv (xdg.environmental);
      const gchar* home         = getenv ("HOME");
      return return_value;
  }
  
- gchar*
- find_xdg_file (int xdg_type, char* filename) {
+ /*@null@*/ gchar*
+ find_xdg_file (int xdg_type, const char* filename) {
      /* xdg_type = 0 => config
         xdg_type = 1 => data
         xdg_type = 2 => cache*/
      if (file_exists (temporary_file)) {
          return temporary_file;
      } else {
+         g_free(temporary_file);
          return NULL;
      }
  }
@@@ -2779,6 -2628,7 +2780,6 @@@ retreive_geometry() 
   * external applications need to do anyhow */
  void
  initialize(int argc, char *argv[]) {
 -    gtk_init (&argc, &argv);
      if (!g_thread_supported ())
          g_thread_init (NULL);
      uzbl.state.executable_path = g_strdup(argv[0]);
      commands_hash ();
      make_var_to_name_hash();
  
 -    uzbl.gui.scrolled_win = create_browser();
 +    create_browser();
  }
  
  #ifndef UZBL_LIBRARY
  main (int argc, char* argv[]) {
      initialize(argc, argv);
  
 +    gtk_init (&argc, &argv);
 +
 +    uzbl.gui.scrolled_win = gtk_scrolled_window_new (NULL, NULL);
 +    //main_window_ref = g_object_ref(scrolled_window);
 +    gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (uzbl.gui.scrolled_win),
 +        GTK_POLICY_NEVER, GTK_POLICY_NEVER); //todo: some sort of display of position/total length. like what emacs does
 +
 +    gtk_container_add (GTK_CONTAINER (uzbl.gui.scrolled_win),
 +        GTK_WIDGET (uzbl.gui.web_view));
 +
      uzbl.gui.vbox = gtk_vbox_new (FALSE, 0);
  
      create_mainbar();
diff --combined uzbl.h
--- 1/uzbl.h
--- 2/uzbl.h
+++ b/uzbl.h
@@@ -116,12 -116,6 +116,12 @@@ typedef struct 
      gchar*   download_handler;
      gchar*   cookie_handler;
      gchar*   new_window;
 +    gchar*   default_font_family;
 +    gchar*   monospace_font_family;
 +    gchar*   sans_serif_font_family;
 +    gchar*   serif_font_family;
 +    gchar*   fantasy_font_family;
 +    gchar*   cursive_font_family;
      gboolean always_insert_mode;
      gboolean show_status;
      gboolean insert_mode;
@@@ -226,7 -220,7 +226,7 @@@ char 
  str_replace (const char* search, const char* replace, const char* string);
  
  GArray*
- read_file_by_line (gchar *path);
+ read_file_by_line (const gchar *path);
  
  gchar*
  parseenv (char* string);
@@@ -241,7 -235,7 +241,7 @@@ sigfunc 
  setup_signal(int signe, sigfunc *shandler);
  
  gboolean
- set_var_value(gchar *name, gchar *val);
+ set_var_value(const gchar *name, gchar *val);
  
  void
  print(WebKitWebView *page, GArray *argv, GString *result);
@@@ -252,7 -246,7 +252,7 @@@ new_window_cb (WebKitWebView *web_view
  gboolean
  mime_policy_cb(WebKitWebView *web_view, WebKitWebFrame *frame, WebKitNetworkRequest *request, gchar *mime_type,  WebKitWebPolicyDecision *policy_decision, gpointer user_data);
  
- WebKitWebView*
+ /*@null@*/ WebKitWebView*
  create_web_view_cb (WebKitWebView  *web_view, WebKitWebFrame *frame, gpointer user_data);
  
  gboolean
@@@ -344,9 -338,6 +344,9 @@@ char
  build_progressbar_ascii(int percent);
  
  void
 +talk_to_socket(WebKitWebView *web_view, GArray *argv, GString *result);
 +
 +void
  spawn(WebKitWebView *web_view, GArray *argv, GString *result);
  
  void
@@@ -364,13 -355,13 +364,13 @@@ parse_command(const char *cmd, const ch
  void
  parse_cmd_line(const char *ctl_line, GString *result);
  
- gchar*
+ /*@null@*/ gchar*
  build_stream_name(int type, const gchar *dir);
  
  gboolean
  control_fifo(GIOChannel *gio, GIOCondition condition);
  
- gchar*
+ /*@null@*/ gchar*
  init_fifo(gchar *dir);
  
  gboolean
@@@ -379,7 -370,7 +379,7 @@@ control_stdin(GIOChannel *gio, GIOCondi
  void
  create_stdin();
  
- gchar*
+ /*@null@*/ gchar*
  init_socket(gchar *dir);
  
  gboolean
@@@ -403,7 -394,7 +403,7 @@@ exec_paramcmd(const Action* act, const 
  void
  initialize (int argc, char *argv[]);
  
 -GtkWidget*
 +void
  create_browser ();
  
  GtkWidget*
@@@ -421,11 -412,11 +421,11 @@@ run_handler (const gchar *act, const gc
  void
  add_binding (const gchar *key, const gchar *act);
  
- gchar*
+ /*@null@*/ gchar*
  get_xdg_var (XDG_Var xdg);
  
- gchar*
- find_xdg_file (int xdg_type, char* filename);
+ /*@null@*/ gchar*
+ find_xdg_file (int xdg_type, const char* filename);
  
  void
  settings_init ();
@@@ -534,24 -525,6 +534,24 @@@ voi
  cmd_font_size();
  
  void
 +cmd_default_font_family();
 +
 +void
 +cmd_monospace_font_family();
 +
 +void
 +cmd_sans_serif_font_family();
 +
 +void
 +cmd_serif_font_family();
 +
 +void
 +cmd_cursive_font_family();
 +
 +void
 +cmd_fantasy_font_family();
 +
 +void
  cmd_zoom_level();
  
  void