NetBSD/distrib/utils/sysinst/main.c

500 lines
12 KiB
C
Raw Normal View History

/* $NetBSD: main.c,v 1.48 2004/11/11 20:17:48 dsl 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 "defs.h"
#include "md.h"
#include "msg_defs.h"
#include "menu_defs.h"
#include "txtwalk.h"
int main(int, char **);
static void init(void);
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
static void
init(void)
{
(void)strlcpy(rel, REL, SSTRSIZE);
(void)strlcpy(machine, MACH, SSTRSIZE);
sizemult = 1;
(void)strlcpy(diskdev, "", SSTRSIZE);
disktype = "unknown";
tmp_mfs_size = 0;
(void)strlcpy(bsddiskname, "mydisk", DISKNAME_SIZE);
doessf = "";
(void)strlcpy(dist_dir, "/usr/INSTALL", STRSIZE);
clean_dist_dir = 0;
(void)strlcpy(ext_dir, "", STRSIZE);
(void)strlcpy(set_dir, "/" MACH "/binary/sets", STRSIZE);
(void)strlcpy(ftp_host, SYSINST_FTP_HOST, STRSIZE);
(void)strlcpy(ftp_dir, SYSINST_FTP_DIR, STRSIZE);
(void)strlcpy(ftp_user, "ftp", SSTRSIZE);
(void)strlcpy(ftp_pass, "", STRSIZE);
(void)strlcpy(ftp_proxy, "", STRSIZE);
(void)strlcpy(nfs_host, "", STRSIZE);
(void)strlcpy(nfs_dir, "/bsd/release", STRSIZE);
(void)strlcpy(cdrom_dev, "cd0a", SSTRSIZE);
(void)strlcpy(localfs_dev, "sd0a", SSTRSIZE);
(void)strlcpy(localfs_fs, "ffs", SSTRSIZE);
(void)strlcpy(localfs_dir, "release", STRSIZE);
(void)strlcpy(targetroot_mnt, "/targetroot", SSTRSIZE);
(void)strlcpy(distfs_mnt, "/mnt2", SSTRSIZE);
mnt2_mounted = 0;
(void)strlcpy(dist_postfix, ".tgz", SSTRSIZE);
}
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. */
1998-06-20 17:05:48 +04:00
strncpy(rel, optarg, SSTRSIZE);
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);
}
/*
* XXX the following is bogus. if screen is too small, message
* XXX window will be overwritten by menus.
*/
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);
}
/* 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");
}
/* Stuff for processing the -f file argument. */
/* Data definitions ... */
static char *rel_ptr = rel;
static char *machine_ptr = machine;
static char *dist_dir_ptr = dist_dir;
static char *ext_dir_ptr = ext_dir;
static char *ftp_host_ptr = ftp_host;
static char *ftp_dir_ptr = ftp_dir;
static char *set_dir_ptr = set_dir;
static char *ftp_user_ptr = ftp_user;
static char *ftp_pass_ptr = ftp_pass;
static char *ftp_proxy_ptr = ftp_proxy;
static char *nfs_host_ptr = nfs_host;
static char *nfs_dir_ptr = nfs_dir;
static char *cdrom_dev_ptr = cdrom_dev;
static char *localfs_dev_ptr = localfs_dev;
static char *localfs_fs_ptr = localfs_fs;
static char *localfs_dir_ptr = localfs_dir;
static char *targetroot_mnt_ptr = targetroot_mnt;
static char *distfs_mnt_ptr = distfs_mnt;
static char *dist_postfix_ptr = dist_postfix;
struct lookfor fflagopts[] = {
{"release", "release = %s", "a $0", &rel_ptr, 1, SSTRSIZE, NULL},
{"machine", "machine = %s", "a $0", &machine_ptr, 1, SSTRSIZE, NULL},
{"dist dir", "dist dir = %s", "a $0", &dist_dir_ptr, 1, STRSIZE, NULL},
{"ext dir", "ext dir = %s", "a $0", &ext_dir_ptr, 1, STRSIZE, NULL},
{"ftp host", "ftp host = %s", "a $0", &ftp_host_ptr, 1, STRSIZE, NULL},
{"ftp dir", "ftp dir = %s", "a $0", &ftp_dir_ptr, 1, STRSIZE, NULL},
{"ftp prefix", "set dir = %s", "a $0", &set_dir_ptr, 1, STRSIZE, NULL},
{"ftp user", "ftp user = %s", "a $0", &ftp_user_ptr, 1, STRSIZE, NULL},
{"ftp pass", "ftp pass = %s", "a $0", &ftp_pass_ptr, 1, STRSIZE, NULL},
{"ftp proxy", "ftp proxy = %s", "a $0", &ftp_proxy_ptr, 1, STRSIZE,
NULL},
{"nfs host", "nfs host = %s", "a $0", &nfs_host_ptr, 1, STRSIZE, NULL},
{"nfs dir", "ftp dir = %s", "a $0", &nfs_dir_ptr, 1, STRSIZE, NULL},
{"cd dev", "cd dev = %s", "a $0", &cdrom_dev_ptr, 1, STRSIZE, NULL},
{"local dev", "local dev = %s", "a $0", &localfs_dev_ptr, 1, STRSIZE,
NULL},
{"local fs", "local fs = %s", "a $0", &localfs_fs_ptr, 1, STRSIZE,
NULL},
{"local dir", "local dir = %s", "a $0", &localfs_dir_ptr, 1, STRSIZE,
NULL},
{"targetroot mount", "targetroot mount = %s", "a $0",
&targetroot_mnt_ptr, 1, STRSIZE, NULL},
{"distfs mount", "distfs mount = %s", "a $0", &distfs_mnt_ptr, 1,
STRSIZE, NULL},
{"dist postfix", "dist postfix = %s", "a $0", &dist_postfix_ptr, 1,
STRSIZE, NULL},
};
/* process function ... */
void
process_f_flag(char *f_name)
{
char *buffer;
int fd;
int fsize;
/* open the file */
fd = open(f_name, O_RDONLY, 0);
if (fd < 0) {
fprintf(stderr, msg_string(MSG_config_open_error), f_name);
exit(1);
}
/* get file size */
fsize = lseek(fd, 0, SEEK_END);
lseek(fd, 0, SEEK_SET);
if (fsize == -1) {
fprintf(stderr, msg_string(MSG_not_regular_file), f_name);
exit(1);
}
/* allocate buffer (error reported) */
buffer = malloc(fsize + 1);
if (buffer == NULL) {
fprintf(stderr, msg_string(MSG_out_of_memory));
exit(1);
}
/* read the file */
if (read(fd,buffer, fsize) != fsize) {
fprintf(stderr, msg_string(MSG_config_read_error), f_name);
exit(1);
}
buffer[fsize] = 0;
/* close the file */
close(fd);
/* Walk the buffer */
walk(buffer, fsize, fflagopts,
sizeof(fflagopts)/sizeof(struct lookfor));
/* free the buffer */
free(buffer);
}