propagator/cdrom.c
Michael Shigorin c998c9b406 cdrom.c: fixed thinko breaking strlen
The misoptimization of the final commit version resulted
in strlen operating on a 5 char too short string thus
delivering "device names" like "/de" and "/de1".

Thanks yet another late colleague :)
2013-03-16 00:54:46 +04:00

249 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/";
strcpy(device_fullname + 5, dev_name);
size_t l = strlen(device_fullname);
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();
}