1995-04-27 18:05:10 +00:00
/*
* The new sysinstall program .
*
* This is probably the last attempt in the ` sysinstall ' line , the next
* generation being slated to essentially a complete rewrite .
*
1995-05-28 03:05:06 +00:00
* $ Id : media . c , v 1.20 1995 / 05 / 27 23 : 39 : 31 phk Exp $
1995-04-27 18:05:10 +00:00
*
* Copyright ( c ) 1995
* Jordan Hubbard . All rights reserved .
*
* Redistribution and use in source and binary forms , with or without
* modification , are permitted provided that the following conditions
* are met :
* 1. Redistributions of source code must retain the above copyright
* notice , this list of conditions and the following disclaimer ,
* verbatim and that no modifications are made prior to this
* point in the file .
* 2. Redistributions in binary form must reproduce the above copyright
* notice , this list of conditions and the following disclaimer in the
* documentation and / or other materials provided with the distribution .
* 3. All advertising materials mentioning features or use of this software
* must display the following acknowledgement :
* This product includes software developed by Jordan Hubbard
* for the FreeBSD Project .
* 4. The name of Jordan Hubbard or the FreeBSD project may not be used to
* endorse or promote products derived from this software without specific
* prior written permission .
*
* THIS SOFTWARE IS PROVIDED BY JORDAN HUBBARD ` ` AS IS ' ' AND
* ANY EXPRESS OR IMPLIED WARRANTIES , INCLUDING , BUT NOT LIMITED TO , THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED . IN NO EVENT SHALL JORDAN HUBBARD OR HIS PETS BE LIABLE
* FOR ANY DIRECT , INDIRECT , INCIDENTAL , SPECIAL , EXEMPLARY , OR CONSEQUENTIAL
* DAMAGES ( INCLUDING , BUT NOT LIMITED TO , PROCUREMENT OF SUBSTITUTE GOODS
* OR SERVICES ; LOSS OF USE , DATA , LIFE OR PROFITS ; OR BUSINESS INTERRUPTION )
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY , WHETHER IN CONTRACT , STRICT
* LIABILITY , OR TORT ( INCLUDING NEGLIGENCE OR OTHERWISE ) ARISING IN ANY WAY
* OUT OF THE USE OF THIS SOFTWARE , EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE .
*
*/
1995-05-16 11:37:27 +00:00
# include <stdio.h>
1995-05-21 15:40:54 +00:00
# include <sys/errno.h>
# include <sys/fcntl.h>
1995-05-27 10:47:44 +00:00
# include <sys/stat.h>
# include <sys/mman.h>
1995-05-21 15:40:54 +00:00
# include <sys/wait.h>
# include <unistd.h>
1995-05-16 11:37:27 +00:00
# include "sysinstall.h"
1995-05-20 10:33:14 +00:00
static int
genericHook ( char * str , DeviceType type )
{
Device * * devs ;
/* Clip garbage off the ends */
string_prune ( str ) ;
str = string_skipwhite ( str ) ;
if ( ! * str )
return 0 ;
devs = deviceFind ( str , type ) ;
if ( devs )
mediaDevice = devs [ 0 ] ;
return devs ? 1 : 0 ;
}
static int
cdromHook ( char * str )
{
return genericHook ( str , DEVICE_TYPE_CDROM ) ;
}
1995-04-27 18:05:10 +00:00
/*
1995-05-16 02:53:31 +00:00
* Return 1 if we successfully found and set the installation type to
1995-04-27 18:05:10 +00:00
* be a CD .
*/
int
1995-05-01 21:56:32 +00:00
mediaSetCDROM ( char * str )
1995-04-27 18:05:10 +00:00
{
1995-05-20 03:49:10 +00:00
Device * * devs ;
1995-05-20 00:13:14 +00:00
int cnt ;
1995-05-20 10:33:14 +00:00
if ( OnCDROM = = TRUE ) {
1995-05-21 15:40:54 +00:00
static Device bootCD ;
/* This may need to be extended a little, but the basic idea is sound */
strcpy ( bootCD . name , " bootCD " ) ;
bootCD . type = DEVICE_TYPE_CDROM ;
bootCD . get = mediaGetCDROM ;
mediaDevice = & bootCD ;
1995-05-16 02:53:31 +00:00
return 1 ;
1995-05-20 10:33:14 +00:00
}
1995-05-16 02:53:31 +00:00
else {
1995-05-20 03:49:10 +00:00
devs = deviceFind ( NULL , DEVICE_TYPE_CDROM ) ;
1995-05-20 00:13:14 +00:00
cnt = deviceCount ( devs ) ;
if ( ! cnt ) {
msgConfirm ( " No CDROM devices found! Please check that your system's \n configuration is correct and that the CDROM drive is of a supported \n type. For more information, consult the hardware guide \n in the Doc menu. " ) ;
return 0 ;
}
else if ( cnt > 1 ) {
1995-05-20 10:33:14 +00:00
DMenu * menu ;
menu = deviceCreateMenu ( & MenuMediaCDROM , DEVICE_TYPE_CDROM , cdromHook ) ;
if ( ! menu )
msgFatal ( " Unable to create CDROM menu! Something is seriously wrong. " ) ;
dmenuOpenSimple ( menu ) ;
free ( menu ) ;
1995-05-20 00:13:14 +00:00
}
1995-05-20 10:33:14 +00:00
else
1995-05-20 00:13:14 +00:00
mediaDevice = devs [ 0 ] ;
1995-05-16 02:53:31 +00:00
}
1995-05-20 10:33:14 +00:00
return mediaDevice ? 1 : 0 ;
}
static int
floppyHook ( char * str )
{
return genericHook ( str , DEVICE_TYPE_FLOPPY ) ;
1995-04-27 18:05:10 +00:00
}
1995-04-29 19:33:06 +00:00
/*
1995-05-16 02:53:31 +00:00
* Return 1 if we successfully found and set the installation type to
1995-04-29 19:33:06 +00:00
* be a floppy
*/
int
1995-05-01 21:56:32 +00:00
mediaSetFloppy ( char * str )
1995-04-29 19:33:06 +00:00
{
1995-05-20 10:33:14 +00:00
Device * * devs ;
int cnt ;
devs = deviceFind ( NULL , DEVICE_TYPE_FLOPPY ) ;
cnt = deviceCount ( devs ) ;
if ( ! cnt ) {
msgConfirm ( " No floppy devices found! Please check that your system's \n configuration is correct. For more information, consult the hardware guide \n in the Doc menu. " ) ;
return 0 ;
}
else if ( cnt > 1 ) {
DMenu * menu ;
menu = deviceCreateMenu ( & MenuMediaFloppy , DEVICE_TYPE_FLOPPY , floppyHook ) ;
if ( ! menu )
msgFatal ( " Unable to create Floppy menu! Something is seriously wrong. " ) ;
dmenuOpenSimple ( menu ) ;
free ( menu ) ;
}
else
mediaDevice = devs [ 0 ] ;
return mediaDevice ? 1 : 0 ;
1995-04-29 19:33:06 +00:00
}
1995-05-21 15:40:54 +00:00
static int
DOSHook ( char * str )
{
return genericHook ( str , DEVICE_TYPE_DOS ) ;
}
1995-04-29 19:33:06 +00:00
/*
1995-05-16 02:53:31 +00:00
* Return 1 if we successfully found and set the installation type to
1995-04-29 19:33:06 +00:00
* be a DOS partition .
*/
int
1995-05-01 21:56:32 +00:00
mediaSetDOS ( char * str )
1995-04-29 19:33:06 +00:00
{
1995-05-16 02:53:31 +00:00
Device * * devs ;
1995-05-20 10:33:14 +00:00
Disk * d ;
Chunk * c1 ;
1995-05-21 15:40:54 +00:00
int i , cnt ;
1995-05-16 02:53:31 +00:00
1995-05-20 11:10:35 +00:00
devs = deviceFind ( NULL , DEVICE_TYPE_DOS ) ;
1995-05-21 15:40:54 +00:00
cnt = deviceCount ( devs ) ;
if ( cnt > 1 ) {
DMenu * menu ;
menu = deviceCreateMenu ( & MenuMediaDOS , DEVICE_TYPE_DOS , DOSHook ) ;
if ( ! menu )
msgFatal ( " Unable to create DOS menu! Something is seriously wrong. " ) ;
dmenuOpenSimple ( menu ) ;
free ( menu ) ;
}
else if ( cnt ) {
1995-05-20 11:10:35 +00:00
mediaDevice = devs [ 0 ] ;
return 1 ;
}
1995-05-21 15:40:54 +00:00
else {
1995-05-20 11:10:35 +00:00
devs = deviceFind ( NULL , DEVICE_TYPE_DISK ) ;
1995-05-21 15:40:54 +00:00
if ( ! devs ) {
msgConfirm ( " No disk devices found! " ) ;
return 0 ;
}
/* Now go chewing through looking for a DOS FAT partition */
for ( i = 0 ; devs [ i ] ; i + + ) {
d = ( Disk * ) devs [ i ] - > private ;
/* Now try to find a DOS partition */
for ( c1 = d - > chunks - > part ; c1 ; c1 = c1 - > next ) {
if ( c1 - > type = = fat ) {
/* Got one! */
mediaDevice = deviceRegister ( c1 - > name , c1 - > name , c1 - > name , DEVICE_TYPE_DOS , TRUE ,
1995-05-24 09:00:58 +00:00
mediaInitDOS , mediaGetDOS , NULL , mediaShutdownDOS , NULL ) ;
1995-05-27 10:47:44 +00:00
mediaDevice - > private = c1 ;
1995-05-21 15:40:54 +00:00
msgDebug ( " Found a DOS partition %s on drive %s \n " , c1 - > name , d - > name ) ;
break ;
}
1995-05-20 10:33:14 +00:00
}
}
}
if ( ! mediaDevice )
msgConfirm ( " No DOS primary partitions found! This installation method is unavailable " ) ;
return mediaDevice ? 1 : 0 ;
1995-04-29 19:33:06 +00:00
}
1995-05-21 15:40:54 +00:00
static int
tapeHook ( char * str )
{
return genericHook ( str , DEVICE_TYPE_TAPE ) ;
}
1995-04-29 19:33:06 +00:00
/*
1995-05-16 02:53:31 +00:00
* Return 1 if we successfully found and set the installation type to
1995-04-29 19:33:06 +00:00
* be a tape drive .
*/
int
1995-05-01 21:56:32 +00:00
mediaSetTape ( char * str )
1995-04-29 19:33:06 +00:00
{
1995-05-21 15:40:54 +00:00
Device * * devs ;
int cnt ;
devs = deviceFind ( NULL , DEVICE_TYPE_TAPE ) ;
cnt = deviceCount ( devs ) ;
if ( ! cnt ) {
msgConfirm ( " No tape drive devices found! Please check that your system's \n configuration is correct. For more information, consult the hardware guide \n in the Doc menu. " ) ;
return 0 ;
}
else if ( cnt > 1 ) {
DMenu * menu ;
menu = deviceCreateMenu ( & MenuMediaTape , DEVICE_TYPE_TAPE , tapeHook ) ;
if ( ! menu )
msgFatal ( " Unable to create tape drive menu! Something is seriously wrong. " ) ;
dmenuOpenSimple ( menu ) ;
free ( menu ) ;
}
else
mediaDevice = devs [ 0 ] ;
return mediaDevice ? 1 : 0 ;
1995-04-29 19:33:06 +00:00
}
/*
* Return 0 if we successfully found and set the installation type to
* be an ftp server
*/
int
1995-05-01 21:56:32 +00:00
mediaSetFTP ( char * str )
1995-04-29 19:33:06 +00:00
{
1995-05-21 15:40:54 +00:00
static Device ftpDevice ;
char * cp ;
1995-05-26 08:41:52 +00:00
tcpDeviceSelect ( NULL ) ;
1995-05-16 11:37:27 +00:00
dmenuOpenSimple ( & MenuMediaFTP ) ;
1995-05-21 15:40:54 +00:00
cp = getenv ( " ftp " ) ;
if ( ! cp )
return 0 ;
1995-05-26 20:31:02 +00:00
if ( ! strcmp ( cp , " other " ) ) {
cp = msgGetInput ( " ftp:// " , " Please specify the URL of a FreeBSD distribution on a \n remote ftp site. This site must accept anonymous ftp! \n A URL looks like this: ftp://<hostname>/<path> " ) ;
if ( ! cp | | strncmp ( " ftp:// " , cp , 6 ) )
return 0 ;
else
variable_set2 ( " ftp " , cp ) ;
}
1995-05-21 15:40:54 +00:00
strcpy ( ftpDevice . name , cp ) ;
ftpDevice . type = DEVICE_TYPE_NETWORK ;
ftpDevice . init = mediaInitFTP ;
ftpDevice . get = mediaGetFTP ;
1995-05-27 10:47:44 +00:00
ftpDevice . close = mediaCloseFTP ;
1995-05-24 09:00:58 +00:00
ftpDevice . shutdown = mediaShutdownFTP ;
1995-05-26 08:41:52 +00:00
ftpDevice . private = mediaDevice ;
1995-05-21 15:40:54 +00:00
mediaDevice = & ftpDevice ;
return 1 ;
1995-04-29 19:33:06 +00:00
}
/*
* Return 0 if we successfully found and set the installation type to
* be some sort of mounted filesystem ( it ' s also mounted at this point )
*/
int
1995-05-01 21:56:32 +00:00
mediaSetFS ( char * str )
1995-04-29 19:33:06 +00:00
{
return 0 ;
}
1995-05-16 02:53:31 +00:00
1995-05-27 23:39:35 +00:00
Boolean
mediaExtractDistBegin ( char * distname , char * dir , int * fd , int * zpid , int * cpid )
{
int i , pfd [ 2 ] , qfd [ 2 ] ;
if ( ! dir )
dir = " / " ;
msgWeHaveOutput ( " Extracting %s into %s directory.. " , distname , dir ) ;
Mkdir ( dir , NULL ) ;
chdir ( dir ) ;
pipe ( pfd ) ;
pipe ( qfd ) ;
* zpid = fork ( ) ;
if ( ! * zpid ) {
dup2 ( qfd [ 0 ] , 0 ) ; close ( qfd [ 0 ] ) ;
dup2 ( pfd [ 1 ] , 1 ) ; close ( pfd [ 1 ] ) ;
if ( DebugFD ! = - 1 )
dup2 ( DebugFD , 2 ) ;
else {
close ( 2 ) ;
open ( " /dev/null " , O_WRONLY ) ;
}
close ( qfd [ 1 ] ) ;
close ( pfd [ 0 ] ) ;
i = execl ( " /stand/gunzip " , " /stand/gunzip " , 0 ) ;
msgDebug ( " /stand/gunzip command returns %d status \n " , i ) ;
exit ( i ) ;
}
* fd = qfd [ 1 ] ;
close ( qfd [ 0 ] ) ;
* cpid = fork ( ) ;
if ( ! * cpid ) {
dup2 ( pfd [ 0 ] , 0 ) ; close ( pfd [ 0 ] ) ;
close ( pfd [ 1 ] ) ;
close ( qfd [ 1 ] ) ;
if ( DebugFD ! = - 1 ) {
dup2 ( DebugFD , 1 ) ;
dup2 ( DebugFD , 2 ) ;
}
else {
close ( 1 ) ; open ( " /dev/null " , O_WRONLY ) ;
dup2 ( 1 , 2 ) ;
}
i = execl ( " /stand/cpio " , " /stand/cpio " , " -iduVm " , " -H " , " tar " , 0 ) ;
msgDebug ( " /stand/cpio command returns %d status \n " , i ) ;
exit ( i ) ;
}
close ( pfd [ 0 ] ) ;
close ( pfd [ 1 ] ) ;
return TRUE ;
}
Boolean
mediaExtractDistEnd ( int zpid , int cpid )
{
int i , j ;
i = waitpid ( zpid , & j , 0 ) ;
if ( i < 0 ) { /* Don't check status - gunzip seems to return a bogus one! */
dialog_clear ( ) ;
msgDebug ( " wait for gunzip returned status of %d! \n " , i ) ;
return FALSE ;
}
i = waitpid ( cpid , & j , 0 ) ;
if ( i < 0 | | WEXITSTATUS ( j ) ) {
dialog_clear ( ) ;
msgDebug ( " cpio returned error status of %d! \n " , WEXITSTATUS ( j ) ) ;
return FALSE ;
}
return TRUE ;
}
1995-05-16 11:37:27 +00:00
Boolean
1995-05-23 02:41:18 +00:00
mediaExtractDist ( char * distname , char * dir , int fd )
1995-05-16 11:37:27 +00:00
{
1995-05-21 15:40:54 +00:00
int i , j , zpid , cpid , pfd [ 2 ] ;
if ( ! dir )
dir = " / " ;
1995-05-23 02:41:18 +00:00
msgWeHaveOutput ( " Extracting %s into %s directory.. " , distname , dir ) ;
1995-05-22 14:10:25 +00:00
Mkdir ( dir , NULL ) ;
chdir ( dir ) ;
pipe ( pfd ) ;
zpid = fork ( ) ;
if ( ! zpid ) {
dup2 ( fd , 0 ) ; close ( fd ) ;
dup2 ( pfd [ 1 ] , 1 ) ; close ( pfd [ 1 ] ) ;
if ( DebugFD ! = - 1 )
dup2 ( DebugFD , 2 ) ;
else {
close ( 2 ) ;
open ( " /dev/null " , O_WRONLY ) ;
1995-05-21 15:40:54 +00:00
}
close ( pfd [ 0 ] ) ;
1995-05-22 14:10:25 +00:00
i = execl ( " /stand/gunzip " , " /stand/gunzip " , 0 ) ;
msgDebug ( " /stand/gunzip command returns %d status \n " , i ) ;
exit ( i ) ;
}
cpid = fork ( ) ;
if ( ! cpid ) {
dup2 ( pfd [ 0 ] , 0 ) ; close ( pfd [ 0 ] ) ;
1995-05-21 15:40:54 +00:00
close ( fd ) ;
1995-05-22 14:10:25 +00:00
close ( pfd [ 1 ] ) ;
if ( DebugFD ! = - 1 ) {
dup2 ( DebugFD , 1 ) ;
dup2 ( DebugFD , 2 ) ;
1995-05-21 15:40:54 +00:00
}
1995-05-22 14:10:25 +00:00
else {
close ( 1 ) ; open ( " /dev/null " , O_WRONLY ) ;
dup2 ( 1 , 2 ) ;
1995-05-21 15:40:54 +00:00
}
1995-05-27 23:39:35 +00:00
i = execl ( " /stand/cpio " , " /stand/cpio " , " -iduVm " , " -H " , " tar " , 0 ) ;
1995-05-22 14:10:25 +00:00
msgDebug ( " /stand/cpio command returns %d status \n " , i ) ;
exit ( i ) ;
1995-05-21 15:40:54 +00:00
}
1995-05-22 14:10:25 +00:00
close ( pfd [ 0 ] ) ;
close ( pfd [ 1 ] ) ;
i = waitpid ( zpid , & j , 0 ) ;
1995-05-23 02:41:18 +00:00
if ( i < 0 ) { /* Don't check status - gunzip seems to return a bogus one! */
1995-05-22 14:10:25 +00:00
dialog_clear ( ) ;
1995-05-27 10:47:44 +00:00
msgDebug ( " wait for gunzip returned status of %d! \n " , i ) ;
1995-05-21 15:40:54 +00:00
return FALSE ;
1995-05-22 14:10:25 +00:00
}
i = waitpid ( cpid , & j , 0 ) ;
1995-05-23 02:41:18 +00:00
if ( i < 0 | | WEXITSTATUS ( j ) ) {
1995-05-22 14:10:25 +00:00
dialog_clear ( ) ;
1995-05-27 10:47:44 +00:00
msgDebug ( " cpio returned error status of %d! \n " , WEXITSTATUS ( j ) ) ;
1995-05-22 14:10:25 +00:00
return FALSE ;
}
1995-05-16 11:37:27 +00:00
return TRUE ;
}
Boolean
mediaGetType ( void )
{
dmenuOpenSimple ( & MenuMedia ) ;
return TRUE ;
1995-05-16 02:53:31 +00:00
}
1995-05-16 11:37:27 +00:00
/* Return TRUE if all the media variables are set up correctly */
Boolean
mediaVerify ( void )
{
1995-05-20 00:13:14 +00:00
if ( ! mediaDevice ) {
msgConfirm ( " Media type not set! Please select a media type \n from the Installation menu before proceeding. " ) ;
1995-05-17 14:40:00 +00:00
return FALSE ;
}
1995-05-16 11:37:27 +00:00
return TRUE ;
}