📄 main.c
字号:
/* main.c * main() * (c) 2002 Mikulas Patocka * This file is a part of the Links program, released under GPL. */#include "links.h"int retval = RET_OK;void unhandle_basic_signals(struct terminal *);void sig_terminate(struct terminal *);void sig_intr(struct terminal *);void sig_ctrl_c(struct terminal *);void poll_fg(void *);void handle_basic_signals(struct terminal *);void end_dump(struct object_request *, void *);void init(void);void terminate_all_subsystems(void);void sig_terminate(struct terminal *t){ unhandle_basic_signals(t); terminate_loop = 1; retval = RET_SIGNAL;}void sig_intr(struct terminal *t){ if (!t) { unhandle_basic_signals(t); terminate_loop = 1; } else { unhandle_basic_signals(t); exit_prog(t, NULL, NULL); }}void sig_ctrl_c(struct terminal *t){ if (!is_blocked()) kbd_ctrl_c();}void sig_ign(void *x){}int fg_poll_timer = -1;void sig_tstp(struct terminal *t){#ifdef SIGSTOP#if defined (SIGCONT) && defined(SIGTTOU) && defined(HAVE_GETPID) pid_t pid = getpid(); pid_t newpid;#endif if (!F) { block_itrm(1); }#ifdef G else { drv->block(NULL); }#endif#if defined (SIGCONT) && defined(SIGTTOU) && defined(HAVE_GETPID) if (!(newpid = fork())) { while (1) { sleep(1); kill(pid, SIGCONT); } }#endif raise(SIGSTOP);#if defined (SIGCONT) && defined(SIGTTOU) && defined(HAVE_GETPID) if (newpid != -1) kill(newpid, SIGKILL);#endif#endif if (fg_poll_timer != -1) kill_timer(fg_poll_timer); fg_poll_timer = install_timer(FG_POLL_TIME, poll_fg, t);}void poll_fg(void *t){ int r; fg_poll_timer = -1; if (!F) { r = unblock_itrm(1);#ifdef G } else { r = drv->unblock(NULL);#endif } if (r == -1) { fg_poll_timer = install_timer(FG_POLL_TIME, poll_fg, t); } if (r == -2) { /* This will unblock externally spawned viewer, if it exists */#ifdef SIGCONT kill(0, SIGCONT);#endif }}void sig_cont(struct terminal *t){ if (!F) { unblock_itrm(1);#ifdef G } else { drv->unblock(NULL);#endif } /*else register_bottom_half(raise, SIGSTOP);*/}void handle_basic_signals(struct terminal *term){ install_signal_handler(SIGHUP, (void (*)(void *))sig_intr, term, 0); if (!F) install_signal_handler(SIGINT, (void (*)(void *))sig_ctrl_c, term, 0); /*install_signal_handler(SIGTERM, (void (*)(void *))sig_terminate, term, 0);*/#ifdef SIGTSTP if (!F) install_signal_handler(SIGTSTP, (void (*)(void *))sig_tstp, term, 0);#endif#ifdef SIGTTIN if (!F) install_signal_handler(SIGTTIN, (void (*)(void *))sig_tstp, term, 0);#endif#ifdef SIGTTOU install_signal_handler(SIGTTOU, (void (*)(void *))sig_ign, term, 0);#endif#ifdef SIGCONT if (!F) install_signal_handler(SIGCONT, (void (*)(void *))sig_cont, term, 0);#endif}void unhandle_terminal_signals(struct terminal *term){ install_signal_handler(SIGHUP, NULL, NULL, 0); if (!F) install_signal_handler(SIGINT, NULL, NULL, 0);#ifdef SIGTSTP install_signal_handler(SIGTSTP, NULL, NULL, 0);#endif#ifdef SIGTTIN install_signal_handler(SIGTTIN, NULL, NULL, 0);#endif#ifdef SIGTTOU install_signal_handler(SIGTTOU, NULL, NULL, 0);#endif#ifdef SIGCONT install_signal_handler(SIGCONT, NULL, NULL, 0);#endif if (fg_poll_timer != -1) kill_timer(fg_poll_timer), fg_poll_timer = -1;}void unhandle_basic_signals(struct terminal *term){ install_signal_handler(SIGHUP, NULL, NULL, 0); if (!F) install_signal_handler(SIGINT, NULL, NULL, 0); /*install_signal_handler(SIGTERM, NULL, NULL, 0);*/#ifdef SIGTSTP install_signal_handler(SIGTSTP, NULL, NULL, 0);#endif#ifdef SIGTTIN install_signal_handler(SIGTTIN, NULL, NULL, 0);#endif#ifdef SIGTTOU install_signal_handler(SIGTTOU, NULL, NULL, 0);#endif#ifdef SIGCONT install_signal_handler(SIGCONT, NULL, NULL, 0);#endif if (fg_poll_timer != -1) kill_timer(fg_poll_timer), fg_poll_timer = -1;}int terminal_pipe[2];int attach_terminal(int in, int out, int ctl, void *info, int len){ struct terminal *term; fcntl(terminal_pipe[0], F_SETFL, O_NONBLOCK); fcntl(terminal_pipe[1], F_SETFL, O_NONBLOCK); handle_trm(in, out, out, terminal_pipe[1], ctl, info, len); mem_free(info); if ((term = init_term(terminal_pipe[0], out, win_func))) { handle_basic_signals(term); /* OK, this is race condition, but it must be so; GPM installs it's own buggy TSTP handler */ return terminal_pipe[1]; } close(terminal_pipe[0]); close(terminal_pipe[1]); return -1;}#ifdef Gint attach_g_terminal(void *info, int len){ struct terminal *term; term = init_gfx_term(win_func, info, len); mem_free(info); return term ? 0 : -1;}#endifstruct object_request *dump_obj;off_t dump_pos;void end_dump(struct object_request *r, void *p){ struct cache_entry *ce; int oh; if (!r->state || (r->state == 1 && dmp != D_SOURCE)) return; if ((oh = get_output_handle()) == -1) return; ce = r->ce; if (dmp == D_SOURCE) { if (ce) { struct fragment *frag; nextfrag: foreach(frag, ce->frag) if (frag->offset <= dump_pos && frag->offset + frag->length > dump_pos) { int l = frag->length - (dump_pos - frag->offset); int w = hard_write(oh, frag->data + dump_pos - frag->offset, l); if (w != l) { detach_object_connection(r, dump_pos); if (w < 0) fprintf(stderr, "Error writing to stdout: %s.\n", strerror(errno)); else fprintf(stderr, "Can't write to stdout.\n"); retval = RET_ERROR; goto terminate; } dump_pos += w; detach_object_connection(r, dump_pos); goto nextfrag; } } if (r->state >= 0) return; } else if (ce) { struct document_options o; struct f_data_c *fd; if (!(fd = create_f_data_c(NULL, NULL))) goto terminate; memset(&o, 0, sizeof(struct document_options)); o.xp = 0; o.yp = 1; o.xw = screen_width; o.yw = 25; o.col = 0; o.cp = dump_codepage == -1 ? 0 : dump_codepage; ds2do(&dds, &o); o.plain = 0; o.frames = 0; o.js_enable = 0; memcpy(&o.default_fg, &default_fg, sizeof(struct rgb)); memcpy(&o.default_bg, &default_bg, sizeof(struct rgb)); memcpy(&o.default_link, &default_link, sizeof(struct rgb)); memcpy(&o.default_vlink, &default_vlink, sizeof(struct rgb)); o.framename = ""; if (!(fd->f_data = cached_format_html(fd, r, r->url, &o, NULL))) goto term_1; dump_to_file(fd->f_data, oh); term_1: reinit_f_data_c(fd); mem_free(fd); } if (r->state != O_OK) { unsigned char *m = get_err_msg(r->stat.state); fprintf(stderr, "%s\n", get_english_translation(m)); retval = RET_ERROR; goto terminate; } terminate: terminate_loop = 1;}int g_argc;unsigned char **g_argv;unsigned char *path_to_exe;int init_b = 0;void initialize_all_subsystems(void);void initialize_all_subsystems_2(void);void init(void){ int uh; void *info; int len; unsigned char *u; initialize_all_subsystems(); utf8_table=get_cp_index("UTF-8");/* OS/2 has some stupid bug and the pipe must be created before socket :-/ */ if (c_pipe(terminal_pipe)) { error("ERROR: can't create pipe for internal communication"); retval = RET_FATAL; goto ttt; } if (!(u = parse_options(g_argc - 1, g_argv + 1))) { retval = RET_SYNTAX; goto ttt; } if (ggr_drv[0] || ggr_mode[0]) ggr = 1; if (dmp) ggr = 0; if (!ggr && !no_connect && (uh = bind_to_af_unix()) != -1) { close(terminal_pipe[0]); close(terminal_pipe[1]); if (!(info = create_session_info(base_session, u, default_target, &len))) { close(uh); retval = RET_FATAL; goto ttt; } initialize_all_subsystems_2(); handle_trm(get_input_handle(), get_output_handle(), uh, uh, get_ctl_handle(), info, len); handle_basic_signals(NULL); /* OK, this is race condition, but it must be so; GPM installs it's own buggy TSTP handler */ mem_free(info); return; } if ((dds.assume_cp = get_cp_index("ISO-8859-1")) == -1) dds.assume_cp = 0; load_config(); init_b = 1; init_bookmarks(); create_initial_extensions(); load_url_history(); init_cookies(); u = parse_options(g_argc - 1, g_argv + 1); if (!u) { ttt: initialize_all_subsystems_2(); tttt: terminate_loop = 1; return; } if (!dmp) { if (ggr) {#ifdef G unsigned char *r; if ((r = init_graphics(ggr_drv, ggr_mode, ggr_display))) { fprintf(stderr, "%s", r); mem_free(r); retval = RET_SYNTAX; goto ttt; } handle_basic_signals(NULL); init_dither(drv->depth); F = 1;#else fprintf(stderr, "Graphics not enabled when compiling\n"); retval = RET_SYNTAX; goto ttt;#endif } initialize_all_subsystems_2(); if (!((info = create_session_info(base_session, u, default_target, &len)) && gf_val(attach_terminal(get_input_handle(), get_output_handle(), get_ctl_handle(), info, len), attach_g_terminal(info, len)) != -1)) { retval = RET_FATAL; terminate_loop = 1; } } else { unsigned char *uu, *wd; initialize_all_subsystems_2(); close(terminal_pipe[0]); close(terminal_pipe[1]); if (!*u) { fprintf(stderr, "URL expected after %s\n.", dmp == D_DUMP ? "-dump" : "-source"); retval = RET_SYNTAX; goto tttt; } if (!(uu = translate_url(u, wd = get_cwd()))) uu = stracpy(u); request_object(NULL, uu, NULL, PRI_MAIN, NC_RELOAD, end_dump, NULL, &dump_obj); mem_free(uu); if (wd) mem_free(wd); }}/* Is called before gaphics driver init */void initialize_all_subsystems(void){ init_trans(); set_sigcld(); init_home(); init_dns(); init_cache(); iinit_bfu(); memset(&dd_opt, 0, sizeof dd_opt);}/* Is called sometimes after and sometimes before graphics driver init */void initialize_all_subsystems_2(void){ GF(init_dip()); init_bfu(); GF(init_imgcache()); init_fcache(); GF(init_grview());}void terminate_all_subsystems(void){ if (!F) af_unix_close(); check_bottom_halves(); abort_all_downloads();#ifdef HAVE_SSL ssl_finish();#endif check_bottom_halves(); destroy_all_terminals(); check_bottom_halves(); shutdown_bfu(); GF(shutdown_dip()); if (!F) free_all_itrms(); release_object(&dump_obj); abort_all_connections(); free_all_caches(); if (init_b) save_url_history(); free_history_lists(); free_term_specs(); free_types(); free_blocks(); if (init_b) finalize_bookmarks(); free_conv_table(); free_blacklist(); if (init_b) cleanup_cookies(); cleanup_auth(); check_bottom_halves(); end_config(); free_strerror_buf(); shutdown_trans(); GF(shutdown_graphics()); terminate_osdep(); if (clipboard) mem_free(clipboard);}int main(int argc, char *argv[]){ path_to_exe = argv[0]; g_argc = argc; g_argv = (unsigned char **)argv; get_path_to_exe(); select_loop(init); terminate_all_subsystems(); if (fg_poll_timer != -1) kill_timer(fg_poll_timer), fg_poll_timer = -1; check_memory_leaks(); return retval;}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -