propagator/live.c

258 lines
6.1 KiB
C
Raw Normal View History

/*
* Anton Farygin (rider@altlinux.com)
*
* Copyright 2004 ALT Linux
*
* This software may be freely redistributed under the terms of the GNU
* public license.
*
* You should have received a copy of the GNU General Public License
* along with this program; if not, write to the Free Software
* Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
*
*/
/*
* Portions from Guillaume Cottenceau (gc@mandrakesoft.com)
*
* Copyright 2000 MandrakeSoft
*
* Portions from Erik Troan (ewt@redhat.com)
*
* Copyright 1996 Red Hat Software
*
*/
#include <stdlib.h>
#include <unistd.h>
#include <string.h>
#include <stdio.h>
#include <sys/mount.h>
#include <linux/loop.h>
#include <fcntl.h>
#include <errno.h>
#include "stage1.h"
#include "frontend.h"
#include "modules.h"
#include "probing.h"
#include "log.h"
#include "mount.h"
#include "live.h"
static int mount_that_cd_device(char * dev_name)
{
char device_fullname[50];
strcpy(device_fullname, "/dev/");
strcat(device_fullname, dev_name);
return my_mount(device_fullname, IMAGE_LOCATION, "iso9660", 0);
}
int do_losetup(char * device, char * target)
{
int loopfd, targfd;
struct loop_info loopInfo;
loopfd = open( device, O_RDONLY );
if ( loopfd < 0 )
{
log_message( "losetup: error opening %s: %s", device, strerror(errno) );
return -1;
}
targfd = open( target, O_RDONLY );
if ( targfd < 0 )
{
log_message( "losetup: error opening %s: %s", target, strerror(errno) );
close( loopfd );
return -1;
}
if ( ioctl(loopfd, LOOP_SET_FD, targfd) < 0 )
{
log_message( "losetup: error setting up loopback device: %s", strerror(errno) );
close( loopfd );
close( targfd );
return -1;
}
memset(&loopInfo, 0, sizeof(loopInfo));
strcpy(loopInfo.lo_name, target);
if ( ioctl(loopfd, LOOP_SET_STATUS, &loopInfo) < 0 )
log_message( "losetup: error setting up loopback device: %s", strerror(errno) );
close( loopfd );
close( targfd );
return 0;
}
static int mount_image(char * dev_name)
{
char device_fullname[50];
strcpy(device_fullname, "/dev/");
strcat(device_fullname, dev_name);
log_message("Setup %s for " IMAGE_LOCATION LIVECD_LOCATION, device_fullname);
if (do_losetup(device_fullname, IMAGE_LOCATION LIVECD_LOCATION) == -1)
fatal_error("could not setup loopback for live cd image");
mkdir(LIVEIMAGE_LOCATION,0755);
return my_mount(device_fullname, LIVEIMAGE_LOCATION, "ext2", 0);
}
static int test_that_livecd()
{
return access(IMAGE_LOCATION LIVECD_LOCATION, R_OK);
}
static enum return_type try_with_live_device(char * dev_name, char * dev_model);
static enum return_type do_with_live_device(char * dev_name, char * dev_model)
{
pid_t preparepid, childpid;
int end_child;
int wait_status;
if (test_that_livecd()) {
enum return_type results;
umount(IMAGE_LOCATION);
results = ask_yes_no("That CDROM disc does not seem to be a " DISTRIB_NAME " LIVE CD.\nRetry with another disc?");
if (results == RETURN_OK)
return try_with_live_device(dev_name, dev_model);
return results;
}
log_message("found a " DISTRIB_NAME " LIVE-CD, good news!");
mount_image("cloop");
#if 0
if (!(preparepid = fork())) {
/* child */
char * child_argv[2];
char * args[3] = { "prepare_live.sh", NULL, };
char * envp[32] = { "HOME=/root", "TERM=linux", NULL, };
chroot("/tmp/live");
sleep(10);
execve("/root/prepare_live.sh",args,envp);
printf("error in exec of prepare_live.sh :-(\n");
return 0;
}
while (!end_child) {
childpid = wait4(-1, &wait_status, 0, NULL);
if (childpid == preparepid)
end_child = 1;
}
#endif
method_name = strdup("live");
return RETURN_OK;
}
static enum return_type try_with_live_device(char * dev_name, char * dev_model)
{
wait_message("Trying to access a CDROM disc (drive %s)", dev_model);
if (mount_that_cd_device(dev_name) == -1) {
enum return_type results;
char msg[500];
unset_param(MODE_AUTOMATIC); /* we are in a fallback mode */
remove_wait_message();
snprintf(msg, sizeof(msg), "I can't access a " DISTRIB_NAME " liveCD disc in your CDROM drive (%s).\nRetry?", dev_model);
results = ask_yes_no(msg);
if (results == RETURN_OK)
return try_with_live_device(dev_name, dev_model);
return results;
}
remove_wait_message();
return do_with_live_device(dev_name, dev_model);
}
enum return_type live_prepare(void)
{
char ** medias, ** ptr, ** medias_models;
char * choice;
int i, count = 0;
enum return_type results;
my_insmod("ide-cd", ANY_DRIVER_TYPE, NULL);
my_insmod("sr_mod", ANY_DRIVER_TYPE, NULL);
my_insmod("zlib_inflate", ANY_DRIVER_TYPE, NULL);
my_insmod("cloop", ANY_DRIVER_TYPE, NULL);
get_medias(CDROM, &medias, &medias_models);
ptr = medias;
while (ptr && *ptr) {
count++;
ptr++;
}
if (count == 0) {
stg1_error_message("No CDROM device found.");
i = ask_insmod(SCSI_ADAPTERS);
if (i == RETURN_BACK)
return RETURN_BACK;
return live_prepare();
}
if (count == 1) {
results = try_with_live_device(*medias, *medias_models);
if (results == RETURN_OK)
return RETURN_OK;
i = ask_insmod(SCSI_ADAPTERS);
if (i == RETURN_BACK)
return RETURN_BACK;
return live_prepare();
}
if (IS_AUTOMATIC) {
char ** model = medias_models;
ptr = medias;
while (ptr && *ptr) {
wait_message("Trying to access " DISTRIB_NAME " LIVE CD disc (drive %s)", *model);
if (mount_that_cd_device(*ptr) != -1) {
if (!test_that_livecd()) {
remove_wait_message();
return do_with_live_device(*ptr, *model);
}
else
umount(IMAGE_LOCATION);
}
remove_wait_message();
ptr++;
model++;
}
unset_param(MODE_AUTOMATIC);
return live_prepare();
}
else {
results = ask_from_list_comments("Please choose the CDROM drive to use for the livecd.", medias, medias_models, &choice);
if (results == RETURN_OK) {
char ** model = medias_models;
ptr = medias;
while (ptr && *ptr && model && *model) {
if (!strcmp(*ptr, choice))
break;
ptr++;
model++;
}
results = try_with_live_device(choice, *model);
} else
return results;
}
if (results == RETURN_OK)
return RETURN_OK;
if (results == RETURN_BACK)
return live_prepare();
i = ask_insmod(SCSI_ADAPTERS);
if (i == RETURN_BACK)
return RETURN_BACK;
return live_prepare();
}