2023-12-07 20:30:48 +01:00
/*
* 86 Box A hypervisor and IBM PC system emulator that specializes in
* running old operating systems and software designed for IBM
* PC systems and compatibles from 1981 through fairly recent
* system designs based on the PCI bus .
*
* This file is part of the 86 Box distribution .
*
* CD - ROM passthrough support .
*
*
*
* Authors : TheCollector1995 , < mariogplayer @ gmail . com > ,
* Miran Grca , < mgrca8 @ gmail . com >
*
* Copyright 2023 TheCollector1995 .
* Copyright 2023 Miran Grca .
*/
# include <inttypes.h>
# include <stdarg.h>
# include <stdio.h>
# include <stdint.h>
# include <string.h>
# include <stdlib.h>
# include <wchar.h>
# define HAVE_STDARG_H
# include <86box/86box.h>
# include <86box/config.h>
# include <86box/path.h>
# include <86box/plat.h>
# include <86box/plat_cdrom.h>
# include <86box/scsi_device.h>
# include <86box/cdrom.h>
# ifdef ENABLE_CDROM_IOCTL_LOG
int cdrom_ioctl_do_log = ENABLE_CDROM_IOCTL_LOG ;
void
cdrom_ioctl_log ( const char * fmt , . . . )
{
va_list ap ;
if ( cdrom_ioctl_do_log ) {
va_start ( ap , fmt ) ;
pclog_ex ( fmt , ap ) ;
va_end ( ap ) ;
}
}
# else
# define cdrom_ioctl_log(fmt, ...)
# endif
/* The addresses sent from the guest are absolute, ie. a LBA of 0 corresponds to a MSF of 00:00:00. Otherwise, the counter displayed by the guest is wrong:
there is a seeming 2 seconds in which audio plays but counter does not move , while a data track before audio jumps to 2 seconds before the actual start
of the audio while audio still plays . With an absolute conversion , the counter is fine . */
# define MSFtoLBA(m, s, f) ((((m * 60) + s) * 75) + f)
static void
cdrom_ioctl_get_tracks ( UNUSED ( cdrom_t * dev ) , int * first , int * last )
{
2024-05-18 19:42:00 +02:00
TMSF tmsf ;
2023-12-07 20:30:48 +01:00
plat_cdrom_get_audio_tracks ( first , last , & tmsf ) ;
}
static void
cdrom_ioctl_get_track_info ( UNUSED ( cdrom_t * dev ) , uint32_t track , int end , track_info_t * ti )
{
TMSF tmsf ;
plat_cdrom_get_audio_track_info ( end , track , & ti - > number , & tmsf , & ti - > attr ) ;
ti - > m = tmsf . min ;
ti - > s = tmsf . sec ;
ti - > f = tmsf . fr ;
}
static void
cdrom_ioctl_get_subchannel ( UNUSED ( cdrom_t * dev ) , uint32_t lba , subchannel_t * subc )
{
TMSF rel_pos ;
TMSF abs_pos ;
plat_cdrom_get_audio_sub ( lba , & subc - > attr , & subc - > track , & subc - > index ,
& rel_pos , & abs_pos ) ;
subc - > abs_m = abs_pos . min ;
subc - > abs_s = abs_pos . sec ;
subc - > abs_f = abs_pos . fr ;
subc - > rel_m = rel_pos . min ;
subc - > rel_s = rel_pos . sec ;
subc - > rel_f = rel_pos . fr ;
}
static int
2024-05-18 19:42:00 +02:00
cdrom_ioctl_get_capacity ( UNUSED ( cdrom_t * dev ) )
{
int ret ;
ret = plat_cdrom_get_last_block ( ) ;
pclog ( " GetCapacity=%x. \n " , ret ) ;
return ret ;
}
static int
cdrom_ioctl_is_track_audio ( cdrom_t * dev , uint32_t pos , int ismsf )
2023-12-07 20:30:48 +01:00
{
uint8_t attr ;
2024-05-18 19:42:00 +02:00
TMSF tmsf ;
2023-12-07 20:30:48 +01:00
int m ;
int s ;
int f ;
2024-05-18 19:42:00 +02:00
int number ;
2023-12-07 20:30:48 +01:00
int track ;
2024-05-18 19:42:00 +02:00
if ( dev - > cd_status = = CD_STATUS_DATA_ONLY )
return 0 ;
2023-12-07 20:30:48 +01:00
if ( ismsf ) {
m = ( pos > > 16 ) & 0xff ;
s = ( pos > > 8 ) & 0xff ;
f = pos & 0xff ;
pos = MSFtoLBA ( m , s , f ) - 150 ;
}
/* GetTrack requires LBA. */
2024-05-18 19:42:00 +02:00
return plat_cdrom_is_track_audio ( pos ) ;
2023-12-07 20:30:48 +01:00
}
static int
2024-05-18 19:42:00 +02:00
cdrom_ioctl_is_track_pre ( UNUSED ( cdrom_t * dev ) , UNUSED ( uint32_t lba ) )
2023-12-07 20:30:48 +01:00
{
2024-05-18 19:42:00 +02:00
return 0 ;
}
static int
cdrom_ioctl_sector_size ( UNUSED ( cdrom_t * dev ) , uint32_t lba )
{
pclog ( " LBA=%x. \n " , lba ) ;
return plat_cdrom_get_sector_size ( lba ) ;
2023-12-07 20:30:48 +01:00
}
static int
cdrom_ioctl_read_sector ( UNUSED ( cdrom_t * dev ) , int type , uint8_t * b , uint32_t lba )
{
switch ( type ) {
case CD_READ_DATA :
return plat_cdrom_read_sector ( b , 0 , lba ) ;
case CD_READ_AUDIO :
return plat_cdrom_read_sector ( b , 1 , lba ) ;
case CD_READ_RAW :
2024-05-18 19:42:00 +02:00
return plat_cdrom_read_sector ( b , 1 , lba ) ;
2023-12-07 20:30:48 +01:00
default :
cdrom_ioctl_log ( " cdrom_ioctl_read_sector(): Unknown CD read type. \n " ) ;
2024-05-18 19:42:00 +02:00
break ;
2023-12-07 20:30:48 +01:00
}
2024-05-18 19:42:00 +02:00
return 0 ;
2023-12-07 20:30:48 +01:00
}
static int
2024-05-18 19:42:00 +02:00
cdrom_ioctl_track_type ( cdrom_t * dev , uint32_t lba )
2023-12-07 20:30:48 +01:00
{
2024-05-18 19:42:00 +02:00
if ( cdrom_ioctl_is_track_audio ( dev , lba , 0 ) )
2023-12-07 20:30:48 +01:00
return CD_TRACK_AUDIO ;
return 0 ;
}
static void
cdrom_ioctl_exit ( cdrom_t * dev )
{
2024-05-18 19:42:00 +02:00
cdrom_ioctl_log ( " CDROM: ioctl_exit(%s) \n " , dev - > image_path ) ;
2023-12-07 20:30:48 +01:00
dev - > cd_status = CD_STATUS_EMPTY ;
2024-05-18 19:42:00 +02:00
plat_cdrom_close ( ) ;
2023-12-07 20:30:48 +01:00
dev - > ops = NULL ;
}
static const cdrom_ops_t cdrom_ioctl_ops = {
cdrom_ioctl_get_tracks ,
cdrom_ioctl_get_track_info ,
cdrom_ioctl_get_subchannel ,
2024-05-18 19:42:00 +02:00
cdrom_ioctl_is_track_pre ,
2023-12-07 20:30:48 +01:00
cdrom_ioctl_sector_size ,
cdrom_ioctl_read_sector ,
cdrom_ioctl_track_type ,
cdrom_ioctl_exit
} ;
void
2024-05-18 19:42:00 +02:00
cdrom_ioctl_close ( cdrom_t * dev )
2023-12-07 20:30:48 +01:00
{
2024-05-18 19:42:00 +02:00
cdrom_ioctl_log ( " CDROM: ioctl_close(%s) \n " , dev - > image_path ) ;
2023-12-07 20:30:48 +01:00
2024-05-18 19:42:00 +02:00
if ( dev & & dev - > ops & & dev - > ops - > exit )
dev - > ops - > exit ( dev ) ;
2023-12-07 20:30:48 +01:00
}
static int
cdrom_ioctl_open_abort ( cdrom_t * dev )
{
2024-05-18 19:42:00 +02:00
cdrom_ioctl_close ( dev ) ;
2023-12-07 20:30:48 +01:00
dev - > ops = NULL ;
dev - > host_drive = 0 ;
2024-05-18 19:42:00 +02:00
dev - > image_path [ 0 ] = 0 ;
2023-12-07 20:30:48 +01:00
return 1 ;
}
2024-05-18 22:37:49 +02:00
void
cdrom_ioctl_eject ( void )
{
plat_cdrom_eject ( ) ;
}
2023-12-07 20:30:48 +01:00
int
2024-05-18 22:37:49 +02:00
cdrom_ioctl_open ( cdrom_t * dev , int drive )
2023-12-07 20:30:48 +01:00
{
2024-05-18 19:42:00 +02:00
/* Open the image. */
2024-05-18 22:37:49 +02:00
int i = plat_cdrom_set_drive ( drive ) ;
2024-05-18 19:42:00 +02:00
if ( ! i )
2023-12-07 20:30:48 +01:00
return cdrom_ioctl_open_abort ( dev ) ;
/* All good, reset state. */
dev - > cd_status = CD_STATUS_STOPPED ;
dev - > is_dir = 0 ;
dev - > seek_pos = 0 ;
dev - > cd_buflen = 0 ;
2024-05-18 19:42:00 +02:00
dev - > cdrom_capacity = cdrom_ioctl_get_capacity ( dev ) ;
2023-12-07 20:30:48 +01:00
pclog ( " CD-ROM capacity: %i sectors (% " PRIi64 " bytes) \n " , dev - > cdrom_capacity , ( ( uint64_t ) dev - > cdrom_capacity ) < < 11ULL ) ;
/* Attach this handler to the drive. */
dev - > ops = & cdrom_ioctl_ops ;
return 0 ;
}
2024-05-18 19:42:00 +02:00