NetBSD/distrib/utils/sysinst/main.c

465 lines
11 KiB
C
Raw Normal View History

/* $NetBSD: main.c,v 1.54 2009/10/18 12:09:48 ahoka Exp $ */
/*
* Copyright 1997 Piermont Information Systems Inc.
* All rights reserved.
*
* Written by Philip A. Nelson for Piermont Information Systems Inc.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. All advertising materials mentioning features or use of this software
* must display the following acknowledgement:
* This product includes software developed for the NetBSD Project by
* Piermont Information Systems Inc.
* 4. The name of Piermont Information Systems Inc. may not be used to endorse
* or promote products derived from this software without specific prior
* written permission.
*
* THIS SOFTWARE IS PROVIDED BY PIERMONT INFORMATION SYSTEMS INC. ``AS IS''
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL PIERMONT INFORMATION SYSTEMS INC. BE
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
* THE POSSIBILITY OF SUCH DAMAGE.
*
*/
/* main sysinst program. */
#include <sys/types.h>
#include <sys/stat.h>
#include <sys/uio.h>
#include <stdio.h>
#include <signal.h>
#include <curses.h>
#include <unistd.h>
#include <fcntl.h>
#include <dirent.h>
#include <locale.h>
#include "defs.h"
#include "md.h"
#include "msg_defs.h"
#include "menu_defs.h"
#include "txtwalk.h"
int main(int, char **);
static void select_language(void);
static void usage(void);
static void miscsighandler(int);
static void ttysighandler(int);
static void cleanup(void);
static void process_f_flag(char *);
static int exit_cleanly = 0; /* Did we finish nicely? */
int logging; /* are we logging everything? */
int scripting; /* are we building a script? */
FILE *logfp; /* log file */
FILE *script; /* script file */
#ifdef DEBUG
extern int log_flip(void);
#endif
/* String defaults and stuff for processing the -f file argument. */
struct f_arg {
const char *name;
const char *dflt;
char *var;
int size;
};
static const struct f_arg fflagopts[] = {
{"release", REL, rel, sizeof rel},
{"machine", MACH, machine, sizeof machine},
{"xfer dir", "/usr/INSTALL", xfer_dir, sizeof xfer_dir},
{"ext dir", "", ext_dir, sizeof ext_dir},
{"ftp host", SYSINST_FTP_HOST, ftp.host, sizeof ftp.host},
{"ftp dir", SYSINST_FTP_DIR, ftp.dir, sizeof ftp.dir},
{"ftp prefix", "/" MACH "/binary/sets", set_dir, sizeof set_dir},
{"ftp user", "ftp", ftp.user, sizeof ftp.user},
{"ftp pass", "", ftp.pass, sizeof ftp.pass},
{"ftp proxy", "", ftp.proxy, sizeof ftp.proxy},
{"nfs host", "", nfs_host, sizeof nfs_host},
{"nfs dir", "/bsd/release", nfs_dir, sizeof nfs_dir},
{"cd dev", "cd0a", cdrom_dev, sizeof cdrom_dev},
{"fd dev", "/dev/fd0a", fd_dev, sizeof fd_dev},
{"local dev", "", localfs_dev, sizeof localfs_dev},
{"local fs", "ffs", localfs_fs, sizeof localfs_fs},
{"local dir", "release", localfs_dir, sizeof localfs_dir},
{"targetroot mount", "/targetroot", targetroot_mnt, sizeof targetroot_mnt},
{"dist postfix", ".tgz", dist_postfix, sizeof dist_postfix},
{"diskname", "mydisk", bsddiskname, sizeof bsddiskname},
{NULL, NULL, NULL, 0}
};
static void
init(void)
{
const struct f_arg *arg;
sizemult = 1;
disktype = "unknown";
tmp_ramdisk_size = 0;
doessf = "";
clean_xfer_dir = 0;
mnt2_mounted = 0;
fd_type = "msdos";
for (arg = fflagopts; arg->name != NULL; arg++)
strlcpy(arg->var, arg->dflt, arg->size);
}
1998-06-20 17:05:48 +04:00
int
main(int argc, char **argv)
{
WINDOW *win;
int ch;
logging = 0; /* shut them off unless turned on by the user */
init();
#ifdef DEBUG
log_flip();
#endif
scripting = 0;
/* Check for TERM ... */
if (!getenv("TERM")) {
1998-06-20 17:05:48 +04:00
(void)fprintf(stderr,
"sysinst: environment variable TERM not set.\n");
exit(1);
}
/* argv processing */
while ((ch = getopt(argc, argv, "Dr:f:")) != -1)
switch(ch) {
case 'D': /* set to get past certain errors in testing */
debug = 1;
break;
case 'r':
/* Release name other than compiled in release. */
strncpy(rel, optarg, sizeof rel);
break;
case 'f':
/* Definition file to read. */
process_f_flag(optarg);
break;
case '?':
default:
usage();
}
md_init();
/* initialize message window */
if (menu_init()) {
__menu_initerror();
exit(1);
}
/*
* Put 'messages' in a window that has a one-character border
* on the real screen.
*/
win = newwin(getmaxy(stdscr) - 2, getmaxx(stdscr) - 2, 1, 1);
if (win == NULL) {
(void)fprintf(stderr,
"sysinst: screen too small\n");
exit(1);
}
if (has_colors()) {
/*
* XXX This color trick should be done so much better,
* but is it worth it?
*/
wbkgd(win, COLOR_PAIR(1));
wattrset(win, COLOR_PAIR(1));
}
msg_window(win);
/* Watch for signals and clean up */
1998-06-20 17:05:48 +04:00
(void)atexit(cleanup);
(void)signal(SIGINT, ttysighandler);
(void)signal(SIGQUIT, ttysighandler);
(void)signal(SIGHUP, miscsighandler);
/* redraw screen */
touchwin(stdscr);
refresh();
/* Ensure we have mountpoint for target filesystems */
mkdir(targetroot_mnt, S_IRWXU| S_IRGRP|S_IXGRP | S_IROTH|S_IXOTH);
select_language();
get_kb_encoding();
/* Menu processing */
process_menu(MENU_netbsd, NULL);
exit_cleanly = 1;
return 0;
}
static int
set_language(menudesc *m, void *arg)
{
char **fnames = arg;
msg_file(fnames[m->cursel]);
return 1;
}
static void
select_language(void)
{
DIR *dir;
struct dirent *dirent;
char **lang_msg, **fnames;
int max_lang = 16, num_lang = 0;
const char *cp;
menu_ent *opt = 0;
int lang_menu = -1;
int lang;
dir = opendir(".");
if (!dir)
return;
lang_msg = malloc(max_lang * sizeof *lang_msg);
fnames = malloc(max_lang * sizeof *fnames);
if (!lang_msg || !fnames)
goto done;
lang_msg[0] = strdup(msg_string(MSG_sysinst_message_language));
fnames[0] = 0;
num_lang = 1;
while ((dirent = readdir(dir)) != 0) {
if (memcmp(dirent->d_name, "sysinstmsgs.", 12))
continue;
if (msg_file(dirent->d_name))
continue;
cp = msg_string(MSG_sysinst_message_language);
if (!strcmp(cp, lang_msg[0]))
continue;
if (num_lang == max_lang) {
char **new;
max_lang *= 2;
new = realloc(lang_msg, max_lang * sizeof *lang_msg);
if (!new)
break;
lang_msg = new;
new = realloc(fnames, max_lang * sizeof *fnames);
if (!new)
break;
fnames = new;
}
fnames[num_lang] = strdup(dirent->d_name);
lang_msg[num_lang++] = strdup(cp);
}
msg_file(0);
closedir(dir);
dir = 0;
if (num_lang == 1)
goto done;
opt = calloc(num_lang, sizeof *opt);
if (!opt)
goto done;
for (lang = 0; lang < num_lang; lang++) {
opt[lang].opt_name = lang_msg[lang];
opt[lang].opt_menu = OPT_NOMENU;
opt[lang].opt_action = set_language;
}
lang_menu = new_menu(NULL, opt, num_lang, -1, 12, 0, 0, MC_NOEXITOPT,
NULL, NULL, NULL, NULL, NULL);
if (lang_menu != -1) {
msg_display(MSG_hello);
process_menu(lang_menu, fnames);
}
done:
if (dir)
closedir(dir);
if (lang_menu != -1)
free_menu(lang_menu);
free(opt);
while (num_lang) {
free(lang_msg[--num_lang]);
free(fnames[num_lang]);
}
free(lang_msg);
free(fnames);
/* set locale according to selected language */
cp = msg_string(MSG_sysinst_message_locale);
if (cp) {
setlocale(LC_CTYPE, cp);
setenv("LC_CTYPE", cp, 1);
}
}
/* toplevel menu handler ... */
1998-06-20 17:05:48 +04:00
void
toplevel(void)
{
1998-06-20 17:05:48 +04:00
2000-05-11 06:23:53 +04:00
/* Display banner message in (english, francais, deutsch..) */
1998-06-20 17:05:48 +04:00
msg_display(MSG_hello);
msg_display_add(MSG_md_hello);
msg_display_add(MSG_thanks);
/*
* Undo any stateful side-effects of previous menu choices.
* XXX must be idempotent, since we get run each time the main
* menu is displayed.
*/
unwind_mounts();
/* ... */
}
/* The usage ... */
static void
usage(void)
{
1998-06-20 17:05:48 +04:00
(void)fprintf(stderr, msg_string(MSG_usage));
exit(1);
}
/* ARGSUSED */
static void
miscsighandler(int signo)
{
1998-06-20 17:05:48 +04:00
/*
* we need to cleanup(), but it was already scheduled with atexit(),
* so it'll be invoked on exit().
*/
exit(1);
}
static void
ttysighandler(int signo)
{
/*
* if we want to ignore a TTY signal (SIGINT or SIGQUIT), then we
* just return. If we want to forward a TTY signal, we forward it
* to the specified process group.
*
* This functionality is used when setting up and displaying child
* output so that the child gets the signal and presumably dies,
* but sysinst continues. We use this rather than actually ignoring
* the signals, because that will be be passed on to a child
* through fork/exec, whereas special handlers get reset on exec..
*/
if (ttysig_ignore)
return;
if (ttysig_forward) {
killpg(ttysig_forward, signo);
return;
}
/*
* we need to cleanup(), but it was already scheduled with atexit(),
* so it'll be invoked on exit().
*/
exit(1);
}
static void
cleanup(void)
{
time_t tloc;
1998-06-20 17:05:48 +04:00
(void)time(&tloc);
#if 0
restore_etc();
#endif
/* Ensure we aren't inside the target tree */
chdir(getenv("HOME"));
unwind_mounts();
Checkpoint a load of changes - seems to work ok still. - rework run_prog() so that the program name (etc) is usually displayed if the program generates any output, or terminates with an error. Allow arguments to included in single quotes. Try to collect console output so it doesn't interfere with curses. - Add a '*' to the cylinder count if non-integral number of cylinders (on disklabel editor) - Only show partition type for unused partitions. - Show size including unused space on '+' partition, remove a..z since the don't relate to partition IDs (netbsd partition sizes) - Fix deleting of 'user' partitions - killed size of next ptn. - Don't default a swap partition is the disk already has one. - Fix deleting of extended MBR partitions - changed size of ptn 2. - Show error message if user tries something illegal in mbr editor. - Default to old diskname (actually disk type - dunno why!) - Use MI enable_rc_conf() to set RC_CONFIGURED=YES, use a single sed command instead of a raft of code playing with files etc. - Float some menus to just below header text, saves counting and lets language variants have different height headers (use y=-1) - Track whether anything is mounted on /mnt2 better. - Put more texts into message file. - Change english prose texts to be more correct. - Stop french and polish versions core dumping if ptn start/size changed. - Fix processing logic for saving /etc (action is still borked) - Do tail-end setup if any sets (eg X) can't be found (but not if you give in (yet)).
2003-10-20 00:17:31 +04:00
umount_mnt2();
endwin();
if (logging) {
fprintf(logfp, "Log ended at: %s\n", asctime(localtime(&tloc)));
fflush(logfp);
fclose(logfp);
}
if (scripting) {
fprintf(script, "# Script ended at: %s\n",
asctime(localtime(&tloc)));
fflush(script);
fclose(script);
}
if (!exit_cleanly)
fprintf(stderr, "\n\nsysinst terminated.\n");
}
/* process function ... */
void
process_f_flag(char *f_name)
{
char buffer[STRSIZE];
int len;
const struct f_arg *arg;
FILE *fp;
char *cp, *cp1;
/* open the file */
fp = fopen(f_name, "r");
if (fp == NULL) {
fprintf(stderr, msg_string(MSG_config_open_error), f_name);
exit(1);
}
while (fgets(buffer, sizeof buffer, fp) != NULL) {
cp = buffer + strspn(buffer, " \t");
if (strchr("#\r\n", *cp) != NULL)
continue;
for (arg = fflagopts; arg->name != NULL; arg++) {
len = strlen(arg->name);
if (memcmp(cp, arg->name, len) != 0)
continue;
cp1 = cp + len;
cp1 += strspn(cp1, " \t");
if (*cp1++ != '=')
continue;
cp1 += strspn(cp1, " \t");
len = strcspn(cp1, " \n\r\t");
cp1[len] = 0;
strlcpy(arg->var, cp1, arg->size);
break;
}
}
fclose(fp);
}