Commit 8151e895 authored by Linus Torvalds's avatar Linus Torvalds

Import 1.1.32

parent 3dd3f561
VERSION = 1 VERSION = 1
PATCHLEVEL = 1 PATCHLEVEL = 1
SUBLEVEL = 31 SUBLEVEL = 32
all: Version zImage all: Version zImage
......
...@@ -81,7 +81,6 @@ if [ "$CONFIG_SLIP" = "y" ]; then ...@@ -81,7 +81,6 @@ if [ "$CONFIG_SLIP" = "y" ]; then
fi fi
bool 'PPP (point-to-point) support' CONFIG_PPP n bool 'PPP (point-to-point) support' CONFIG_PPP n
bool 'PLIP (parallel port) support' CONFIG_PLIP n bool 'PLIP (parallel port) support' CONFIG_PLIP n
bool 'SK_G16 support' CONFIG_SK_G16 n
bool 'Load balancing support (experimental)' CONFIG_SLAVE_BALANCING n bool 'Load balancing support (experimental)' CONFIG_SLAVE_BALANCING n
bool 'Do you want to be offered ALPHA test drivers' CONFIG_NET_ALPHA n bool 'Do you want to be offered ALPHA test drivers' CONFIG_NET_ALPHA n
bool 'Western Digital/SMC cards' CONFIG_NET_VENDOR_SMC n bool 'Western Digital/SMC cards' CONFIG_NET_VENDOR_SMC n
...@@ -110,6 +109,7 @@ if [ "$CONFIG_NET_ISA" = "y" ]; then ...@@ -110,6 +109,7 @@ if [ "$CONFIG_NET_ISA" = "y" ]; then
fi fi
bool 'HP PCLAN support' CONFIG_HPLAN n bool 'HP PCLAN support' CONFIG_HPLAN n
bool 'NE2000/NE1000 support' CONFIG_NE2000 y bool 'NE2000/NE1000 support' CONFIG_NE2000 y
bool 'SK_G16 support' CONFIG_SK_G16 n
fi fi
bool 'EISA and on board controllers' CONFIG_NET_EISA n bool 'EISA and on board controllers' CONFIG_NET_EISA n
if [ "$CONFIG_NET_ALPHA" = "y" ]; then if [ "$CONFIG_NET_ALPHA" = "y" ]; then
......
IDE Performance Enhancements Version 2.0
============================ ===========
This version of hd.c includes support for two optional features:
(1) The disk I/O routines can now run with interrupts unmasked
most of the time, making them much friendlier to high
speed serial ports and other system activity.
(2) Support is included for IDE "Multiple Sector Mode", the use
of which can reduce disk I/O kernel overhead by 10-30%
on many systems, with a corresponding 10-20% increase in
data throughput.
By default, both features are DISABLED, for compatibility with
systems on which they may cause troubles.
The IRQ unmasking has been known to CORRUPT FILESYSTEMS in the
past on systems with strange hard drives. Backup before trying!
It works on most systems, but use at your own risk!!
Drives which support "Multiple Sector Mode" are identified by the
kernel at boot time, and a message is displayed indicating the
largest possible setting for "MaxMult". I recommend using settings
of 8, 16, or 32. Many drives also support non-powers of two,
but many other drives do not -- try strange values at your own risk!
For more detailed boot-time information about your drive, change
the definition of VERBOSE_DRIVE_INFO from 0 to 1 near the top
of hd.c and rebuild/reinstall the kernel.
Some drives (mostly older CONNER drives) do not implement multiple mode
correctly, and data corruption may occur.. but if you wait long enough
the error recovery logic *should* be able to recover eventually.
To try this out more safely, mount the drive's partitions read-only
before using hdparm (see below) for the first time. If it doesn't
work, email me (mlord@bnr.ca) with the drive name as displayed at
boot time, so I can warn others and possibly add a hook to the code.
To enable the features, a small program is included: hdparm.c
This one is *different* from previous versions -- be sure to recompile it!
Compile this using cc -O -o /usr/bin/hdparm hdparm.c
and then use it to enable/disable the new features, as follows:
To turn on 16-sector multiple mode, with interrupt unmasking:
hdparm /dev/hda 16 1
To view the current settings:
hdparm /dev/hda
If you have more than one drive, a separate command would need to be issued for the second drive as well, using the same or different
settings as desired:
hdparm /dev/hdb 16 1
To turn off both features on the first drive, use:
hdparm /dev/hda 0 0
To benchmark the performance difference, try:
hdparm /dev/hda 0 0
sync
time dd if=/dev/hda of=/dev/null bs=1024k count=30
time dd if=/dev/hda of=/dev/null bs=1024k count=30
time dd if=/dev/hda of=/dev/null bs=1024k count=30
hdparm /dev/hda 16 1
sync
time dd if=/dev/hda of=/dev/null bs=1024k count=30
time dd if=/dev/hda of=/dev/null bs=1024k count=30
time dd if=/dev/hda of=/dev/null bs=1024k count=30
This gives before and after views. Compare the total elapsed times,
as well as the percent of CPU used in each case. Run several trials
to ensure/verify consistent results. Some drives are actually *slower*
with multiple mode enabled, but those are very rare indeed. Most systems
experience a 10-30% increase in throughput, with a corresponding 5-50%
decrease in kernel/system CPU usage.
If you are using linux kernel 1.1.4 or higher (with the "cluster" code),
then you may not notice a big difference with the above tests. However,
I have noticed that the iozone benchmark program does seem to work fairly
reliably under kernels with the "cluster" code, so you could try that
instead (under older kernels, iozone seems to give wildly varying results
from trial to trial, at least on my system).
To have your favorite settings installed automatically at boot time,
place the hdparm command(s) into the /etc/rc.d/rc.local file.
Alternatively, one could modify the DEFAULTs near the top of hd.c
and rebuild and install the new kernel.
Enjoy,
mlord@bnr.ca
This diff is collapsed.
...@@ -83,6 +83,7 @@ static struct termios *console_termios_locked[NR_CONSOLES]; ...@@ -83,6 +83,7 @@ static struct termios *console_termios_locked[NR_CONSOLES];
int set_selection(const int arg); int set_selection(const int arg);
int paste_selection(struct tty_struct *tty); int paste_selection(struct tty_struct *tty);
static void clear_selection(void); static void clear_selection(void);
static void highlight_pointer(const int currcons, const int where);
/* Variables for selection control. */ /* Variables for selection control. */
#define SEL_BUFFER_SIZE 4096 #define SEL_BUFFER_SIZE 4096
...@@ -1703,6 +1704,9 @@ void update_screen(int new_console) ...@@ -1703,6 +1704,9 @@ void update_screen(int new_console)
return; return;
lock = 1; lock = 1;
kbdsave(new_console); kbdsave(new_console);
#ifdef CONFIG_SELECTION
highlight_pointer(fg_console,-1);
#endif /* CONFIG_SELECTION */
get_scrmem(fg_console); get_scrmem(fg_console);
fg_console = new_console; fg_console = new_console;
set_scrmem(fg_console); set_scrmem(fg_console);
...@@ -1778,8 +1782,49 @@ static void highlight(const int currcons, const int s, const int e) ...@@ -1778,8 +1782,49 @@ static void highlight(const int currcons, const int s, const int e)
*p = (*p & 0x88) | ((*p << 4) & 0x70) | ((*p >> 4) & 0x07); *p = (*p & 0x88) | ((*p << 4) & 0x70) | ((*p >> 4) & 0x07);
} }
/* is c in range [a-zA-Z0-9_]? */ /* use complementary color to show the pointer */
static inline int inword(const char c) { return (isalnum(c) || c == '_'); } static void highlight_pointer(const int currcons, const int where)
{
unsigned char *p;
static char *prev=NULL;
if (where==-1) /* remove the pointer */
{
if (prev)
{
*prev ^= 0x77;
prev=NULL;
}
}
else
{
p = (unsigned char *)origin - hwscroll_offset + where + 1;
*p ^= 0x77;
if (prev) *prev ^= 0x77; /* remove the previous one */
prev=p;
}
}
/*
* This function uses a 128-bit look up table
*/
static unsigned long inwordLut[4]={
0x00000000, /* control chars */
0x03FF0000, /* digits */
0x87FFFFFE, /* uppercase and '_' */
0x07FFFFFE /* lowercase */
};
static inline int inword(const char c) {
return ( inwordLut[(c>>5)&3] >> (c&0x1F) ) & 1;
}
/* set inwordLut conntents. Invoked by ioctl(). */
int sel_loadlut(const int arg)
{
memcpy_fromfs(inwordLut,(unsigned long *)(arg+4),16);
return 0;
}
/* does screen address p correspond to character at LH/RH edge of screen? */ /* does screen address p correspond to character at LH/RH edge of screen? */
static inline int atedge(const int p) static inline int atedge(const int p)
...@@ -1828,7 +1873,6 @@ int set_selection(const int arg) ...@@ -1828,7 +1873,6 @@ int set_selection(const int arg)
switch (sel_mode) switch (sel_mode)
{ {
case 0: /* character-by-character selection */ case 0: /* character-by-character selection */
default:
new_sel_start = ps; new_sel_start = ps;
new_sel_end = pe; new_sel_end = pe;
break; break;
...@@ -1859,6 +1903,17 @@ int set_selection(const int arg) ...@@ -1859,6 +1903,17 @@ int set_selection(const int arg)
new_sel_end = pe + video_size_row new_sel_end = pe + video_size_row
- pe % video_size_row - 2; - pe % video_size_row - 2;
break; break;
case 3: /* pointer highlight */
if (sel_cons != currcons)
{
highlight_pointer(sel_cons,-1);
clear_selection();
sel_cons = currcons;
}
highlight_pointer(sel_cons,pe);
return 0; /* nothing more */
default:
return -EINVAL;
} }
/* select to end of line if on trailing space */ /* select to end of line if on trailing space */
if (new_sel_end > new_sel_start && if (new_sel_end > new_sel_start &&
...@@ -1962,6 +2017,7 @@ int paste_selection(struct tty_struct *tty) ...@@ -1962,6 +2017,7 @@ int paste_selection(struct tty_struct *tty)
the selection. */ the selection. */
static void clear_selection() static void clear_selection()
{ {
highlight_pointer(sel_cons, -1); /* hide the pointer */
if (sel_start != -1) if (sel_start != -1)
{ {
highlight(sel_cons, sel_start, sel_end); highlight(sel_cons, sel_start, sel_end);
......
...@@ -86,7 +86,12 @@ static unsigned long key_down[8] = { 0, }; ...@@ -86,7 +86,12 @@ static unsigned long key_down[8] = { 0, };
static int want_console = -1; static int want_console = -1;
static int last_console = 0; /* last used VC */ static int last_console = 0; /* last used VC */
static int dead_key_next = 0; static int dead_key_next = 0;
static int shift_state = 0; /*
* In order to retrieve the shift_state (for the mouse server), either
* the variable must be global, or a new procedure must be create to
* return the value. I chose the former way.
*/
/*static*/ int shift_state = 0;
static int npadch = -1; /* -1 or number assembled on pad */ static int npadch = -1; /* -1 or number assembled on pad */
static unsigned char diacr = 0; static unsigned char diacr = 0;
static char rep = 0; /* flag telling character repeat */ static char rep = 0; /* flag telling character repeat */
......
...@@ -420,7 +420,7 @@ long chr_dev_init(long mem_start, long mem_end) ...@@ -420,7 +420,7 @@ long chr_dev_init(long mem_start, long mem_end)
#ifdef CONFIG_FTAPE #ifdef CONFIG_FTAPE
/* allocate NR_FTAPE_BUFFERS 32Kb buffers at aligned address */ /* allocate NR_FTAPE_BUFFERS 32Kb buffers at aligned address */
ftape_big_buffer= (char*) ((mem_start + 0x7fff) & ~0x7fff); ftape_big_buffer= (char*) ((mem_start + 0x7fff) & ~0x7fff);
printk( "ftape: allocated %d buffers alligned at: %p\n", printk( "ftape: allocated %d buffers aligned at: %p\n",
NR_FTAPE_BUFFERS, ftape_big_buffer); NR_FTAPE_BUFFERS, ftape_big_buffer);
mem_start = (long) ftape_big_buffer + NR_FTAPE_BUFFERS * 0x8000; mem_start = (long) ftape_big_buffer + NR_FTAPE_BUFFERS * 0x8000;
#endif #endif
......
...@@ -68,6 +68,8 @@ ...@@ -68,6 +68,8 @@
#ifdef CONFIG_SELECTION #ifdef CONFIG_SELECTION
extern int set_selection(const int arg); extern int set_selection(const int arg);
extern int paste_selection(struct tty_struct *tty); extern int paste_selection(struct tty_struct *tty);
extern int sel_loadlut(const int arg);
extern int shift_state;
#endif /* CONFIG_SELECTION */ #endif /* CONFIG_SELECTION */
extern int do_screendump(int arg); extern int do_screendump(int arg);
...@@ -1385,6 +1387,11 @@ static int tty_ioctl(struct inode * inode, struct file * file, ...@@ -1385,6 +1387,11 @@ static int tty_ioctl(struct inode * inode, struct file * file,
case 4: case 4:
unblank_screen(); unblank_screen();
return 0; return 0;
case 5:
return sel_loadlut(arg);
case 6:
put_fs_byte(shift_state,arg);
return 0;
#endif /* CONFIG_SELECTION */ #endif /* CONFIG_SELECTION */
default: default:
return -EINVAL; return -EINVAL;
......
...@@ -54,7 +54,9 @@ extern int el16_probe(struct device *); ...@@ -54,7 +54,9 @@ extern int el16_probe(struct device *);
extern int elplus_probe(struct device *); extern int elplus_probe(struct device *);
extern int ac3200_probe(struct device *); extern int ac3200_probe(struct device *);
extern int e2100_probe(struct device *); extern int e2100_probe(struct device *);
extern int SK_init(struct device *dev); extern int ni52_probe(struct device *);
extern int ni65_probe(struct device *);
extern int SK_init(struct device *);
/* Detachable devices ("pocket adaptors" and special PCMCIA drivers). */ /* Detachable devices ("pocket adaptors" and special PCMCIA drivers). */
extern int atp_init(struct device *); extern int atp_init(struct device *);
...@@ -129,6 +131,12 @@ ethif_probe(struct device *dev) ...@@ -129,6 +131,12 @@ ethif_probe(struct device *dev)
#endif #endif
#if defined(CONFIG_SK_G16) #if defined(CONFIG_SK_G16)
&& SK_init(dev) && SK_init(dev)
#endif
#ifdef CONFIG_NI52
&& ni52_probe(dev)
#endif
#ifdef CONFIG_NI65
&& ni65_probe(dev)
#endif #endif
&& 1 ) { && 1 ) {
return 1; /* -ENODEV or -EAGAIN would be more accurate. */ return 1; /* -ENODEV or -EAGAIN would be more accurate. */
......
...@@ -267,7 +267,6 @@ ppp_init(struct device *dev) ...@@ -267,7 +267,6 @@ ppp_init(struct device *dev)
dev->mtu = PPP_MTU; dev->mtu = PPP_MTU;
dev->hard_start_xmit = ppp_xmit; dev->hard_start_xmit = ppp_xmit;
dev->open = ppp_dev_open; dev->open = ppp_dev_open;
dev->do_ioctl = ppp_dev_ioctl;
dev->stop = ppp_dev_close; dev->stop = ppp_dev_close;
dev->get_stats = ppp_get_stats; dev->get_stats = ppp_get_stats;
dev->hard_header = ppp_header; dev->hard_header = ppp_header;
...@@ -280,6 +279,8 @@ ppp_init(struct device *dev) ...@@ -280,6 +279,8 @@ ppp_init(struct device *dev)
#ifdef NET02D #ifdef NET02D
dev->add_arp = ppp_add_arp; dev->add_arp = ppp_add_arp;
dev->queue_xmit = dev_queue_xmit; dev->queue_xmit = dev_queue_xmit;
#else
dev->do_ioctl = ppp_dev_ioctl;
#endif #endif
for (i = 0; i < DEV_NUMBUFFS; i++) for (i = 0; i < DEV_NUMBUFFS; i++)
...@@ -605,6 +606,7 @@ ppp_dev_close(struct device *dev) ...@@ -605,6 +606,7 @@ ppp_dev_close(struct device *dev)
return 0; return 0;
} }
#ifndef NET02D
static int ppp_dev_ioctl(struct device *dev, struct ifreq *ifr) static int ppp_dev_ioctl(struct device *dev, struct ifreq *ifr)
{ {
struct ppp *ppp = &ppp_ctrl[dev->base_addr]; struct ppp *ppp = &ppp_ctrl[dev->base_addr];
...@@ -630,6 +632,7 @@ static int ppp_dev_ioctl(struct device *dev, struct ifreq *ifr) ...@@ -630,6 +632,7 @@ static int ppp_dev_ioctl(struct device *dev, struct ifreq *ifr)
return error; return error;
} }
#endif
/************************************************************* /*************************************************************
* TTY OUTPUT * TTY OUTPUT
...@@ -650,7 +653,11 @@ ppp_output_done (void *ppp) ...@@ -650,7 +653,11 @@ ppp_output_done (void *ppp)
/* If the device is still up then enable the transmitter of the /* If the device is still up then enable the transmitter of the
next frame. */ next frame. */
if (((struct ppp *) ppp)->dev->flags & IFF_UP) if (((struct ppp *) ppp)->dev->flags & IFF_UP)
#ifndef NET02D
mark_bh (NET_BH);
#else
dev_tint (((struct ppp *) ppp)->dev); dev_tint (((struct ppp *) ppp)->dev);
#endif
/* enable any blocked process pending transmission */ /* enable any blocked process pending transmission */
wake_up_interruptible (&((struct ppp *) ppp)->write_wait); wake_up_interruptible (&((struct ppp *) ppp)->write_wait);
...@@ -1124,9 +1131,7 @@ ppp_do_ip (struct ppp *ppp, unsigned short proto, unsigned char *c, ...@@ -1124,9 +1131,7 @@ ppp_do_ip (struct ppp *ppp, unsigned short proto, unsigned char *c,
} }
/* receive the frame through the network software */ /* receive the frame through the network software */
while ((dev_rint(c, count, 0, ppp->dev) & ~1) != 0) (void) dev_rint (c, count, 0, ppp->dev);
;
return 1; return 1;
} }
......
...@@ -5,7 +5,7 @@ ...@@ -5,7 +5,7 @@
# #
# #
VERSION = 2.90 VERSION = 2.90-2
TARGET_OS = linux TARGET_OS = linux
.c.s: .c.s:
......
...@@ -12,8 +12,8 @@ VoxWare v2.90 release notes ...@@ -12,8 +12,8 @@ VoxWare v2.90 release notes
there are some new features required by a popular there are some new features required by a popular
application. In addition there is also support application. In addition there is also support
for the GUS MAX and the 16 bit sampling option of GUS. for the GUS MAX and the 16 bit sampling option of GUS.
Also the Windows Sound System stuff is there but may not
work yet (may work with some WSS compatible cards). The MSS/WSS support works now. At least with SG NX Pro 16.
********* IMPORTANT ***************************************** ********* IMPORTANT *****************************************
Linux 1.0 or later is required to by this driver version. Linux 1.0 or later is required to by this driver version.
...@@ -65,6 +65,7 @@ contributors. (I could have forgotten some names.) ...@@ -65,6 +65,7 @@ contributors. (I could have forgotten some names.)
Markus Aroharju and Markus Aroharju and
Risto Kankkunen Major contributions to the mixer support Risto Kankkunen Major contributions to the mixer support
of GUS v3.7. of GUS v3.7.
Hunyue Yau Mixer support for SG NX Pro.
Marc Hoffman PSS support. Marc Hoffman PSS support.
Regards, Regards,
......
...@@ -903,16 +903,14 @@ ad1848_interrupt (int irq) ...@@ -903,16 +903,14 @@ ad1848_interrupt (int irq)
int int
probe_ms_sound (struct address_info *hw_config) probe_ms_sound (struct address_info *hw_config)
{ {
int config_port = hw_config->io_base + 0, version_port = hw_config->io_base + 3;
if ((INB (hw_config->io_base + 3) & 0x04) == 0) if ((INB (hw_config->io_base + 3) & 0x04) == 0)
return 0; /* WSS ID test failed */ return 0; /* WSS ID test failed */
if (hw_config->irq > 11) if (hw_config->irq > 11)
return; return 0;
if (hw_config->dma > 3 || hw_config->dma == 2) if (hw_config->dma != 0 && hw_config->dma != 1 && hw_config->dma != 3)
return; return 0;
return ad1848_detect (hw_config->io_base + 4); return ad1848_detect (hw_config->io_base + 4);
} }
...@@ -920,10 +918,12 @@ probe_ms_sound (struct address_info *hw_config) ...@@ -920,10 +918,12 @@ probe_ms_sound (struct address_info *hw_config)
long long
attach_ms_sound (long mem_start, struct address_info *hw_config) attach_ms_sound (long mem_start, struct address_info *hw_config)
{ {
static unsigned char interrupt_bits[11] = static unsigned char interrupt_bits[12] =
{-1, -1, -1, -1, -1, -1, -1, 0x08, -1, 0x10, 0x18, 0x20}; {-1, -1, -1, -1, -1, -1, -1, 0x08, -1, 0x10, 0x18, 0x20};
char bits; char bits;
static unsigned char dma_bits[4] = {1, 2, 0, 3};
int config_port = hw_config->io_base + 0, version_port = hw_config->io_base + 3; int config_port = hw_config->io_base + 0, version_port = hw_config->io_base + 3;
if (!ad1848_detect (hw_config->io_base + 4)) if (!ad1848_detect (hw_config->io_base + 4))
...@@ -941,7 +941,7 @@ attach_ms_sound (long mem_start, struct address_info *hw_config) ...@@ -941,7 +941,7 @@ attach_ms_sound (long mem_start, struct address_info *hw_config)
if ((INB (version_port) & 0x40) == 0) if ((INB (version_port) & 0x40) == 0)
printk ("[IRQ?]"); printk ("[IRQ?]");
OUTB (bits | hw_config->dma, config_port); /* Write IRQ+DMA setup */ OUTB (bits | dma_bits[hw_config->dma], config_port); /* Write IRQ+DMA setup */
ad1848_init ("MS Sound System", hw_config->io_base + 4, ad1848_init ("MS Sound System", hw_config->io_base + 4,
hw_config->irq, hw_config->irq,
......
...@@ -322,6 +322,13 @@ main (int argc, char *argv[]) ...@@ -322,6 +322,13 @@ main (int argc, char *argv[])
} }
} }
if (selected_options & B (OPT_SBPRO))
{
fprintf(stderr, "Do you want support for the mixer of SG NX Pro ? ");
if (think_positively (0))
printf("#define __SGNXPRO__\n");
}
if (selected_options & B (OPT_SB16)) if (selected_options & B (OPT_SB16))
selected_options |= B (OPT_SBPRO); selected_options |= B (OPT_SBPRO);
...@@ -333,7 +340,7 @@ main (int argc, char *argv[]) ...@@ -333,7 +340,7 @@ main (int argc, char *argv[])
"if you wish to emulate the soundblaster and you have a DSPxxx.LD.\n" "if you wish to emulate the soundblaster and you have a DSPxxx.LD.\n"
"then you must include the LD in the kernel.\n" "then you must include the LD in the kernel.\n"
"(do you wish to include a LD) ? "); "(do you wish to include a LD) ? ");
if (think_positively (1)) if (think_positively (0))
{ {
char path[512]; char path[512];
......
...@@ -278,6 +278,9 @@ struct sound_timer_operations { ...@@ -278,6 +278,9 @@ struct sound_timer_operations {
#endif #endif
#ifndef EXCLUDE_MSS #ifndef EXCLUDE_MSS
{SNDCARD_MSS, {MSS_BASE, MSS_IRQ, MSS_DMA}, SND_DEFAULT_ENABLE}, {SNDCARD_MSS, {MSS_BASE, MSS_IRQ, MSS_DMA}, SND_DEFAULT_ENABLE},
# ifdef MSS2_BASE
{SNDCARD_MSS, {MSS2_BASE, MSS2_IRQ, MSS2_DMA}, SND_DEFAULT_ENABLE},
# endif
#endif #endif
#if !defined(EXCLUDE_UART6850) && !defined(EXCLUDE_MIDI) #if !defined(EXCLUDE_UART6850) && !defined(EXCLUDE_MIDI)
......
...@@ -320,7 +320,7 @@ DMAbuf_getrdbuffer (int dev, char **buf, int *len) ...@@ -320,7 +320,7 @@ DMAbuf_getrdbuffer (int dev, char **buf, int *len)
if (!(dmap->flags & DMA_ALLOC_DONE)) if (!(dmap->flags & DMA_ALLOC_DONE))
reorganize_buffers (dev); reorganize_buffers (dev);
if (dmap->dma_mode) if (!dmap->dma_mode)
{ {
int err; int err;
......
/* /*
* sound/sb_dsp.c * sound/sb_dsp.c
* *
* The low level driver for the SoundBlaster DSP chip. * The low level driver for the SoundBlaster DSP chip (SB1.0 to 2.1, SB Pro).
* *
* Copyright by Hannu Savolainen 1993 * Copyright by Hannu Savolainen 1994
* *
* Redistribution and use in source and binary forms, with or without * Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are * modification, are permitted provided that the following conditions are
...@@ -25,6 +25,10 @@ ...@@ -25,6 +25,10 @@
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE. * SUCH DAMAGE.
* *
* Modified:
* Hunyue Yau Jan 6 1994
* Added code to support Sound Galaxy NX Pro
*
*/ */
#include "sound_config.h" #include "sound_config.h"
...@@ -756,6 +760,7 @@ long ...@@ -756,6 +760,7 @@ long
sb_dsp_init (long mem_start, struct address_info *hw_config) sb_dsp_init (long mem_start, struct address_info *hw_config)
{ {
int i; int i;
int mixer_type = 0;
sbc_major = sbc_minor = 0; sbc_major = sbc_minor = 0;
sb_dsp_command (0xe1); /* sb_dsp_command (0xe1); /*
...@@ -786,7 +791,7 @@ sb_dsp_init (long mem_start, struct address_info *hw_config) ...@@ -786,7 +791,7 @@ sb_dsp_init (long mem_start, struct address_info *hw_config)
#ifndef EXCLUDE_SBPRO #ifndef EXCLUDE_SBPRO
if (sbc_major >= 3) if (sbc_major >= 3)
sb_mixer_init (sbc_major); mixer_type = sb_mixer_init (sbc_major);
#endif #endif
#ifndef EXCLUDE_YM8312 #ifndef EXCLUDE_YM8312
...@@ -799,7 +804,16 @@ sb_dsp_init (long mem_start, struct address_info *hw_config) ...@@ -799,7 +804,16 @@ sb_dsp_init (long mem_start, struct address_info *hw_config)
if (sbc_major >= 3) if (sbc_major >= 3)
{ {
#ifndef SCO #ifndef SCO
sprintf (sb_dsp_operations.name, "SoundBlaster Pro %d.%d", sbc_major, sbc_minor); # ifdef __SGNXPRO__
if (mixer_type == 2)
{
sprintf (sb_dsp_operations.name, "Sound Galaxy NX Pro %d.%d", sbc_major, sbc_minor);
}
else
# endif
{
sprintf (sb_dsp_operations.name, "SoundBlaster Pro %d.%d", sbc_major, sbc_minor);
}
#endif #endif
} }
else else
......
...@@ -4,7 +4,7 @@ ...@@ -4,7 +4,7 @@
* *
* The low level mixer driver for the SoundBlaster Pro and SB16 cards. * The low level mixer driver for the SoundBlaster Pro and SB16 cards.
* *
* Copyright by Hannu Savolainen 1993 * Copyright by Hannu Savolainen 1994
* *
* Redistribution and use in source and binary forms, with or without * Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are * modification, are permitted provided that the following conditions are
...@@ -26,6 +26,10 @@ ...@@ -26,6 +26,10 @@
* OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE. * SUCH DAMAGE.
* *
* Modified:
* Hunyue Yau Jan 6 1994
* Added code to support the Sound Galaxy NX Pro mixer.
*
*/ */
#include "sound_config.h" #include "sound_config.h"
...@@ -91,9 +95,21 @@ sb_mixer_set_stereo (int mode) ...@@ -91,9 +95,21 @@ sb_mixer_set_stereo (int mode)
| (mode ? STEREO_DAC : MONO_DAC))); | (mode ? STEREO_DAC : MONO_DAC)));
} }
/*
* Returns:
* 0 No mixer detected.
* 1 Only a plain Sound Blaster Pro style mixer detected.
* 2 The Sound Galaxy NX Pro mixer detected.
*/
static int static int
detect_mixer (void) detect_mixer (void)
{ {
#ifdef __SGNXPRO__
int oldbass, oldtreble;
#endif
int retcode = 1;
/* /*
* Detect the mixer by changing parameters of two volume channels. If the * Detect the mixer by changing parameters of two volume channels. If the
* values read back match with the values written, the mixer is there (is * values read back match with the values written, the mixer is there (is
...@@ -109,7 +125,30 @@ detect_mixer (void) ...@@ -109,7 +125,30 @@ detect_mixer (void)
if (sb_getmixer (VOC_VOL) != 0x33) if (sb_getmixer (VOC_VOL) != 0x33)
return 0; return 0;
return 1; #ifdef __SGNXPRO__
/* Attempt to detect the SG NX Pro by check for valid bass/treble
* registers.
*/
oldbass = sb_getmixer (BASS_LVL);
oldtreble = sb_getmixer (TREBLE_LVL);
sb_setmixer (BASS_LVL, 0xaa);
sb_setmixer (TREBLE_LVL, 0x55);
if ((sb_getmixer (BASS_LVL) != 0xaa) ||
(sb_getmixer (TREBLE_LVL) != 0x55))
{
retcode = 1; /* 1 == Only SB Pro detected */
}
else
retcode = 2; /* 2 == SG NX Pro detected */
/* Restore register in either case since SG NX Pro has EEPROM with
* 'preferred' values stored.
*/
sb_setmixer (BASS_LVL, oldbass);
sb_setmixer (TREBLE_LVL, oldtreble);
#endif
return retcode;
} }
static void static void
...@@ -350,17 +389,23 @@ sb_mixer_reset (void) ...@@ -350,17 +389,23 @@ sb_mixer_reset (void)
set_recmask (SOUND_MASK_MIC); set_recmask (SOUND_MASK_MIC);
} }
void /*
* Returns a code depending on whether a SG NX Pro was detected.
* 1 == Plain SB Pro
* 2 == SG NX Pro detected.
* 3 == SB16
*
* Used to update message.
*/
int
sb_mixer_init (int major_model) sb_mixer_init (int major_model)
{ {
sb_setmixer (0x00, 0); /* int mixer_type = 0;
* Reset mixer
*/
if (!detect_mixer ()) sb_setmixer (0x00, 0); /* Reset mixer */
return; /*
* No mixer. Why? if (!(mixer_type = detect_mixer ()))
*/ return 0; /* No mixer. Why? */
mixer_initialized = 1; mixer_initialized = 1;
mixer_model = major_model; mixer_model = major_model;
...@@ -369,9 +414,21 @@ sb_mixer_init (int major_model) ...@@ -369,9 +414,21 @@ sb_mixer_init (int major_model)
{ {
case 3: case 3:
mixer_caps = SOUND_CAP_EXCL_INPUT; mixer_caps = SOUND_CAP_EXCL_INPUT;
supported_devices = SBPRO_MIXER_DEVICES; #ifdef __SGNXPRO__
supported_rec_devices = SBPRO_RECORDING_DEVICES; if (mixer_type == 2) /* A SGNXPRO was detected */
iomap = &sbpro_mix; {
supported_devices = SGNXPRO_MIXER_DEVICES;
supported_rec_devices = SGNXPRO_RECORDING_DEVICES;
iomap = &sgnxpro_mix;
}
else
#endif
{
supported_devices = SBPRO_MIXER_DEVICES;
supported_rec_devices = SBPRO_RECORDING_DEVICES;
iomap = &sbpro_mix;
mixer_type = 1;
}
break; break;
case 4: case 4:
...@@ -379,16 +436,18 @@ sb_mixer_init (int major_model) ...@@ -379,16 +436,18 @@ sb_mixer_init (int major_model)
supported_devices = SB16_MIXER_DEVICES; supported_devices = SB16_MIXER_DEVICES;
supported_rec_devices = SB16_RECORDING_DEVICES; supported_rec_devices = SB16_RECORDING_DEVICES;
iomap = &sb16_mix; iomap = &sb16_mix;
mixer_type = 3;
break; break;
default: default:
printk ("SB Warning: Unsupported mixer type\n"); printk ("SB Warning: Unsupported mixer type\n");
return; return 0;
} }
if (num_mixers < MAX_MIXER_DEV) if (num_mixers < MAX_MIXER_DEV)
mixer_devs[num_mixers++] = &sb_mixer_operations; mixer_devs[num_mixers++] = &sb_mixer_operations;
sb_mixer_reset (); sb_mixer_reset ();
return mixer_type;
} }
#endif #endif
...@@ -24,13 +24,29 @@ ...@@ -24,13 +24,29 @@
* LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY * 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 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE. * SUCH DAMAGE.
*
* Modified:
* Hunyue Yau Jan 6 1994
* Added defines for the Sound Galaxy NX Pro mixer.
* *
*/ */
#define SBPRO_RECORDING_DEVICES (SOUND_MASK_LINE | SOUND_MASK_MIC | SOUND_MASK_CD) #define SBPRO_RECORDING_DEVICES (SOUND_MASK_LINE | SOUND_MASK_MIC | SOUND_MASK_CD)
/* Same as SB Pro, unless I find otherwise */
#define SGNXPRO_RECORDING_DEVICES SBPRO_RECORDING_DEVICES
#define SBPRO_MIXER_DEVICES (SOUND_MASK_SYNTH | SOUND_MASK_PCM | SOUND_MASK_LINE | SOUND_MASK_MIC | \ #define SBPRO_MIXER_DEVICES (SOUND_MASK_SYNTH | SOUND_MASK_PCM | SOUND_MASK_LINE | SOUND_MASK_MIC | \
SOUND_MASK_CD | SOUND_MASK_VOLUME) SOUND_MASK_CD | SOUND_MASK_VOLUME)
/* SG NX Pro has treble and bass settings on the mixer. The 'speaker'
* channel is the COVOX/DisneySoundSource emulation volume control
* on the mixer. It does NOT control speaker volume. Should have own
* mask eventually?
*/
#define SGNXPRO_MIXER_DEVICES (SBPRO_MIXER_DEVICES|SOUND_MASK_BASS| \
SOUND_MASK_TREBLE|SOUND_MASK_SPEAKER )
#define SB16_RECORDING_DEVICES (SOUND_MASK_SYNTH | SOUND_MASK_LINE | SOUND_MASK_MIC | \ #define SB16_RECORDING_DEVICES (SOUND_MASK_SYNTH | SOUND_MASK_LINE | SOUND_MASK_MIC | \
SOUND_MASK_CD) SOUND_MASK_CD)
...@@ -62,6 +78,13 @@ ...@@ -62,6 +78,13 @@
#define IRQ_STAT 0x82 #define IRQ_STAT 0x82
#define OPSW 0x3c #define OPSW 0x3c
/*
* Additional registers on the SG NX Pro
*/
#define COVOX_VOL 0x42
#define TREBLE_LVL 0x44
#define BASS_LVL 0x46
#define FREQ_HI (1 << 3)/* Use High-frequency ANFI filters */ #define FREQ_HI (1 << 3)/* Use High-frequency ANFI filters */
#define FREQ_LOW 0 /* Use Low-frequency ANFI filters */ #define FREQ_LOW 0 /* Use Low-frequency ANFI filters */
#define FILT_ON 0 /* Yes, 0 to turn it on, 1 for off */ #define FILT_ON 0 /* Yes, 0 to turn it on, 1 for off */
...@@ -108,6 +131,23 @@ MIX_ENT(SOUND_MIXER_ALTPCM, 0x00, 0, 0, 0x00, 0, 0), ...@@ -108,6 +131,23 @@ MIX_ENT(SOUND_MIXER_ALTPCM, 0x00, 0, 0, 0x00, 0, 0),
MIX_ENT(SOUND_MIXER_RECLEV, 0x00, 0, 0, 0x00, 0, 0) MIX_ENT(SOUND_MIXER_RECLEV, 0x00, 0, 0, 0x00, 0, 0)
}; };
#ifdef __SGNXPRO__
mixer_tab sgnxpro_mix = {
MIX_ENT(SOUND_MIXER_VOLUME, 0x22, 7, 4, 0x22, 3, 4),
MIX_ENT(SOUND_MIXER_BASS, 0x46, 2, 3, 0x00, 0, 0),
MIX_ENT(SOUND_MIXER_TREBLE, 0x44, 2, 3, 0x00, 0, 0),
MIX_ENT(SOUND_MIXER_SYNTH, 0x26, 7, 4, 0x26, 3, 4),
MIX_ENT(SOUND_MIXER_PCM, 0x04, 7, 4, 0x04, 3, 4),
MIX_ENT(SOUND_MIXER_SPEAKER, 0x42, 2, 3, 0x00, 0, 0),
MIX_ENT(SOUND_MIXER_LINE, 0x2e, 7, 4, 0x2e, 3, 4),
MIX_ENT(SOUND_MIXER_MIC, 0x0a, 2, 3, 0x00, 0, 0),
MIX_ENT(SOUND_MIXER_CD, 0x28, 7, 4, 0x28, 3, 4),
MIX_ENT(SOUND_MIXER_IMIX, 0x00, 0, 0, 0x00, 0, 0),
MIX_ENT(SOUND_MIXER_ALTPCM, 0x00, 0, 0, 0x00, 0, 0),
MIX_ENT(SOUND_MIXER_RECLEV, 0x00, 0, 0, 0x00, 0, 0)
};
#endif
mixer_tab sb16_mix = { mixer_tab sb16_mix = {
MIX_ENT(SOUND_MIXER_VOLUME, 0x30, 7, 5, 0x31, 7, 5), MIX_ENT(SOUND_MIXER_VOLUME, 0x30, 7, 5, 0x31, 7, 5),
MIX_ENT(SOUND_MIXER_BASS, 0x46, 7, 4, 0x47, 7, 4), MIX_ENT(SOUND_MIXER_BASS, 0x46, 7, 4, 0x47, 7, 4),
......
...@@ -140,7 +140,7 @@ void sb_midi_init(int model); ...@@ -140,7 +140,7 @@ void sb_midi_init(int model);
void sb_setmixer (unsigned int port, unsigned int value); void sb_setmixer (unsigned int port, unsigned int value);
int sb_getmixer (unsigned int port); int sb_getmixer (unsigned int port);
void sb_mixer_set_stereo(int mode); void sb_mixer_set_stereo(int mode);
void sb_mixer_init(int major_model); int sb_mixer_init(int major_model);
/* From opl3.c */ /* From opl3.c */
int opl3_detect (int ioaddr); int opl3_detect (int ioaddr);
......
...@@ -180,7 +180,7 @@ If your card has nonstandard I/O address or IRQ number, change defines ...@@ -180,7 +180,7 @@ If your card has nonstandard I/O address or IRQ number, change defines
* In v3.0 it's /dev/sndproc but this could be a temporary solution. * In v3.0 it's /dev/sndproc but this could be a temporary solution.
*/ */
#define SND_NDEVS 64 /* Number of supported devices */ #define SND_NDEVS 256 /* Number of supported devices */
#define SND_DEV_CTL 0 /* Control port /dev/mixer */ #define SND_DEV_CTL 0 /* Control port /dev/mixer */
#define SND_DEV_SEQ 1 /* Sequencer output /dev/sequencer (FM #define SND_DEV_SEQ 1 /* Sequencer output /dev/sequencer (FM
synthesizer and MIDI output) */ synthesizer and MIDI output) */
...@@ -199,7 +199,7 @@ If your card has nonstandard I/O address or IRQ number, change defines ...@@ -199,7 +199,7 @@ If your card has nonstandard I/O address or IRQ number, change defines
#define ON 1 #define ON 1
#define OFF 0 #define OFF 0
#define MAX_AUDIO_DEV 4 #define MAX_AUDIO_DEV 5
#define MAX_MIXER_DEV 2 #define MAX_MIXER_DEV 2
#define MAX_SYNTH_DEV 3 #define MAX_SYNTH_DEV 3
#define MAX_MIDI_DEV 6 #define MAX_MIDI_DEV 6
......
...@@ -42,6 +42,12 @@ ...@@ -42,6 +42,12 @@
#define WIN_DIAGNOSE 0x90 #define WIN_DIAGNOSE 0x90
#define WIN_SPECIFY 0x91 #define WIN_SPECIFY 0x91
#define WIN_MULTREAD 0xC4 /* read multiple sectors */
#define WIN_MULTWRITE 0xC5 /* write multiple sectors */
#define WIN_SETMULT 0xC6 /* enable read multiple */
#define WIN_IDENTIFY 0xEC /* ask drive to identify itself */
#define WIN_SETFEATURES 0xEF /* set special drive features */
/* Bits for HD_ERROR */ /* Bits for HD_ERROR */
#define MARK_ERR 0x01 /* Bad address mark */ #define MARK_ERR 0x01 /* Bad address mark */
#define TRK0_ERR 0x02 /* couldn't find track 0 */ #define TRK0_ERR 0x02 /* couldn't find track 0 */
...@@ -61,4 +67,9 @@ struct hd_geometry { ...@@ -61,4 +67,9 @@ struct hd_geometry {
unsigned short cylinders; unsigned short cylinders;
unsigned long start; unsigned long start;
}; };
#define HDIO_GETUNMASKINTR 0x302
#define HDIO_SETUNMASKINTR 0x303
#define HDIO_GETMULTCOUNT 0x304
#define HDIO_SETMULTCOUNT 0x305
#define HDIO_SETFEATURE 0x306
#endif #endif
...@@ -145,10 +145,10 @@ extern int unmap_page_range(unsigned long from, unsigned long size); ...@@ -145,10 +145,10 @@ extern int unmap_page_range(unsigned long from, unsigned long size);
extern int remap_page_range(unsigned long from, unsigned long to, unsigned long size, int mask); extern int remap_page_range(unsigned long from, unsigned long to, unsigned long size, int mask);
extern int zeromap_page_range(unsigned long from, unsigned long size, int mask); extern int zeromap_page_range(unsigned long from, unsigned long size, int mask);
extern void do_wp_page(unsigned long error_code, unsigned long address, extern void do_wp_page(struct vm_area_struct * vma, unsigned long address,
struct task_struct *tsk); unsigned long error_code);
extern void do_no_page(unsigned long error_code, unsigned long address, extern void do_no_page(struct vm_area_struct * vma, unsigned long address,
struct task_struct *tsk); unsigned long error_code);
extern unsigned long paging_init(unsigned long start_mem, unsigned long end_mem); extern unsigned long paging_init(unsigned long start_mem, unsigned long end_mem);
extern void mem_init(unsigned long low_start_mem, extern void mem_init(unsigned long low_start_mem,
......
...@@ -6,6 +6,8 @@ ...@@ -6,6 +6,8 @@
#define __WCLONE 0x80000000 #define __WCLONE 0x80000000
#ifdef __KERNEL__
struct wait_queue { struct wait_queue {
struct task_struct * task; struct task_struct * task;
struct wait_queue * next; struct wait_queue * next;
...@@ -31,4 +33,6 @@ typedef struct select_table_struct { ...@@ -31,4 +33,6 @@ typedef struct select_table_struct {
#define __MAX_SELECT_TABLE_ENTRIES (4096 / sizeof (struct select_table_entry)) #define __MAX_SELECT_TABLE_ENTRIES (4096 / sizeof (struct select_table_entry))
#endif /* __KERNEL__ */
#endif #endif
...@@ -81,24 +81,20 @@ static inline int put_stack_long(struct task_struct *task, int offset, ...@@ -81,24 +81,20 @@ static inline int put_stack_long(struct task_struct *task, int offset,
* tables. NOTE! You should check that the long isn't on a page boundary, * tables. NOTE! You should check that the long isn't on a page boundary,
* and that it is in the task area before calling this: this routine does * and that it is in the task area before calling this: this routine does
* no checking. * no checking.
*
* NOTE2! This uses "tsk->tss.cr3" even though we know it's currently always
* zero. This routine shouldn't have to change when we make a better mm.
*/ */
static unsigned long get_long(struct task_struct * tsk, static unsigned long get_long(struct vm_area_struct * vma, unsigned long addr)
unsigned long addr)
{ {
unsigned long page; unsigned long page;
repeat: repeat:
page = *PAGE_DIR_OFFSET(tsk->tss.cr3,addr); page = *PAGE_DIR_OFFSET(vma->vm_task->tss.cr3, addr);
if (page & PAGE_PRESENT) { if (page & PAGE_PRESENT) {
page &= PAGE_MASK; page &= PAGE_MASK;
page += PAGE_PTR(addr); page += PAGE_PTR(addr);
page = *((unsigned long *) page); page = *((unsigned long *) page);
} }
if (!(page & PAGE_PRESENT)) { if (!(page & PAGE_PRESENT)) {
do_no_page(0,addr,tsk); do_no_page(vma, addr, 0);
goto repeat; goto repeat;
} }
/* this is a hack for non-kernel-mapped video buffers and similar */ /* this is a hack for non-kernel-mapped video buffers and similar */
...@@ -118,14 +114,14 @@ static unsigned long get_long(struct task_struct * tsk, ...@@ -118,14 +114,14 @@ static unsigned long get_long(struct task_struct * tsk,
* Now keeps R/W state of page so that a text page stays readonly * Now keeps R/W state of page so that a text page stays readonly
* even if a debugger scribbles breakpoints into it. -M.U- * even if a debugger scribbles breakpoints into it. -M.U-
*/ */
static void put_long(struct task_struct * tsk, unsigned long addr, static void put_long(struct vm_area_struct * vma, unsigned long addr,
unsigned long data) unsigned long data)
{ {
unsigned long page, pte = 0; unsigned long page, pte = 0;
int readonly = 0; int readonly = 0;
repeat: repeat:
page = *PAGE_DIR_OFFSET(tsk->tss.cr3,addr); page = *PAGE_DIR_OFFSET(vma->vm_task->tss.cr3, addr);
if (page & PAGE_PRESENT) { if (page & PAGE_PRESENT) {
page &= PAGE_MASK; page &= PAGE_MASK;
page += PAGE_PTR(addr); page += PAGE_PTR(addr);
...@@ -133,13 +129,13 @@ static void put_long(struct task_struct * tsk, unsigned long addr, ...@@ -133,13 +129,13 @@ static void put_long(struct task_struct * tsk, unsigned long addr,
page = *((unsigned long *) page); page = *((unsigned long *) page);
} }
if (!(page & PAGE_PRESENT)) { if (!(page & PAGE_PRESENT)) {
do_no_page(0 /* PAGE_RW */ ,addr,tsk); do_no_page(vma, addr, 0 /* PAGE_RW */);
goto repeat; goto repeat;
} }
if (!(page & PAGE_RW)) { if (!(page & PAGE_RW)) {
if(!(page & PAGE_COW)) if (!(page & PAGE_COW))
readonly = 1; readonly = 1;
do_wp_page(PAGE_RW | PAGE_PRESENT,addr,tsk); do_wp_page(vma, addr, PAGE_RW | PAGE_PRESENT);
goto repeat; goto repeat;
} }
/* this is a hack for non-kernel-mapped video buffers and similar */ /* this is a hack for non-kernel-mapped video buffers and similar */
...@@ -150,12 +146,34 @@ static void put_long(struct task_struct * tsk, unsigned long addr, ...@@ -150,12 +146,34 @@ static void put_long(struct task_struct * tsk, unsigned long addr,
page &= PAGE_MASK; page &= PAGE_MASK;
page += addr & ~PAGE_MASK; page += addr & ~PAGE_MASK;
*(unsigned long *) page = data; *(unsigned long *) page = data;
if(readonly) { if (readonly) {
*(unsigned long *) pte &=~ (PAGE_RW|PAGE_COW); *(unsigned long *) pte &=~ (PAGE_RW|PAGE_COW);
invalidate(); invalidate();
} }
} }
static struct vm_area_struct * find_vma(struct task_struct * tsk, unsigned long addr)
{
struct vm_area_struct * vma;
addr &= PAGE_MASK;
for (vma = current->mm->mmap ; ; vma = vma->vm_next) {
if (!vma)
return NULL;
if (vma->vm_end > addr)
break;
}
if (vma->vm_start <= addr)
return vma;
if (!(vma->vm_flags & VM_GROWSDOWN))
return NULL;
if (vma->vm_end - addr > tsk->rlim[RLIMIT_STACK].rlim_cur)
return NULL;
vma->vm_offset -= vma->vm_start - addr;
vma->vm_start = addr;
return vma;
}
/* /*
* This routine checks the page boundaries, and that the offset is * This routine checks the page boundaries, and that the offset is
* within the task area. It then calls get_long() to read a long. * within the task area. It then calls get_long() to read a long.
...@@ -163,13 +181,21 @@ static void put_long(struct task_struct * tsk, unsigned long addr, ...@@ -163,13 +181,21 @@ static void put_long(struct task_struct * tsk, unsigned long addr,
static int read_long(struct task_struct * tsk, unsigned long addr, static int read_long(struct task_struct * tsk, unsigned long addr,
unsigned long * result) unsigned long * result)
{ {
unsigned long low,high; struct vm_area_struct * vma = find_vma(tsk, addr);
if (addr > TASK_SIZE-sizeof(long)) if (!vma)
return -EIO; return -EIO;
if ((addr & ~PAGE_MASK) > PAGE_SIZE-sizeof(long)) { if ((addr & ~PAGE_MASK) > PAGE_SIZE-sizeof(long)) {
low = get_long(tsk,addr & ~(sizeof(long)-1)); unsigned long low,high;
high = get_long(tsk,(addr+sizeof(long)) & ~(sizeof(long)-1)); struct vm_area_struct * vma_high = vma;
if (addr + sizeof(long) >= vma->vm_end) {
vma_high = vma->vm_next;
if (!vma_high || vma_high->vm_start != vma->vm_end)
return -EIO;
}
low = get_long(vma, addr & ~(sizeof(long)-1));
high = get_long(vma_high, (addr+sizeof(long)) & ~(sizeof(long)-1));
switch (addr & (sizeof(long)-1)) { switch (addr & (sizeof(long)-1)) {
case 1: case 1:
low >>= 8; low >>= 8;
...@@ -186,7 +212,7 @@ static int read_long(struct task_struct * tsk, unsigned long addr, ...@@ -186,7 +212,7 @@ static int read_long(struct task_struct * tsk, unsigned long addr,
} }
*result = low; *result = low;
} else } else
*result = get_long(tsk,addr); *result = get_long(vma, addr);
return 0; return 0;
} }
...@@ -197,13 +223,21 @@ static int read_long(struct task_struct * tsk, unsigned long addr, ...@@ -197,13 +223,21 @@ static int read_long(struct task_struct * tsk, unsigned long addr,
static int write_long(struct task_struct * tsk, unsigned long addr, static int write_long(struct task_struct * tsk, unsigned long addr,
unsigned long data) unsigned long data)
{ {
unsigned long low,high; struct vm_area_struct * vma = find_vma(tsk, addr);
if (addr > TASK_SIZE-sizeof(long)) if (!vma)
return -EIO; return -EIO;
if ((addr & ~PAGE_MASK) > PAGE_SIZE-sizeof(long)) { if ((addr & ~PAGE_MASK) > PAGE_SIZE-sizeof(long)) {
low = get_long(tsk,addr & ~(sizeof(long)-1)); unsigned long low,high;
high = get_long(tsk,(addr+sizeof(long)) & ~(sizeof(long)-1)); struct vm_area_struct * vma_high = vma;
if (addr + sizeof(long) >= vma->vm_end) {
vma_high = vma->vm_next;
if (!vma_high || vma_high->vm_start != vma->vm_end)
return -EIO;
}
low = get_long(vma, addr & ~(sizeof(long)-1));
high = get_long(vma_high, (addr+sizeof(long)) & ~(sizeof(long)-1));
switch (addr & (sizeof(long)-1)) { switch (addr & (sizeof(long)-1)) {
case 0: /* shouldn't happen, but safety first */ case 0: /* shouldn't happen, but safety first */
low = data; low = data;
...@@ -227,10 +261,10 @@ static int write_long(struct task_struct * tsk, unsigned long addr, ...@@ -227,10 +261,10 @@ static int write_long(struct task_struct * tsk, unsigned long addr,
high |= data >> 8; high |= data >> 8;
break; break;
} }
put_long(tsk,addr & ~(sizeof(long)-1),low); put_long(vma, addr & ~(sizeof(long)-1),low);
put_long(tsk,(addr+sizeof(long)) & ~(sizeof(long)-1),high); put_long(vma_high, (addr+sizeof(long)) & ~(sizeof(long)-1),high);
} else } else
put_long(tsk,addr,data); put_long(vma, addr, data);
return 0; return 0;
} }
......
...@@ -148,7 +148,6 @@ asmlinkage void math_emulate(long arg) ...@@ -148,7 +148,6 @@ asmlinkage void math_emulate(long arg)
unsigned long itimer_ticks = 0; unsigned long itimer_ticks = 0;
unsigned long itimer_next = ~0; unsigned long itimer_next = ~0;
static unsigned long lost_ticks = 0;
/* /*
* 'schedule()' is the scheduler function. It's a very simple and nice * 'schedule()' is the scheduler function. It's a very simple and nice
...@@ -361,54 +360,76 @@ void sleep_on(struct wait_queue **p) ...@@ -361,54 +360,76 @@ void sleep_on(struct wait_queue **p)
__sleep_on(p,TASK_UNINTERRUPTIBLE); __sleep_on(p,TASK_UNINTERRUPTIBLE);
} }
static struct timer_list * next_timer = NULL; /*
* The head for the timer-list has a "expires" field of MAX_UINT,
* and the sorting routine counts on this..
*/
static struct timer_list timer_head = { &timer_head, &timer_head, ~0, 0, NULL };
#define SLOW_BUT_DEBUGGING_TIMERS 1
void add_timer(struct timer_list * timer) void add_timer(struct timer_list * timer)
{ {
unsigned long flags; unsigned long flags;
struct timer_list ** p; struct timer_list *p;
if (!timer) #if SLOW_BUT_DEBUGGING_TIMERS
if (timer->next || timer->prev) {
printk("add_timer() called with non-zero list from %08lx\n",
((unsigned long *) &timer)[-1]);
return; return;
timer->next = NULL; }
p = &next_timer; #endif
p = &timer_head;
timer->expires += jiffies;
save_flags(flags); save_flags(flags);
cli(); cli();
while (*p) { do {
if ((*p)->expires > timer->expires) { p = p->next;
(*p)->expires -= timer->expires; } while (timer->expires > p->expires);
timer->next = *p; timer->next = p;
break; timer->prev = p->prev;
} p->prev = timer;
timer->expires -= (*p)->expires; timer->prev->next = timer;
p = &(*p)->next;
}
*p = timer;
restore_flags(flags); restore_flags(flags);
} }
int del_timer(struct timer_list * timer) int del_timer(struct timer_list * timer)
{ {
unsigned long flags; unsigned long flags;
unsigned long expires = 0; #if SLOW_BUT_DEBUGGING_TIMERS
struct timer_list **p; struct timer_list * p;
p = &next_timer; p = &timer_head;
save_flags(flags); save_flags(flags);
cli(); cli();
while (*p) { while ((p = p->next) != &timer_head) {
if (*p == timer) { if (p == timer) {
if ((*p = timer->next) != NULL) timer->next->prev = timer->prev;
(*p)->expires += timer->expires; timer->prev->next = timer->next;
timer->expires += expires; timer->next = timer->prev = NULL;
restore_flags(flags); restore_flags(flags);
timer->expires -= jiffies;
return 1; return 1;
} }
expires += (*p)->expires;
p = &(*p)->next;
} }
if (p->next || p->prev)
printk("del_timer() called with timer not initialized\n");
restore_flags(flags); restore_flags(flags);
return 0; return 0;
#else
save_flags(flags);
cli();
if (timer->next) {
timer->next->prev = timer->prev;
timer->prev->next = timer->next;
timer->next = timer->prev = NULL;
restore_flags(flags);
timer->expires -= jiffies;
return 1;
}
restore_flags(flags);
return 0;
#endif
} }
unsigned long timer_active = 0; unsigned long timer_active = 0;
...@@ -528,12 +549,15 @@ static void timer_bh(void * unused) ...@@ -528,12 +549,15 @@ static void timer_bh(void * unused)
{ {
unsigned long mask; unsigned long mask;
struct timer_struct *tp; struct timer_struct *tp;
struct timer_list * timer;
cli(); cli();
while (next_timer && next_timer->expires == 0) { while ((timer = timer_head.next) != &timer_head && timer->expires < jiffies) {
void (*fn)(unsigned long) = next_timer->function; void (*fn)(unsigned long) = timer->function;
unsigned long data = next_timer->data; unsigned long data = timer->data;
next_timer = next_timer->next; timer->next->prev = timer->prev;
timer->prev->next = timer->next;
timer->next = timer->prev = NULL;
sti(); sti();
fn(data); fn(data);
cli(); cli();
...@@ -667,16 +691,8 @@ static void do_timer(struct pt_regs * regs) ...@@ -667,16 +691,8 @@ static void do_timer(struct pt_regs * regs)
itimer_ticks++; itimer_ticks++;
if (itimer_ticks > itimer_next) if (itimer_ticks > itimer_next)
need_resched = 1; need_resched = 1;
if (next_timer) { if (timer_head.next->expires < jiffies)
if (next_timer->expires) { mark_bh(TIMER_BH);
next_timer->expires--;
if (!next_timer->expires)
mark_bh(TIMER_BH);
} else {
lost_ticks++;
mark_bh(TIMER_BH);
}
}
if (tq_timer != &tq_last) if (tq_timer != &tq_last)
mark_bh(TQUEUE_BH); mark_bh(TQUEUE_BH);
sti(); sti();
......
This diff is collapsed.
...@@ -615,6 +615,7 @@ static int inet_create(struct socket *sock, int protocol) ...@@ -615,6 +615,7 @@ static int inet_create(struct socket *sock, int protocol)
sk->timeout = 0; sk->timeout = 0;
sk->broadcast = 0; sk->broadcast = 0;
sk->localroute = 0; sk->localroute = 0;
sk->timer.next = sk->timer.prev = NULL;
sk->timer.data = (unsigned long)sk; sk->timer.data = (unsigned long)sk;
sk->timer.function = &net_timer; sk->timer.function = &net_timer;
skb_queue_head_init(&sk->back_log); skb_queue_head_init(&sk->back_log);
......
...@@ -752,7 +752,7 @@ int arp_rcv(struct sk_buff *skb, struct device *dev, struct packet_type *pt) ...@@ -752,7 +752,7 @@ int arp_rcv(struct sk_buff *skb, struct device *dev, struct packet_type *pt)
skb_queue_head_init(&entry->skb); skb_queue_head_init(&entry->skb);
entry->next = arp_tables[hash]; entry->next = arp_tables[hash];
arp_tables[hash] = entry; arp_tables[hash] = entry;
entry->timer.next = entry->timer.prev = NULL;
sti(); sti();
} }
...@@ -841,6 +841,7 @@ int arp_find(unsigned char *haddr, unsigned long paddr, struct device *dev, ...@@ -841,6 +841,7 @@ int arp_find(unsigned char *haddr, unsigned long paddr, struct device *dev,
entry->next = arp_tables[hash]; entry->next = arp_tables[hash];
entry->dev = dev; entry->dev = dev;
arp_tables[hash] = entry; arp_tables[hash] = entry;
entry->timer.next = entry->timer.prev = NULL;
entry->timer.function = arp_expire_request; entry->timer.function = arp_expire_request;
entry->timer.data = (unsigned long)entry; entry->timer.data = (unsigned long)entry;
entry->timer.expires = ARP_RES_TIME; entry->timer.expires = ARP_RES_TIME;
...@@ -1048,6 +1049,7 @@ static int arp_req_set(struct arpreq *req) ...@@ -1048,6 +1049,7 @@ static int arp_req_set(struct arpreq *req)
entry->hlen = hlen; entry->hlen = hlen;
entry->htype = htype; entry->htype = htype;
entry->next = arp_tables[hash]; entry->next = arp_tables[hash];
entry->timer.next = entry->timer.prev = NULL;
arp_tables[hash] = entry; arp_tables[hash] = entry;
skb_queue_head_init(&entry->skb); skb_queue_head_init(&entry->skb);
} }
......
...@@ -336,6 +336,8 @@ int dev_close(struct device *dev) ...@@ -336,6 +336,8 @@ int dev_close(struct device *dev)
void dev_queue_xmit(struct sk_buff *skb, struct device *dev, int pri) void dev_queue_xmit(struct sk_buff *skb, struct device *dev, int pri)
{ {
unsigned long flags; unsigned long flags;
int nitcount;
struct packet_type *ptype;
int where = 0; /* used to say if the packet should go */ int where = 0; /* used to say if the packet should go */
/* at the front or the back of the */ /* at the front or the back of the */
/* queue - front is a retranmsit try */ /* queue - front is a retranmsit try */
...@@ -419,6 +421,17 @@ void dev_queue_xmit(struct sk_buff *skb, struct device *dev, int pri) ...@@ -419,6 +421,17 @@ void dev_queue_xmit(struct sk_buff *skb, struct device *dev, int pri)
} }
restore_flags(flags); restore_flags(flags);
/* copy outgoing packets to any sniffer packet handlers */
for (nitcount = dev_nit, ptype = ptype_base; nitcount > 0 && ptype != NULL; ptype = ptype->next) {
if (ptype->type == htons(ETH_P_ALL)) {
struct sk_buff *skb2;
if ((skb2 = skb_clone(skb, GFP_ATOMIC)) == NULL)
break;
ptype->func(skb2, skb->dev, ptype);
nitcount--;
}
}
if (dev->hard_start_xmit(skb, dev) == 0) { if (dev->hard_start_xmit(skb, dev) == 0) {
/* /*
* Packet is now solely the responsibility of the driver * Packet is now solely the responsibility of the driver
......
...@@ -681,8 +681,7 @@ struct sk_buff * tcp_dequeue_partial(struct sock * sk) ...@@ -681,8 +681,7 @@ struct sk_buff * tcp_dequeue_partial(struct sock * sk)
save_flags(flags); save_flags(flags);
cli(); cli();
skb = sk->partial; skb = sk->partial;
if (skb) if (skb) {
{
sk->partial = NULL; sk->partial = NULL;
del_timer(&sk->partial_timer); del_timer(&sk->partial_timer);
} }
...@@ -711,6 +710,7 @@ void tcp_enqueue_partial(struct sk_buff * skb, struct sock * sk) ...@@ -711,6 +710,7 @@ void tcp_enqueue_partial(struct sk_buff * skb, struct sock * sk)
if (tmp) if (tmp)
del_timer(&sk->partial_timer); del_timer(&sk->partial_timer);
sk->partial = skb; sk->partial = skb;
sk->partial_timer.next = sk->partial_timer.prev = NULL;
sk->partial_timer.expires = HZ; sk->partial_timer.expires = HZ;
sk->partial_timer.function = (void (*)(unsigned long)) tcp_send_partial; sk->partial_timer.function = (void (*)(unsigned long)) tcp_send_partial;
sk->partial_timer.data = (unsigned long) sk; sk->partial_timer.data = (unsigned long) sk;
...@@ -1987,6 +1987,7 @@ static void tcp_conn_request(struct sock *sk, struct sk_buff *skb, ...@@ -1987,6 +1987,7 @@ static void tcp_conn_request(struct sock *sk, struct sk_buff *skb,
newsk->urg_data = 0; newsk->urg_data = 0;
newsk->retransmits = 0; newsk->retransmits = 0;
newsk->destroy = 0; newsk->destroy = 0;
newsk->timer.next = newsk->timer.prev = NULL;
newsk->timer.data = (unsigned long)newsk; newsk->timer.data = (unsigned long)newsk;
newsk->timer.function = &net_timer; newsk->timer.function = &net_timer;
newsk->dummy_th.source = skb->h.th->dest; newsk->dummy_th.source = skb->h.th->dest;
...@@ -2217,6 +2218,15 @@ static void tcp_close(struct sock *sk, int timeout) ...@@ -2217,6 +2218,15 @@ static void tcp_close(struct sock *sk, int timeout)
* XXX if retransmit count reaches limit, is tcp_close() * XXX if retransmit count reaches limit, is tcp_close()
* called with timeout == 1 ? if not, we need to fix that. * called with timeout == 1 ? if not, we need to fix that.
*/ */
if (!timeout) {
int timer_active;
timer_active = del_timer(&sk->timer);
if (timer_active)
add_timer(&sk->timer);
else
reset_timer(sk, TIME_CLOSE, 4 * sk->rto);
}
#ifdef NOTDEF #ifdef NOTDEF
/* /*
* Start a timer. * Start a timer.
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment