248 lines
5.5 KiB
C
248 lines
5.5 KiB
C
/*
|
|
* Guillaume Cottenceau (gc@mandrakesoft.com)
|
|
*
|
|
* Copyright 2000 MandrakeSoft
|
|
*
|
|
* 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 Erik Troan (ewt@redhat.com)
|
|
*
|
|
* Copyright 1996 Red Hat Software
|
|
*
|
|
*/
|
|
|
|
#include <sys/types.h>
|
|
#include <sys/stat.h>
|
|
#include <stdlib.h>
|
|
#include <unistd.h>
|
|
#include <string.h>
|
|
#include <stdio.h>
|
|
#include <sys/mount.h>
|
|
#include <dirent.h>
|
|
#include <ctype.h>
|
|
#include "stage1.h"
|
|
#include "frontend.h"
|
|
#include "modules.h"
|
|
#include "probing.h"
|
|
#include "log.h"
|
|
#include "mount.h"
|
|
|
|
#include "cdrom.h"
|
|
|
|
extern char version[];
|
|
|
|
static int mount_that_cd_device(char * dev_name)
|
|
{
|
|
char device_fullname[64] = "/dev/";
|
|
size_t l = strlen(strcpy(device_fullname + 5, dev_name));
|
|
|
|
if (islower(device_fullname[l - 1])) {
|
|
int ret;
|
|
|
|
device_fullname[l] = '1';
|
|
device_fullname[l + 1] = '\0';
|
|
ret = my_mount(device_fullname, IMAGE_LOCATION, "iso9660", 0);
|
|
if (ret != -1)
|
|
return ret;
|
|
device_fullname[l] = '\0';
|
|
}
|
|
|
|
return my_mount(device_fullname, IMAGE_LOCATION, "iso9660", 0);
|
|
}
|
|
|
|
|
|
static int test_that_cd()
|
|
{
|
|
log_message("test file on cd: %s\n", get_ramdisk_path(NULL));
|
|
return access(get_ramdisk_path(NULL), R_OK);
|
|
}
|
|
|
|
|
|
static enum return_type try_with_device(char * dev_name, char * dev_model);
|
|
|
|
static enum return_type do_with_device(char * dev_name, char * dev_model)
|
|
{
|
|
struct stat st;
|
|
char msg[256];
|
|
char * ramdisk_path;
|
|
if (test_that_cd()) {
|
|
enum return_type results;
|
|
umount(IMAGE_LOCATION);
|
|
snprintf(msg, sizeof(msg), "That ISO does not seem like proper %s"
|
|
" media.\nRetry with another one?", version);
|
|
results = ask_yes_no(msg);
|
|
if (results == RETURN_OK)
|
|
return try_with_device(dev_name, dev_model);
|
|
return results;
|
|
}
|
|
|
|
log_message("found %s ISO9660 media, good news!", version);
|
|
|
|
if (IS_RESCUE) {
|
|
load_ramdisk(NULL);
|
|
umount(IMAGE_LOCATION);
|
|
}
|
|
|
|
ramdisk_path = get_ramdisk_path(NULL);
|
|
stat(ramdisk_path, &st);
|
|
if (S_ISDIR(st.st_mode)) {
|
|
mount(ramdisk_path, STAGE2_LOCATION, NULL, MS_BIND, NULL);
|
|
} else {
|
|
do_losetup(LIVE_DEVICE,ramdisk_path);
|
|
my_mount(LIVE_DEVICE, STAGE2_LOCATION, (IS_LIVE) ? LIVEFS : STAGE2FS, 0);
|
|
}
|
|
|
|
method_name = strdup("cdrom");
|
|
add_to_env("METHOD", method_name);
|
|
add_to_env("PREFIX", "/");
|
|
|
|
return RETURN_OK;
|
|
}
|
|
|
|
static enum return_type try_with_device(char * dev_name, char * dev_model)
|
|
{
|
|
wait_message("Trying to access media in 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 %s media in drive %s.\nRetry?", version, dev_model);
|
|
results = ask_yes_no(msg);
|
|
if (results == RETURN_OK)
|
|
return try_with_device(dev_name, dev_model);
|
|
return results;
|
|
}
|
|
remove_wait_message();
|
|
|
|
return do_with_device(dev_name, dev_model);
|
|
}
|
|
|
|
int try_automatic(char ** medias, char ** medias_models)
|
|
{
|
|
static char * already_tried[50] = { NULL };
|
|
char ** model = medias_models;
|
|
char ** ptr = medias;
|
|
int i = 0;
|
|
while (ptr && *ptr) {
|
|
char ** p;
|
|
for (p = already_tried; p && *p; p++)
|
|
if (streq(*p, *ptr))
|
|
goto try_automatic_already_tried;
|
|
*p = strdup(*ptr);
|
|
*(p+1) = NULL;
|
|
|
|
wait_message("Trying to access media in drive %s", *model);
|
|
if (mount_that_cd_device(*ptr) != -1) {
|
|
if (!test_that_cd()) {
|
|
remove_wait_message();
|
|
return i;
|
|
}
|
|
else
|
|
umount(IMAGE_LOCATION);
|
|
}
|
|
remove_wait_message();
|
|
|
|
try_automatic_already_tried:
|
|
ptr++;
|
|
model++;
|
|
i++;
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
enum return_type cdrom_prepare(void)
|
|
{
|
|
char ** medias, ** ptr, ** medias_models;
|
|
char * choice;
|
|
int i, count = 0;
|
|
enum return_type results;
|
|
int timeout=60;
|
|
|
|
update_splash("prepare");
|
|
|
|
fprintf(stderr,"%c[1A",27);
|
|
for(i=0; i<timeout; i++) {
|
|
sleep(1);
|
|
if(opendir("/dev/disk/"))break;
|
|
fprintf(stderr,"\rwaiting for /dev/disk/ %d...",i);
|
|
}
|
|
|
|
update_splash("wait_media");
|
|
get_medias(CDROM, &medias, &medias_models);
|
|
|
|
if (IS_AUTOMATIC) {
|
|
ptr = medias;
|
|
while (ptr && *ptr) {
|
|
count++;
|
|
ptr++;
|
|
}
|
|
if ((i = try_automatic(medias, medias_models)) != -1)
|
|
return do_with_device(medias[i], medias_models[i]);
|
|
|
|
if (count == 0) {
|
|
/* too bad, let's recount one more time */
|
|
ptr = medias;
|
|
while (ptr && *ptr) {
|
|
count++;
|
|
ptr++;
|
|
}
|
|
if (count == 0)
|
|
stg1_error_message("No CD/DVD/Pendrive found.");
|
|
|
|
i = ask_insmod();
|
|
if (i == RETURN_BACK)
|
|
return RETURN_BACK;
|
|
return cdrom_prepare();
|
|
}
|
|
|
|
if (count == 1) {
|
|
results = try_with_device(*medias, *medias_models);
|
|
if (results == RETURN_OK)
|
|
return RETURN_OK;
|
|
i = ask_insmod();
|
|
if (i == RETURN_BACK)
|
|
return RETURN_BACK;
|
|
return cdrom_prepare();
|
|
}
|
|
|
|
unset_param(MODE_AUTOMATIC);
|
|
}
|
|
|
|
results = ask_from_list_comments("Please choose the drive to use for the installation.", 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_device(choice, *model);
|
|
} else
|
|
return results;
|
|
if (results == RETURN_OK)
|
|
{
|
|
update_splash("found_media");
|
|
return RETURN_OK;
|
|
}
|
|
if (results == RETURN_BACK)
|
|
return cdrom_prepare();
|
|
|
|
i = ask_insmod();
|
|
if (i == RETURN_BACK)
|
|
return RETURN_BACK;
|
|
return cdrom_prepare();
|
|
}
|