#ident "$Id: syslinux.c,v 1.11 1999/03/19 21:04:32 hpa Exp $"
/* ----------------------------------------------------------------------- *
*
* Copyright 1998-1999 H. Peter Anvin - All Rights Reserved
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, Inc., 675 Mass Ave, Cambridge MA 02139,
* USA; either version 2 of the License, or (at your option) any later
* version; incorporated herein by reference.
*
* ----------------------------------------------------------------------- */
/*
* syslinux.c - Linux installer program for SYSLINUX
*
* This program ought to be portable. I hope so, at least.
*
* HPA note: this program needs too much privilege. We should probably
* access the filesystem directly like mtools does so we don't have to
* mount the disk. Either that or if Linux gets an fmount() system call
* we probably could do the mounting ourselves, and make this program
* setuid safe.
*
*/
int main(int argc, char *argv[])
{
static unsigned char sectbuf[512];
unsigned char *dp;
const unsigned char *cdp;
int dev_fd, fd;
struct stat st;
int nb, left, veryold;
unsigned int sectors, clusters;
int err = 0;
pid_t f, w;
int status;
char *mntpath = NULL, mntname[64];
char *ldlinux_name, **argp, *opt;
int my_umask;
int force = 0; /* -f (force) option */
program = argv[0];
device = NULL;
for ( argp = argv+1 ; *argp ; argp++ ) {
if ( **argp == '-' ) {
opt = *argp + 1;
if ( !*opt )
usage();
while ( *opt ) {
if ( *opt == 's' ) {
make_stupid(); /* Use "safe, slow and stupid" code */
} else if ( *opt == 'f' ) {
force = 1;
} else {
usage();
}
opt++;
}
} else {
if ( device )
usage();
device = *argp;
}
}
if ( !device )
usage();
/*
* First make sure we can open the device at all, and that we have
* read/write permission.
*/
dev_fd = open(device, O_RDWR);
if ( dev_fd < 0 || fstat(dev_fd, &st) < 0 ) {
perror(device);
exit(1);
}
if ( !force && !S_ISBLK(st.st_mode) && !S_ISREG(st.st_mode) ) {
fprintf(stderr, "%s: not a block device or regular file (use -f to override)\n", device);
exit(1);
}
left = 512;
dp = sectbuf;
while ( left ) {
nb = read(dev_fd, dp, left);
if ( nb == -1 && errno == EINTR )
continue;
if ( nb < 0 ) {
perror(device);
exit(1);
} else if ( nb == 0 ) {
fprintf(stderr, "%s: no boot sector\n", device);
exit(1);
}
dp += nb;
left -= nb;
}
close(dev_fd);
sync();
/*
* Check to see that what we got was indeed an MS-DOS boot sector/superblock
*/
if ( sectbuf[bsBootSignature] == 0x29 ) {
/* It's DOS, and it has all the new nice fields */
if ( get_16(sectbuf+bsBytesPerSec) != 512 ) {
fprintf(stderr, "%s: Sector sizes other than 512 not supported\n",
device);
exit(1);
}
if ( sectbuf[bsSecPerClust] > 32 ) {
fprintf(stderr, "%s: Cluster sizes larger than 16K not supported\n",
device);
}
/*
* Now mount the device. If we are non-root we need to find an fstab
* entry for this device which has the user flag and the appropriate
* options set.
*/
if ( (euid = geteuid()) ) {
FILE *fstab;
struct mntent *mnt;
if ( !(fstab = setmntent(MNTTAB, "r")) ) {
fprintf(stderr, "%s: cannot open " MNTTAB "\n", program);
}
while ( (mnt = getmntent(fstab)) ) {
if ( !strcmp(device, mnt->mnt_fsname) &&
( !strcmp(mnt->mnt_type, "msdos") ||
!strcmp(mnt->mnt_type, "umsdos") ||
!strcmp(mnt->mnt_type, "vfat") ||
!strcmp(mnt->mnt_type, "uvfat") ||
!strcmp(mnt->mnt_type, "auto") ) &&
hasmntopt(mnt, "user") &&
!hasmntopt(mnt, "ro") &&
mnt->mnt_dir[0] == '/' &&
!!hasmntopt(mnt, "loop") == !!S_ISREG(st.st_mode)) {
/* Okay, this is an fstab entry we should be able to live with. */
if ( !mntpath ) {
fprintf(stderr, "%s: not root and no appropriate entry for %s in "
MNTTAB "\n", program, device);
exit(1);
}
f = fork();
if ( f < 0 ) {
perror(program);
exit(1);
} else if ( f == 0 ) {
execl(_PATH_MOUNT, _PATH_MOUNT, mntpath, NULL);
_exit(255); /* If execl failed, trouble... */
}
} else {
int i = 0;
/* We're root. Make a temp dir and pass all the gunky options to mount. */
cdp = ldlinux;
left = ldlinux_len;
while ( left ) {
nb = write(fd, cdp, left);
if ( nb == -1 && errno == EINTR )
continue;
else if ( nb <= 0 ) {
perror(device);
err = 1;
goto umount;
}
dp += nb;
left -= nb;
}
/*
* I don't understand why I need this. Does the DOS filesystems
* not honour the mode passed to open()?
*/
my_umask = umask(0777);
umask(my_umask);
fchmod(fd, 0444 & ~my_umask);
close(fd);
umount:
f = fork();
if ( f < 0 ) {
perror("fork");
exit(1);
} else if ( f == 0 ) {
execl(_PATH_UMOUNT, _PATH_UMOUNT, mntpath, NULL);
}
w = waitpid(f, &status, 0);
if ( w != f || status ) {
exit(1);
}
sync();
if ( !euid )
rmdir(mntpath);
if ( err )
exit(err);
/*
* To finish up, write the boot sector
*/
dev_fd = open(device, O_RDWR);
if ( dev_fd < 0 ) {
perror(device);
exit(1);
}
/* Copy the old superblock into the new boot sector */
memcpy(bootsect+bsCopyStart, sectbuf+bsCopyStart, bsCopyLen);
dp = bootsect;
left = bootsect_len;
while ( left ) {
nb = write(dev_fd, dp, left);
if ( nb == -1 && errno == EINTR )
continue;
else if ( nb <= 0 ) {
perror(device);
exit(1);
}