1 /* ----------------------------------------------------------------------- *
3 * Copyright 1998-2008 H. Peter Anvin - All Rights Reserved
4 * Copyright 2009-2010 Intel Corporation; author: H. Peter Anvin
6 * This program is free software; you can redistribute it and/or modify
7 * it under the terms of the GNU General Public License as published by
8 * the Free Software Foundation, Inc., 53 Temple Place Ste 330,
9 * Boston MA 02111-1307, USA; either version 2 of the License, or
10 * (at your option) any later version; incorporated herein by reference.
12 * ----------------------------------------------------------------------- */
15 * syslinux.c - Linux installer program for SYSLINUX
17 * This is Linux-specific by now.
19 * This is an alternate version of the installer which doesn't require
20 * mtools, but requires root privilege.
24 * If DO_DIRECT_MOUNT is 0, call mount(8)
25 * If DO_DIRECT_MOUNT is 1, call mount(2)
28 # define DO_DIRECT_MOUNT 1
30 # define DO_DIRECT_MOUNT 0 /* glibc has broken losetup ioctls */
34 #define _XOPEN_SOURCE 500 /* For pread() pwrite() */
35 #define _FILE_OFFSET_BITS 64
46 #include <sys/types.h>
48 #include <sys/mount.h>
50 #include "linuxioctl.h"
54 # define _PATH_MOUNT "/bin/mount"
57 # define _PATH_UMOUNT "/bin/umount"
60 # define _PATH_TMP "/tmp/"
66 # include <linux/loop.h>
74 #include "syslxopt.h" /* unified options */
76 extern const char *program
; /* Name of program */
79 char *mntpath
= NULL
; /* Path on which to mount */
82 int loop_fd
= -1; /* Loop device */
85 void __attribute__ ((noreturn
)) die(const char *msg
)
87 fprintf(stderr
, "%s: %s\n", program
, msg
);
91 ioctl(loop_fd
, LOOP_CLR_FD
, 0); /* Free loop device */
106 int do_mount(int dev_fd
, int *cookie
, const char *mntpath
, const char *fstype
)
112 if (fstat(dev_fd
, &st
) < 0)
117 if (!S_ISBLK(st
.st_mode
)) {
118 /* It's file, need to mount it loopback */
120 struct loop_info64 loopinfo
;
123 for (n
= 0; loop_fd
< 0; n
++) {
124 snprintf(devfdname
, sizeof devfdname
, "/dev/loop%u", n
);
125 loop_fd
= open(devfdname
, O_RDWR
);
126 if (loop_fd
< 0 && errno
== ENOENT
) {
127 die("no available loopback device!");
129 if (ioctl(loop_fd
, LOOP_SET_FD
, (void *)dev_fd
)) {
133 die("cannot set up loopback device");
138 if (ioctl(loop_fd
, LOOP_GET_STATUS64
, &loopinfo
) ||
139 (loopinfo
.lo_offset
= opt
.offset
,
140 ioctl(loop_fd
, LOOP_SET_STATUS64
, &loopinfo
)))
141 die("cannot set up loopback device");
146 snprintf(devfdname
, sizeof devfdname
, "/proc/%lu/fd/%d",
147 (unsigned long)mypid
, dev_fd
);
151 return mount(devfdname
, mntpath
, fstype
,
152 MS_NOEXEC
| MS_NOSUID
, "umask=077,quiet");
156 char devfdname
[128], mnt_opts
[128];
160 snprintf(devfdname
, sizeof devfdname
, "/proc/%lu/fd/%d",
161 (unsigned long)mypid
, dev_fd
);
167 if (!S_ISBLK(st
.st_mode
)) {
168 snprintf(mnt_opts
, sizeof mnt_opts
,
169 "rw,nodev,noexec,loop,offset=%llu,umask=077,quiet",
170 (unsigned long long)opt
.offset
);
172 snprintf(mnt_opts
, sizeof mnt_opts
,
173 "rw,nodev,noexec,umask=077,quiet");
175 execl(_PATH_MOUNT
, _PATH_MOUNT
, "-t", fstype
, "-o", mnt_opts
,
176 devfdname
, mntpath
, NULL
);
177 _exit(255); /* execl failed */
180 w
= waitpid(f
, &status
, 0);
181 return (w
!= f
|| status
) ? -1 : 0;
189 void do_umount(const char *mntpath
, int cookie
)
192 int loop_fd
= cookie
;
194 if (umount2(mntpath
, 0))
195 die("could not umount path");
198 ioctl(loop_fd
, LOOP_CLR_FD
, 0); /* Free loop device */
212 execl(_PATH_UMOUNT
, _PATH_UMOUNT
, mntpath
, NULL
);
215 w
= waitpid(f
, &status
, 0);
216 if (w
!= f
|| status
) {
223 * Modify the ADV of an existing installation
225 int modify_existing_adv(const char *path
)
228 syslinux_reset_adv(syslinux_adv
);
229 else if (read_adv(path
, "ldlinux.sys") < 0)
232 if (modify_adv() < 0)
235 if (write_adv(path
, "ldlinux.sys") < 0)
241 int do_open_file(char *name
)
245 if ((fd
= open(name
, O_RDONLY
)) >= 0) {
246 uint32_t zero_attr
= 0;
247 ioctl(fd
, FAT_IOCTL_SET_ATTRIBUTES
, &zero_attr
);
252 fd
= open(name
, O_WRONLY
| O_CREAT
| O_TRUNC
, 0444);
259 int main(int argc
, char *argv
[])
261 static unsigned char sectbuf
[SECTOR_SIZE
];
269 sector_t
*sectors
= NULL
;
270 int ldlinux_sectors
= (boot_image_len
+ SECTOR_SIZE
- 1) >> SECTOR_SHIFT
;
278 parse_options(argc
, argv
, MODE_SYSLINUX
);
280 /* Note: subdir is guaranteed to start and end in / */
281 if (opt
.directory
&& opt
.directory
[0]) {
282 int len
= strlen(opt
.directory
);
283 int rv
= asprintf(&subdir
, "%s%s%s",
284 opt
.directory
[0] == '/' ? "" : "/",
286 opt
.directory
[len
-1] == '/' ? "" : "/");
287 if (rv
< 0 || !subdir
) {
295 if (!opt
.device
|| opt
.install_mbr
|| opt
.activate_partition
)
296 usage(EX_USAGE
, MODE_SYSLINUX
);
299 * First make sure we can open the device at all, and that we have
300 * read/write permission.
302 dev_fd
= open(opt
.device
, O_RDWR
);
303 if (dev_fd
< 0 || fstat(dev_fd
, &st
) < 0) {
308 if (!S_ISBLK(st
.st_mode
) && !S_ISREG(st
.st_mode
) && !S_ISCHR(st
.st_mode
)) {
309 die("not a device or regular file");
312 if (opt
.offset
&& S_ISBLK(st
.st_mode
)) {
313 die("can't combine an offset with a block device");
316 xpread(dev_fd
, sectbuf
, SECTOR_SIZE
, opt
.offset
);
320 * Check to see that what we got was indeed an FAT/NTFS
321 * boot sector/superblock
323 if ((errmsg
= syslinux_check_bootsect(sectbuf
, &fs_type
))) {
324 fprintf(stderr
, "%s: %s\n", opt
.device
, errmsg
);
329 * Now mount the device.
332 die("This program needs root privilege");
338 /* We're root or at least setuid.
339 Make a temp dir and pass all the gunky options to mount. */
341 if (chdir(_PATH_TMP
)) {
342 fprintf(stderr
, "%s: Cannot access the %s directory.\n",
346 #define TMP_MODE (S_IXUSR|S_IWUSR|S_IXGRP|S_IWGRP|S_IWOTH|S_IXOTH|S_ISVTX)
348 if (stat(".", &dst
) || !S_ISDIR(dst
.st_mode
) ||
349 (dst
.st_mode
& TMP_MODE
) != TMP_MODE
) {
350 die("possibly unsafe " _PATH_TMP
" permissions");
354 snprintf(mntname
, sizeof mntname
, "syslinux.mnt.%lu.%d",
355 (unsigned long)mypid
, i
);
357 if (lstat(mntname
, &dst
) != -1 || errno
!= ENOENT
)
360 rv
= mkdir(mntname
, 0000);
363 if (errno
== EEXIST
|| errno
== EINTR
)
369 if (lstat(mntname
, &dst
) || dst
.st_mode
!= (S_IFDIR
| 0000) ||
371 die("someone is trying to symlink race us!");
373 break; /* OK, got something... */
379 if (fs_type
== VFAT
) {
380 if (do_mount(dev_fd
, &mnt_cookie
, mntpath
, "vfat") &&
381 do_mount(dev_fd
, &mnt_cookie
, mntpath
, "msdos")) {
383 die("failed on mounting fat volume");
385 } else if (fs_type
== NTFS
) {
386 if (do_mount(dev_fd
, &mnt_cookie
, mntpath
, "ntfs-3g")) {
388 die("failed on mounting ntfs volume");
392 ldlinux_path
= alloca(strlen(mntpath
) + strlen(subdir
) + 1);
393 sprintf(ldlinux_path
, "%s%s", mntpath
, subdir
);
395 ldlinux_name
= alloca(strlen(ldlinux_path
) + 14);
401 sprintf(ldlinux_name
, "%sldlinux.sys", ldlinux_path
);
403 /* update ADV only ? */
404 if (opt
.update_only
== -1) {
405 if (opt
.reset_adv
|| opt
.set_once
) {
406 modify_existing_adv(ldlinux_path
);
407 do_umount(mntpath
, mnt_cookie
);
411 } else if (opt
.update_only
&& !syslinux_already_installed(dev_fd
)) {
412 fprintf(stderr
, "%s: no previous syslinux boot sector found\n",
416 fprintf(stderr
, "%s: please specify --install or --update for the future\n", argv
[0]);
421 /* Read a pre-existing ADV, if already installed */
423 syslinux_reset_adv(syslinux_adv
);
424 else if (read_adv(ldlinux_path
, "ldlinux.sys") < 0)
425 syslinux_reset_adv(syslinux_adv
);
426 if (modify_adv() < 0)
429 fd
= do_open_file(ldlinux_name
);
435 /* Write it the first time */
436 if (xpwrite(fd
, boot_image
, boot_image_len
, 0) != (int)boot_image_len
||
437 xpwrite(fd
, syslinux_adv
, 2 * ADV_SIZE
,
438 boot_image_len
) != 2 * ADV_SIZE
) {
439 fprintf(stderr
, "%s: write failure on %s\n", program
, ldlinux_name
);
448 uint32_t attr
= 0x07; /* Hidden+System+Readonly */
449 ioctl(fd
, FAT_IOCTL_SET_ATTRIBUTES
, &attr
);
453 * Create a block map.
455 ldlinux_sectors
+= 2; /* 2 ADV sectors */
456 sectors
= calloc(ldlinux_sectors
, sizeof *sectors
);
457 if (sectmap(fd
, sectors
, ldlinux_sectors
)) {
464 sprintf(ldlinux_name
, "%sldlinux.c32", ldlinux_path
);
465 fd
= do_open_file(ldlinux_name
);
471 rv
= xpwrite(fd
, syslinux_ldlinuxc32
, syslinux_ldlinuxc32_len
, 0);
472 if (rv
!= (int)syslinux_ldlinuxc32_len
) {
473 fprintf(stderr
, "%s: write failure on %s\n", program
, ldlinux_name
);
482 uint32_t attr
= 0x07; /* Hidden+System+Readonly */
483 ioctl(fd
, FAT_IOCTL_SET_ATTRIBUTES
, &attr
);
490 do_umount(mntpath
, mnt_cookie
);
498 * Patch ldlinux.sys and the boot sector
500 i
= syslinux_patch(sectors
, ldlinux_sectors
, opt
.stupid_mode
,
501 opt
.raid_mode
, subdir
, NULL
);
502 patch_sectors
= (i
+ SECTOR_SIZE
- 1) >> SECTOR_SHIFT
;
505 * Write the now-patched first sectors of ldlinux.sys
507 for (i
= 0; i
< patch_sectors
; i
++) {
508 xpwrite(dev_fd
, boot_image
+ i
* SECTOR_SIZE
, SECTOR_SIZE
,
509 opt
.offset
+ ((off_t
) sectors
[i
] << SECTOR_SHIFT
));
513 * To finish up, write the boot sector
516 /* Read the superblock again since it might have changed while mounted */
517 xpread(dev_fd
, sectbuf
, SECTOR_SIZE
, opt
.offset
);
519 /* Copy the syslinux code into the boot sector */
520 syslinux_make_bootsect(sectbuf
, fs_type
);
522 /* Write new boot sector */
523 xpwrite(dev_fd
, sectbuf
, SECTOR_SIZE
, opt
.offset
);