154 lines
3.6 KiB
C
154 lines
3.6 KiB
C
/*
|
|
* $Id: drv_toshiba.c,v 1.3 1999/02/14 09:50:42 dirk Exp $
|
|
*
|
|
* This file is part of WorkMan, the civilized CD player library
|
|
* (c) 1991-1997 by Steven Grimm (original author)
|
|
* (c) by Dirk Försterling (current 'author' = maintainer)
|
|
* The maintainer can be contacted by his e-mail address:
|
|
* milliByte@DeathsDoor.com
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Library General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Library General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Library General Public
|
|
* License along with this library; if not, write to the Free
|
|
* Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
*
|
|
*
|
|
* Vendor-specific drive control routines for Toshiba XM-3401 series.
|
|
*/
|
|
|
|
static char drv_toshiba_id[] = "$Id: drv_toshiba.c,v 1.3 1999/02/14 09:50:42 dirk Exp $";
|
|
|
|
#include <stdio.h>
|
|
#include <errno.h>
|
|
#include "include/wm_config.h"
|
|
#include "include/wm_struct.h"
|
|
#include "include/wm_scsi.h"
|
|
|
|
#define SCMD_TOSH_EJECT 0xc4
|
|
|
|
/* local prototypes */
|
|
static int tosh_init( struct wm_drive *d );
|
|
static int tosh_eject( struct wm_drive *d );
|
|
static int tosh_set_volume( struct wm_drive *d, int left, int right );
|
|
static int tosh_get_volume( struct wm_drive *d, int *left, int *right );
|
|
|
|
struct wm_drive toshiba_proto = {
|
|
-1, /* fd */
|
|
"Toshiba", /* vendor */
|
|
"", /* model */
|
|
"", /* revision */
|
|
NULL, /* aux */
|
|
NULL, /* daux */
|
|
|
|
tosh_init, /* functions... */
|
|
gen_get_trackcount,
|
|
gen_get_cdlen,
|
|
gen_get_trackinfo,
|
|
gen_get_drive_status,
|
|
tosh_get_volume,
|
|
tosh_set_volume,
|
|
gen_pause,
|
|
gen_resume,
|
|
gen_stop,
|
|
gen_play,
|
|
tosh_eject,
|
|
gen_closetray
|
|
};
|
|
|
|
/*
|
|
* Initialize the driver.
|
|
*/
|
|
static int
|
|
tosh_init( struct wm_drive *d )
|
|
{
|
|
extern int min_volume;
|
|
|
|
/* Sun use Toshiba drives as well */
|
|
#if defined(SYSV) && defined(CODEC)
|
|
codec_init();
|
|
#endif
|
|
min_volume = 0;
|
|
return ( 0 );
|
|
}
|
|
|
|
/*
|
|
* Send the Toshiba code to eject the CD.
|
|
*/
|
|
static int
|
|
tosh_eject( struct wm_drive *d )
|
|
{
|
|
return (sendscsi(d, NULL, 0, 0, SCMD_TOSH_EJECT, 1, 0,0,0,0,0,0,0,0,0,0));
|
|
}
|
|
|
|
/*
|
|
* Set the volume. The low end of the scale is more sensitive than the high
|
|
* end, so make up for that by transforming the volume parameters to a square
|
|
* curve.
|
|
*/
|
|
static int
|
|
tosh_set_volume( struct wm_drive *d, int left, int right )
|
|
{
|
|
left = (left * left * left) / 10000;
|
|
right = (right * right * right) / 10000;
|
|
return (gen_set_volume(d, left, right));
|
|
}
|
|
|
|
/*
|
|
* Undo the transformation above using a binary search (so no floating-point
|
|
* math is required.)
|
|
*/
|
|
static int
|
|
unscale_volume(cd_vol, max)
|
|
int cd_vol, max;
|
|
{
|
|
int vol = 0, top = max, bot = 0, scaled = 0;
|
|
|
|
/*cd_vol = (cd_vol * 100 + (max_volume - 1)) / max_volume;*/
|
|
|
|
while (bot <= top)
|
|
{
|
|
vol = (top + bot) / 2;
|
|
scaled = (vol * vol) / max;
|
|
if (cd_vol <= scaled)
|
|
top = vol - 1;
|
|
else
|
|
bot = vol + 1;
|
|
}
|
|
|
|
/* Might have looked down too far for repeated scaled values */
|
|
if (cd_vol < scaled)
|
|
vol++;
|
|
|
|
if (vol < 0)
|
|
vol = 0;
|
|
else if (vol > max)
|
|
vol = max;
|
|
|
|
return (vol);
|
|
}
|
|
|
|
/*
|
|
* Get the volume.
|
|
*/
|
|
static int
|
|
tosh_get_volume( struct wm_drive *d, int *left, int *right )
|
|
{
|
|
int status;
|
|
|
|
status = gen_get_volume(d, left, right);
|
|
if (status < 0)
|
|
return (status);
|
|
*left = unscale_volume(*left, 100);
|
|
*right = unscale_volume(*right, 100);
|
|
|
|
return (0);
|
|
}
|