/* vi: set sw=4 ts=4: */ /* * eject implementation for busybox * * Copyright (C) 2004 Peter Willis <psyphreak@phreaker.net> * Copyright (C) 2005 Tito Ragusa <farmatito@tiscali.it> * * Licensed under the GPL v2 or later, see the file LICENSE in this tarball. */ /* * This is a simple hack of eject based on something Erik posted in #uclibc. * Most of the dirty work blatantly ripped off from cat.c =) */ #include "busybox.h" /* various defines swiped from linux/cdrom.h */ #define CDROMCLOSETRAY 0x5319 /* pendant of CDROMEJECT */ #define CDROMEJECT 0x5309 /* Ejects the cdrom media */ #define CDROM_DRIVE_STATUS 0x5326 /* Get tray position, etc. */ /* drive status possibilities returned by CDROM_DRIVE_STATUS ioctl */ #define CDS_TRAY_OPEN 2 #define FLAG_CLOSE 1 #define FLAG_SMART 2 int eject_main(int argc, char **argv); int eject_main(int argc, char **argv) { unsigned long flags; const char *device; int dev, cmd; opt_complementary = "?:?1:t--T:T--t"; flags = getopt32(argc, argv, "tT"); device = argv[optind] ? : "/dev/cdrom"; // We used to do "umount <device>" here, but it was buggy // if something was mounted OVER cdrom and // if cdrom is mounted many times. // // This works equally well (or better): // #!/bin/sh // umount /dev/cdrom // eject dev = xopen(device, O_RDONLY|O_NONBLOCK); cmd = CDROMEJECT; if (flags & FLAG_CLOSE || (flags & FLAG_SMART && ioctl(dev, CDROM_DRIVE_STATUS) == CDS_TRAY_OPEN)) cmd = CDROMCLOSETRAY; if (ioctl(dev, cmd)) { bb_perror_msg_and_die("%s", device); } if (ENABLE_FEATURE_CLEAN_UP) close(dev); return EXIT_SUCCESS; }