Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
L
linux
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
0
Merge Requests
0
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
Kirill Smelkov
linux
Commits
f4a818e0
Commit
f4a818e0
authored
Nov 23, 2007
by
Linus Torvalds
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Import 1.1.55
parent
057f54fb
Changes
23
Show whitespace changes
Inline
Side-by-side
Showing
23 changed files
with
348 additions
and
245 deletions
+348
-245
Makefile
Makefile
+1
-1
drivers/block/floppy.c
drivers/block/floppy.c
+11
-10
drivers/block/hd.c
drivers/block/hd.c
+140
-167
drivers/char/lp.c
drivers/char/lp.c
+8
-0
drivers/sound/Makefile
drivers/sound/Makefile
+5
-1
fs/binfmt_elf.c
fs/binfmt_elf.c
+2
-2
fs/ext/dir.c
fs/ext/dir.c
+17
-5
fs/ext2/dir.c
fs/ext2/dir.c
+70
-24
fs/ext2/namei.c
fs/ext2/namei.c
+5
-2
fs/hpfs/hpfs_fs.c
fs/hpfs/hpfs_fs.c
+6
-3
fs/isofs/dir.c
fs/isofs/dir.c
+4
-1
fs/minix/dir.c
fs/minix/dir.c
+14
-6
fs/minix/namei.c
fs/minix/namei.c
+6
-1
fs/msdos/dir.c
fs/msdos/dir.c
+5
-2
fs/nfs/dir.c
fs/nfs/dir.c
+4
-1
fs/proc/link.c
fs/proc/link.c
+2
-1
fs/read_write.c
fs/read_write.c
+12
-4
fs/sysv/dir.c
fs/sysv/dir.c
+4
-1
fs/xiafs/dir.c
fs/xiafs/dir.c
+17
-5
include/linux/hdreg.h
include/linux/hdreg.h
+0
-1
include/linux/tty.h
include/linux/tty.h
+1
-0
kernel/ksyms.c
kernel/ksyms.c
+4
-0
net/inet/tcp.c
net/inet/tcp.c
+10
-7
No files found.
Makefile
View file @
f4a818e0
VERSION
=
1
VERSION
=
1
PATCHLEVEL
=
1
PATCHLEVEL
=
1
SUBLEVEL
=
5
4
SUBLEVEL
=
5
5
ARCH
=
i386
ARCH
=
i386
...
...
drivers/block/floppy.c
View file @
f4a818e0
...
@@ -25,9 +25,9 @@
...
@@ -25,9 +25,9 @@
/* Define the following if you don't like that your drives seek audibly
/* Define the following if you don't like that your drives seek audibly
* after a disk change
* after a disk change
(but it may not work correctly for everybody)
*/
*/
#define SILENT_DC_CLEAR
/* #define SILENT_DC_CLEAR */
/* End of configuration */
/* End of configuration */
...
@@ -304,7 +304,7 @@ static struct floppy_struct floppy_type[32] = {
...
@@ -304,7 +304,7 @@ static struct floppy_struct floppy_type[32] = {
{
5760
,
36
,
2
,
80
,
0
,
0x1B
,
0x43
,
0xAF
,
0x54
,
"CompaQ"
},
/* 9 2.88MB 3.5" */
{
5760
,
36
,
2
,
80
,
0
,
0x1B
,
0x43
,
0xAF
,
0x54
,
"CompaQ"
},
/* 9 2.88MB 3.5" */
{
2880
,
18
,
2
,
80
,
0
,
0x25
,
0x00
,
0xDF
,
0x02
,
"h1440"
},
/* 10 1.44MB 5.25" */
{
2880
,
18
,
2
,
80
,
0
,
0x25
,
0x00
,
0xDF
,
0x02
,
"h1440"
},
/* 10 1.44MB 5.25" */
{
3360
,
21
,
2
,
80
,
0
,
0x1C
,
0x00
,
0xCF
,
0x
6
C
,
"H1680"
},
/* 11 1.68MB 3.5" */
{
3360
,
21
,
2
,
80
,
0
,
0x1C
,
0x00
,
0xCF
,
0x
0
C
,
"H1680"
},
/* 11 1.68MB 3.5" */
{
820
,
10
,
2
,
41
,
1
,
0x25
,
0x01
,
0xDF
,
0x2E
,
"h410"
},
/* 12 410KB 5.25" */
{
820
,
10
,
2
,
41
,
1
,
0x25
,
0x01
,
0xDF
,
0x2E
,
"h410"
},
/* 12 410KB 5.25" */
{
1640
,
10
,
2
,
82
,
0
,
0x25
,
0x02
,
0xDF
,
0x2E
,
"H820"
},
/* 13 820KB 3.5" */
{
1640
,
10
,
2
,
82
,
0
,
0x25
,
0x02
,
0xDF
,
0x2E
,
"H820"
},
/* 13 820KB 3.5" */
{
2952
,
18
,
2
,
82
,
0
,
0x25
,
0x00
,
0xDF
,
0x02
,
"h1476"
},
/* 14 1.48MB 5.25" */
{
2952
,
18
,
2
,
82
,
0
,
0x25
,
0x00
,
0xDF
,
0x02
,
"h1476"
},
/* 14 1.48MB 5.25" */
...
@@ -317,8 +317,8 @@ static struct floppy_struct floppy_type[32] = {
...
@@ -317,8 +317,8 @@ static struct floppy_struct floppy_type[32] = {
{
1760
,
11
,
2
,
80
,
0
,
0x1C
,
0x09
,
0xCF
,
0x6C
,
"h880"
},
/* 20 880KB 5.25" */
{
1760
,
11
,
2
,
80
,
0
,
0x1C
,
0x09
,
0xCF
,
0x6C
,
"h880"
},
/* 20 880KB 5.25" */
{
2080
,
13
,
2
,
80
,
0
,
0x1C
,
0x01
,
0xCF
,
0x6C
,
"D1040"
},
/* 21 1.04MB 3.5" */
{
2080
,
13
,
2
,
80
,
0
,
0x1C
,
0x01
,
0xCF
,
0x6C
,
"D1040"
},
/* 21 1.04MB 3.5" */
{
2240
,
14
,
2
,
80
,
0
,
0x1C
,
0x19
,
0xCF
,
0x6C
,
"D1120"
},
/* 22 1.12MB 3.5" */
{
2240
,
14
,
2
,
80
,
0
,
0x1C
,
0x19
,
0xCF
,
0x6C
,
"D1120"
},
/* 22 1.12MB 3.5" */
{
3200
,
20
,
2
,
80
,
0
,
0x1C
,
0x20
,
0xCF
,
0x
6
C
,
"h1600"
},
/* 23 1.6MB 5.25" */
{
3200
,
20
,
2
,
80
,
0
,
0x1C
,
0x20
,
0xCF
,
0x
2
C
,
"h1600"
},
/* 23 1.6MB 5.25" */
{
3520
,
22
,
2
,
80
,
0
,
0x1C
,
0x08
,
0xCF
,
0x
6C
,
"H1760"
},
/* 24 1.76MB 3.5" */
{
3520
,
22
,
2
,
80
,
0
,
0x1C
,
0x08
,
0xCF
,
0x
2e
,
"H1760"
},
/* 24 1.76MB 3.5" */
{
3840
,
24
,
2
,
80
,
0
,
0x1C
,
0x20
,
0xCF
,
0x6C
,
"H1920"
},
/* 25 1.92MB 3.5" */
{
3840
,
24
,
2
,
80
,
0
,
0x1C
,
0x20
,
0xCF
,
0x6C
,
"H1920"
},
/* 25 1.92MB 3.5" */
{
6400
,
40
,
2
,
80
,
0
,
0x25
,
0x5B
,
0xCF
,
0x6C
,
"E3200"
},
/* 26 3.20MB 3.5" */
{
6400
,
40
,
2
,
80
,
0
,
0x25
,
0x5B
,
0xCF
,
0x6C
,
"E3200"
},
/* 26 3.20MB 3.5" */
{
7040
,
44
,
2
,
80
,
0
,
0x25
,
0x5B
,
0xCF
,
0x6C
,
"E3520"
},
/* 27 3.52MB 3.5" */
{
7040
,
44
,
2
,
80
,
0
,
0x25
,
0x5B
,
0xCF
,
0x6C
,
"E3520"
},
/* 27 3.52MB 3.5" */
...
@@ -326,7 +326,7 @@ static struct floppy_struct floppy_type[32] = {
...
@@ -326,7 +326,7 @@ static struct floppy_struct floppy_type[32] = {
{
3680
,
23
,
2
,
80
,
0
,
0x1C
,
0x10
,
0xCF
,
0x6C
,
"H1840"
},
/* 29 1.84MB 3.5" */
{
3680
,
23
,
2
,
80
,
0
,
0x1C
,
0x10
,
0xCF
,
0x6C
,
"H1840"
},
/* 29 1.84MB 3.5" */
{
1600
,
10
,
2
,
80
,
0
,
0x25
,
0x02
,
0xDF
,
0x2E
,
"D800"
},
/* 30 800KB 3.5" */
{
1600
,
10
,
2
,
80
,
0
,
0x25
,
0x02
,
0xDF
,
0x2E
,
"D800"
},
/* 30 800KB 3.5" */
{
3200
,
20
,
2
,
80
,
0
,
0x1C
,
0x00
,
0xCF
,
0x
6
C
,
"H1600"
},
/* 31 1.6MB 3.5" */
{
3200
,
20
,
2
,
80
,
0
,
0x1C
,
0x00
,
0xCF
,
0x
2
C
,
"H1600"
},
/* 31 1.6MB 3.5" */
};
};
#define NUMBER(x) (sizeof(x) / sizeof(*(x)))
#define NUMBER(x) (sizeof(x) / sizeof(*(x)))
...
@@ -1278,9 +1278,10 @@ static void recal_interrupt(void)
...
@@ -1278,9 +1278,10 @@ static void recal_interrupt(void)
debugt
(
"recal interrupt need 1 recal:"
);
debugt
(
"recal interrupt need 1 recal:"
);
#endif
#endif
/* after a second recalibrate, we still haven't
/* after a second recalibrate, we still haven't
* reached track 0. Probably no drive */
* reached track 0. Probably no drive. Raise an
DRS
->
track
=
PROVEN_ABSENT
;
* error, as failing immediately might upset
cont
->
done
(
0
);
* computers possessed by the Devil :-) */
cont
->
error
();
cont
->
redo
();
cont
->
redo
();
return
;
return
;
case
NEED_2_RECAL
:
case
NEED_2_RECAL
:
...
@@ -1691,7 +1692,7 @@ static void setup_format_params(void)
...
@@ -1691,7 +1692,7 @@ static void setup_format_params(void)
int
count
,
head_shift
,
track_shift
;
int
count
,
head_shift
,
track_shift
;
raw_cmd
.
flags
=
FD_RAW_WRITE
|
FD_RAW_INTR
|
FD_RAW_SPIN
|
raw_cmd
.
flags
=
FD_RAW_WRITE
|
FD_RAW_INTR
|
FD_RAW_SPIN
|
FD_RAW_NEED_DISK
|
FD_RAW_NEED_SEEK
;
/*FD_RAW_NEED_DISK |*/
FD_RAW_NEED_SEEK
;
raw_cmd
.
rate
=
floppy
->
rate
&
0x3
;
raw_cmd
.
rate
=
floppy
->
rate
&
0x3
;
raw_cmd
.
cmd_count
=
NR_F
;
raw_cmd
.
cmd_count
=
NR_F
;
COMMAND
=
FM_MODE
(
floppy
,
FD_FORMAT
);
COMMAND
=
FM_MODE
(
floppy
,
FD_FORMAT
);
...
...
drivers/block/hd.c
View file @
f4a818e0
...
@@ -15,10 +15,8 @@
...
@@ -15,10 +15,8 @@
* Thanks to Branko Lankester, lankeste@fwi.uva.nl, who found a bug
* Thanks to Branko Lankester, lankeste@fwi.uva.nl, who found a bug
* in the early extended-partition checks and added DM partitions
* in the early extended-partition checks and added DM partitions
*
*
* IDE IRQ-unmask & drive-id & multiple-mode code added by Mark Lord.
* IRQ-unmask, drive-id, multiple-mode, support for ">16 heads",
*
* and general streamlining by mlord@bnr.ca (Mark Lord).
* Support for BIOS drive geometry translation added by Mark Lord.
* -- hd.c no longer chokes on drives with "more than 16 heads".
*/
*/
#define DEFAULT_MULT_COUNT 0
/* set to 0 to disable multiple mode at boot */
#define DEFAULT_MULT_COUNT 0
/* set to 0 to disable multiple mode at boot */
...
@@ -62,6 +60,9 @@ static inline unsigned char CMOS_READ(unsigned char addr)
...
@@ -62,6 +60,9 @@ static inline unsigned char CMOS_READ(unsigned char addr)
#define RECAL_FREQ 4
/* Recalibrate every 4th retry */
#define RECAL_FREQ 4
/* Recalibrate every 4th retry */
#define MAX_HD 2
#define MAX_HD 2
#define STAT_OK (READY_STAT|SEEK_STAT)
#define OK_STATUS(s) (((s)&(STAT_OK|(BUSY_STAT|WRERR_STAT|ERR_STAT)))==STAT_OK)
static
void
recal_intr
(
void
);
static
void
recal_intr
(
void
);
static
void
bad_rw_intr
(
void
);
static
void
bad_rw_intr
(
void
);
...
@@ -135,12 +136,12 @@ void hd_setup(char *str, int *ints)
...
@@ -135,12 +136,12 @@ void hd_setup(char *str, int *ints)
static
void
dump_status
(
char
*
msg
,
unsigned
int
stat
)
static
void
dump_status
(
char
*
msg
,
unsigned
int
stat
)
{
{
unsigned
long
flags
;
unsigned
long
flags
;
char
dev
;
char
dev
c
;
dev
=
CURRENT
?
'a'
+
DEVICE_NR
(
CURRENT
->
dev
)
:
'?'
;
dev
c
=
CURRENT
?
'a'
+
DEVICE_NR
(
CURRENT
->
dev
)
:
'?'
;
save_flags
(
flags
);
save_flags
(
flags
);
sti
();
sti
();
printk
(
"hd%c: %s: status=0x%02x { "
,
dev
,
msg
,
stat
&
0xff
);
printk
(
"hd%c: %s: status=0x%02x { "
,
dev
c
,
msg
,
stat
&
0xff
);
if
(
stat
&
BUSY_STAT
)
printk
(
"Busy "
);
if
(
stat
&
BUSY_STAT
)
printk
(
"Busy "
);
if
(
stat
&
READY_STAT
)
printk
(
"DriveReady "
);
if
(
stat
&
READY_STAT
)
printk
(
"DriveReady "
);
if
(
stat
&
WRERR_STAT
)
printk
(
"WriteFault "
);
if
(
stat
&
WRERR_STAT
)
printk
(
"WriteFault "
);
...
@@ -150,55 +151,48 @@ static void dump_status (char *msg, unsigned int stat)
...
@@ -150,55 +151,48 @@ static void dump_status (char *msg, unsigned int stat)
if
(
stat
&
INDEX_STAT
)
printk
(
"Index "
);
if
(
stat
&
INDEX_STAT
)
printk
(
"Index "
);
if
(
stat
&
ERR_STAT
)
printk
(
"Error "
);
if
(
stat
&
ERR_STAT
)
printk
(
"Error "
);
printk
(
"}
\n
"
);
printk
(
"}
\n
"
);
if
(
stat
&
ERR_STAT
)
{
if
((
stat
&
ERR_STAT
)
==
0
)
{
unsigned
int
err
=
inb
(
HD_ERROR
);
hd_error
=
0
;
printk
(
"hd%c: %s: error=0x%02x { "
,
dev
,
msg
,
err
&
0xff
);
}
else
{
if
(
err
&
BBD_ERR
)
printk
(
"BadSector "
);
hd_error
=
inb
(
HD_ERROR
);
if
(
err
&
ECC_ERR
)
printk
(
"UncorrectableError "
);
printk
(
"hd%c: %s: error=0x%02x { "
,
devc
,
msg
,
hd_error
&
0xff
);
if
(
err
&
ID_ERR
)
printk
(
"SectorIdNotFound "
);
if
(
hd_error
&
BBD_ERR
)
printk
(
"BadSector "
);
if
(
err
&
ABRT_ERR
)
printk
(
"DriveStatusError "
);
if
(
hd_error
&
ECC_ERR
)
printk
(
"UncorrectableError "
);
if
(
err
&
TRK0_ERR
)
printk
(
"TrackZeroNotFound "
);
if
(
hd_error
&
ID_ERR
)
printk
(
"SectorIdNotFound "
);
if
(
err
&
MARK_ERR
)
printk
(
"AddrMarkNotFound "
);
if
(
hd_error
&
ABRT_ERR
)
printk
(
"DriveStatusError "
);
if
(
hd_error
&
TRK0_ERR
)
printk
(
"TrackZeroNotFound "
);
if
(
hd_error
&
MARK_ERR
)
printk
(
"AddrMarkNotFound "
);
printk
(
"}"
);
printk
(
"}"
);
if
(
err
&
(
BBD_ERR
|
ECC_ERR
|
ID_ERR
|
MARK_ERR
))
{
if
(
hd_error
&
(
BBD_ERR
|
ECC_ERR
|
ID_ERR
|
MARK_ERR
))
{
if
(
CURRENT
)
printk
(
", sector=%ld"
,
CURRENT
->
sector
);
printk
(
", CHS=%d/%d/%d"
,
(
inb
(
HD_HCYL
)
<<
8
)
+
inb
(
HD_LCYL
),
printk
(
", CHS=%d/%d/%d"
,
(
inb
(
HD_HCYL
)
<<
8
)
+
inb
(
HD_LCYL
),
inb
(
HD_CURRENT
)
&
0xf
,
inb
(
HD_SECTOR
));
inb
(
HD_CURRENT
)
&
0xf
,
inb
(
HD_SECTOR
));
if
(
CURRENT
)
printk
(
", sector=%ld"
,
CURRENT
->
sector
);
}
}
printk
(
"
\n
"
);
printk
(
"
\n
"
);
}
}
restore_flags
(
flags
);
restore_flags
(
flags
);
}
}
static
int
win_result
(
void
)
void
check_status
(
void
)
{
{
int
i
=
inb_p
(
HD_STATUS
);
int
i
=
inb_p
(
HD_STATUS
);
if
((
i
&
(
BUSY_STAT
|
READY_STAT
|
WRERR_STAT
|
SEEK_STAT
|
ERR_STAT
))
if
(
!
OK_STATUS
(
i
))
{
==
(
READY_STAT
|
SEEK_STAT
))
{
dump_status
(
"check_status"
,
i
);
hd_error
=
0
;
bad_rw_intr
();
return
0
;
/* ok */
}
}
dump_status
(
"win_result"
,
i
);
return
1
;
}
}
static
int
controller_busy
(
void
);
static
int
controller_busy
(
void
)
static
int
status_ok
(
void
);
static
int
controller_ready
(
unsigned
int
drive
,
unsigned
int
head
)
{
{
int
retry
=
100
;
int
retries
=
100000
;
unsigned
char
status
;
do
{
do
{
if
(
controller_busy
()
&
BUSY_STAT
)
status
=
inb_p
(
HD_STATUS
);
return
0
;
}
while
((
status
&
BUSY_STAT
)
&&
--
retries
);
outb_p
(
0xA0
|
(
drive
<<
4
)
|
head
,
HD_CURRENT
);
return
status
;
if
(
status_ok
())
return
1
;
}
while
(
--
retry
);
return
0
;
}
}
static
int
status_ok
(
void
)
static
int
status_ok
(
void
)
...
@@ -206,7 +200,7 @@ static int status_ok(void)
...
@@ -206,7 +200,7 @@ static int status_ok(void)
unsigned
char
status
=
inb_p
(
HD_STATUS
);
unsigned
char
status
=
inb_p
(
HD_STATUS
);
if
(
status
&
BUSY_STAT
)
if
(
status
&
BUSY_STAT
)
return
1
;
return
1
;
/* Ancient, but does it make sense??? */
if
(
status
&
WRERR_STAT
)
if
(
status
&
WRERR_STAT
)
return
0
;
return
0
;
if
(
!
(
status
&
READY_STAT
))
if
(
!
(
status
&
READY_STAT
))
...
@@ -216,15 +210,18 @@ static int status_ok(void)
...
@@ -216,15 +210,18 @@ static int status_ok(void)
return
1
;
return
1
;
}
}
static
int
controller_
busy
(
voi
d
)
static
int
controller_
ready
(
unsigned
int
drive
,
unsigned
int
hea
d
)
{
{
int
retries
=
100000
;
int
retry
=
100
;
unsigned
char
status
;
do
{
do
{
status
=
inb_p
(
HD_STATUS
);
if
(
controller_busy
()
&
BUSY_STAT
)
}
while
((
status
&
BUSY_STAT
)
&&
--
retries
);
return
0
;
return
status
;
outb_p
(
0xA0
|
(
drive
<<
4
)
|
head
,
HD_CURRENT
);
if
(
status_ok
())
return
1
;
}
while
(
--
retry
);
return
0
;
}
}
static
void
hd_out
(
unsigned
int
drive
,
unsigned
int
nsect
,
unsigned
int
sect
,
static
void
hd_out
(
unsigned
int
drive
,
unsigned
int
nsect
,
unsigned
int
sect
,
...
@@ -233,13 +230,6 @@ static void hd_out(unsigned int drive,unsigned int nsect,unsigned int sect,
...
@@ -233,13 +230,6 @@ static void hd_out(unsigned int drive,unsigned int nsect,unsigned int sect,
{
{
unsigned
short
port
;
unsigned
short
port
;
#ifdef DEBUG
if
(
drive
>
1
||
head
>
15
)
{
printk
(
"bad drive mapping, trying to access drive=%d, cyl=%d, head=%d, sect=%d
\n
"
,
drive
,
cyl
,
head
,
sect
);
panic
(
"harddisk driver problem"
);
}
#endif
#if (HD_DELAY > 0)
#if (HD_DELAY > 0)
while
(
read_timer
()
-
last_req
<
HD_DELAY
)
while
(
read_timer
()
-
last_req
<
HD_DELAY
)
/* nothing */
;
/* nothing */
;
...
@@ -346,7 +336,7 @@ static void identify_intr(void)
...
@@ -346,7 +336,7 @@ static void identify_intr(void)
/*
/*
* Early model Quantum drives go weird at this point,
* Early model Quantum drives go weird at this point,
* but doing a recalibrate seems to "fix" them.
* but doing a recalibrate seems to "fix" them.
* (Doing a full reset confuses some
new
er model Quantums)
* (Doing a full reset confuses some
oth
er model Quantums)
*/
*/
if
(
!
strncmp
(
id
->
model
,
"QUANTUM"
,
7
))
if
(
!
strncmp
(
id
->
model
,
"QUANTUM"
,
7
))
special_op
[
dev
]
=
recalibrate
[
dev
]
=
1
;
special_op
[
dev
]
=
recalibrate
[
dev
]
=
1
;
...
@@ -366,7 +356,7 @@ static void set_multmode_intr(void)
...
@@ -366,7 +356,7 @@ static void set_multmode_intr(void)
sti
();
sti
();
if
(
stat
&
(
BUSY_STAT
|
ERR_STAT
))
{
if
(
stat
&
(
BUSY_STAT
|
ERR_STAT
))
{
mult_req
[
dev
]
=
mult_count
[
dev
]
=
0
;
mult_req
[
dev
]
=
mult_count
[
dev
]
=
0
;
dump_status
(
"set mult
iple
mode failed"
,
stat
);
dump_status
(
"set multmode failed"
,
stat
);
}
else
{
}
else
{
if
((
mult_count
[
dev
]
=
mult_req
[
dev
]))
if
((
mult_count
[
dev
]
=
mult_req
[
dev
]))
printk
(
" hd%c: enabled %d-sector multiple mode
\n
"
,
printk
(
" hd%c: enabled %d-sector multiple mode
\n
"
,
...
@@ -388,8 +378,7 @@ static int drive_busy(void)
...
@@ -388,8 +378,7 @@ static int drive_busy(void)
for
(
i
=
0
;
i
<
500000
;
i
++
)
{
for
(
i
=
0
;
i
<
500000
;
i
++
)
{
c
=
inb_p
(
HD_STATUS
);
c
=
inb_p
(
HD_STATUS
);
c
&=
(
BUSY_STAT
|
READY_STAT
|
SEEK_STAT
);
if
((
c
&
(
BUSY_STAT
|
READY_STAT
|
SEEK_STAT
))
==
STAT_OK
)
if
(
c
==
(
READY_STAT
|
SEEK_STAT
))
return
0
;
return
0
;
}
}
dump_status
(
"reset timed out"
,
c
);
dump_status
(
"reset timed out"
,
c
);
...
@@ -405,9 +394,9 @@ static void reset_controller(void)
...
@@ -405,9 +394,9 @@ static void reset_controller(void)
outb_p
(
hd_info
[
0
].
ctl
&
0x0f
,
HD_CMD
);
outb_p
(
hd_info
[
0
].
ctl
&
0x0f
,
HD_CMD
);
for
(
i
=
0
;
i
<
1000
;
i
++
)
nop
();
for
(
i
=
0
;
i
<
1000
;
i
++
)
nop
();
if
(
drive_busy
())
if
(
drive_busy
())
printk
(
"
HD-
controller still busy
\n
"
);
printk
(
"
hd:
controller still busy
\n
"
);
if
((
hd_error
=
inb
(
HD_ERROR
))
!=
1
)
else
if
((
hd_error
=
inb
(
HD_ERROR
))
!=
1
)
printk
(
"
HD-
controller reset failed: %02x
\n
"
,
hd_error
);
printk
(
"
hd:
controller reset failed: %02x
\n
"
,
hd_error
);
}
}
static
void
reset_hd
(
void
)
static
void
reset_hd
(
void
)
...
@@ -419,12 +408,13 @@ static void reset_hd(void)
...
@@ -419,12 +408,13 @@ static void reset_hd(void)
reset
=
0
;
reset
=
0
;
i
=
-
1
;
i
=
-
1
;
reset_controller
();
reset_controller
();
}
else
if
(
win_result
())
{
}
else
{
bad_rw_intr
();
check_status
();
if
(
reset
)
if
(
reset
)
goto
repeat
;
goto
repeat
;
}
}
if
(
++
i
<
NR_HD
)
{
if
(
++
i
<
NR_HD
)
{
special_op
[
i
]
=
recalibrate
[
i
]
=
1
;
if
(
unmask_intr
[
i
])
{
if
(
unmask_intr
[
i
])
{
unmask_intr
[
i
]
=
DEFAULT_UNMASK_INTR
;
unmask_intr
[
i
]
=
DEFAULT_UNMASK_INTR
;
printk
(
"hd%c: reset irq-unmasking to %d
\n
"
,
i
+
'a'
,
printk
(
"hd%c: reset irq-unmasking to %d
\n
"
,
i
+
'a'
,
...
@@ -444,7 +434,6 @@ static void reset_hd(void)
...
@@ -444,7 +434,6 @@ static void reset_hd(void)
hd_request
();
hd_request
();
}
}
/*
/*
* Ok, don't know what to do with the unexpected interrupts: on some machines
* Ok, don't know what to do with the unexpected interrupts: on some machines
* doing a reset and a retry seems to result in an eternal loop. Right now I
* doing a reset and a retry seems to result in an eternal loop. Right now I
...
@@ -478,11 +467,11 @@ static void bad_rw_intr(void)
...
@@ -478,11 +467,11 @@ static void bad_rw_intr(void)
dev
=
DEVICE_NR
(
CURRENT
->
dev
);
dev
=
DEVICE_NR
(
CURRENT
->
dev
);
if
(
++
CURRENT
->
errors
>=
MAX_ERRORS
||
(
hd_error
&
BBD_ERR
))
{
if
(
++
CURRENT
->
errors
>=
MAX_ERRORS
||
(
hd_error
&
BBD_ERR
))
{
end_request
(
0
);
end_request
(
0
);
special_op
[
dev
]
+
=
recalibrate
[
dev
]
=
1
;
special_op
[
dev
]
=
recalibrate
[
dev
]
=
1
;
}
else
if
(
CURRENT
->
errors
%
RESET_FREQ
==
0
)
}
else
if
(
CURRENT
->
errors
%
RESET_FREQ
==
0
)
reset
=
1
;
reset
=
1
;
else
if
((
hd_error
&
TRK0_ERR
)
||
CURRENT
->
errors
%
RECAL_FREQ
==
0
)
else
if
((
hd_error
&
TRK0_ERR
)
||
CURRENT
->
errors
%
RECAL_FREQ
==
0
)
special_op
[
dev
]
+
=
recalibrate
[
dev
]
=
1
;
special_op
[
dev
]
=
recalibrate
[
dev
]
=
1
;
/* Otherwise just retry */
/* Otherwise just retry */
}
}
...
@@ -497,9 +486,6 @@ static inline int wait_DRQ(void)
...
@@ -497,9 +486,6 @@ static inline int wait_DRQ(void)
return
-
1
;
return
-
1
;
}
}
#define STAT_MASK (BUSY_STAT | READY_STAT | WRERR_STAT | SEEK_STAT | ERR_STAT)
#define STAT_OK (READY_STAT | SEEK_STAT)
static
void
read_intr
(
void
)
static
void
read_intr
(
void
)
{
{
unsigned
int
dev
=
DEVICE_NR
(
CURRENT
->
dev
);
unsigned
int
dev
=
DEVICE_NR
(
CURRENT
->
dev
);
...
@@ -511,7 +497,7 @@ static void read_intr(void)
...
@@ -511,7 +497,7 @@ static void read_intr(void)
i
=
(
unsigned
)
inb_p
(
HD_STATUS
);
i
=
(
unsigned
)
inb_p
(
HD_STATUS
);
if
(
i
&
BUSY_STAT
)
if
(
i
&
BUSY_STAT
)
continue
;
continue
;
if
(
(
i
&
STAT_MASK
)
!=
STAT_OK
)
if
(
!
OK_STATUS
(
i
)
)
break
;
break
;
if
(
i
&
DRQ_STAT
)
if
(
i
&
DRQ_STAT
)
goto
ok_to_read
;
goto
ok_to_read
;
...
@@ -534,9 +520,9 @@ static void read_intr(void)
...
@@ -534,9 +520,9 @@ static void read_intr(void)
i
=
(
CURRENT
->
nr_sectors
-=
nsect
);
i
=
(
CURRENT
->
nr_sectors
-=
nsect
);
#ifdef DEBUG
#ifdef DEBUG
printk
(
"hd%c: read: sectors(%ld-%ld), remaining=%ld, buffer=%08lx
\n
"
,
printk
(
"hd%c: read: sectors(%ld-%ld), remaining=%ld, buffer=
0x
%08lx
\n
"
,
dev
+
'a'
,
CURRENT
->
sector
,
CURRENT
->
sector
+
nsect
,
dev
+
'a'
,
CURRENT
->
sector
,
CURRENT
->
sector
+
nsect
,
CURRENT
->
nr_sectors
,
(
long
)
CURRENT
->
buffer
+
(
nsect
<<
9
));
CURRENT
->
nr_sectors
,
(
unsigned
long
)
CURRENT
->
buffer
+
(
nsect
<<
9
));
#endif
#endif
if
((
CURRENT
->
current_nr_sectors
-=
nsect
)
<=
0
)
if
((
CURRENT
->
current_nr_sectors
-=
nsect
)
<=
0
)
end_request
(
1
);
end_request
(
1
);
...
@@ -581,7 +567,7 @@ static void multwrite_intr(void)
...
@@ -581,7 +567,7 @@ static void multwrite_intr(void)
if
(
unmask_intr
[
dev
])
if
(
unmask_intr
[
dev
])
sti
();
sti
();
if
(
((
i
=
inb_p
(
HD_STATUS
))
&
STAT_MASK
)
==
STAT_OK
)
{
if
(
OK_STATUS
(
i
=
inb_p
(
HD_STATUS
))
)
{
if
(
i
&
DRQ_STAT
)
{
if
(
i
&
DRQ_STAT
)
{
if
(
WCURRENT
.
nr_sectors
)
{
if
(
WCURRENT
.
nr_sectors
)
{
multwrite
(
dev
);
multwrite
(
dev
);
...
@@ -619,7 +605,7 @@ static void write_intr(void)
...
@@ -619,7 +605,7 @@ static void write_intr(void)
i
=
(
unsigned
)
inb_p
(
HD_STATUS
);
i
=
(
unsigned
)
inb_p
(
HD_STATUS
);
if
(
i
&
BUSY_STAT
)
if
(
i
&
BUSY_STAT
)
continue
;
continue
;
if
(
(
i
&
STAT_MASK
)
!=
STAT_OK
)
if
(
!
OK_STATUS
(
i
)
)
break
;
break
;
if
((
CURRENT
->
nr_sectors
<=
1
)
||
(
i
&
DRQ_STAT
))
if
((
CURRENT
->
nr_sectors
<=
1
)
||
(
i
&
DRQ_STAT
))
goto
ok_to_write
;
goto
ok_to_write
;
...
@@ -650,8 +636,7 @@ static void write_intr(void)
...
@@ -650,8 +636,7 @@ static void write_intr(void)
static
void
recal_intr
(
void
)
static
void
recal_intr
(
void
)
{
{
if
(
win_result
())
check_status
();
bad_rw_intr
();
#if (HD_DELAY > 0)
#if (HD_DELAY > 0)
last_req
=
read_timer
();
last_req
=
read_timer
();
#endif
#endif
...
@@ -664,116 +649,118 @@ static void recal_intr(void)
...
@@ -664,116 +649,118 @@ static void recal_intr(void)
*/
*/
static
void
hd_times_out
(
void
)
static
void
hd_times_out
(
void
)
{
{
unsigned
int
dev
;
DEVICE_INTR
=
NULL
;
DEVICE_INTR
=
NULL
;
sti
();
if
(
!
CURRENT
)
if
(
!
CURRENT
)
return
;
return
;
disable_irq
(
HD_IRQ
);
sti
();
reset
=
1
;
reset
=
1
;
printk
(
KERN_DEBUG
"HD timeout
\n
"
);
dev
=
DEVICE_NR
(
CURRENT
->
dev
);
printk
(
"hd%c: timeout
\n
"
,
dev
+
'a'
);
if
(
++
CURRENT
->
errors
>=
MAX_ERRORS
)
{
if
(
++
CURRENT
->
errors
>=
MAX_ERRORS
)
{
#ifdef DEBUG
#ifdef DEBUG
printk
(
"hd
: too many errors.
\n
"
);
printk
(
"hd
%c: too many errors
\n
"
,
dev
+
'a'
);
#endif
#endif
end_request
(
0
);
end_request
(
0
);
}
}
cli
();
cli
();
hd_request
();
hd_request
();
enable_irq
(
HD_IRQ
);
}
int
do_special_op
(
unsigned
int
dev
)
{
if
(
recalibrate
[
dev
])
{
recalibrate
[
dev
]
=
0
;
hd_out
(
dev
,
hd_info
[
dev
].
sect
,
0
,
0
,
0
,
WIN_RESTORE
,
&
recal_intr
);
return
reset
;
}
if
(
!
identified
[
dev
])
{
identified
[
dev
]
=
1
;
unmask_intr
[
dev
]
=
DEFAULT_UNMASK_INTR
;
mult_req
[
dev
]
=
DEFAULT_MULT_COUNT
;
hd_out
(
dev
,
0
,
0
,
0
,
0
,
WIN_IDENTIFY
,
&
identify_intr
);
return
reset
;
}
if
(
mult_req
[
dev
]
!=
mult_count
[
dev
])
{
hd_out
(
dev
,
mult_req
[
dev
],
0
,
0
,
0
,
WIN_SETMULT
,
&
set_multmode_intr
);
return
reset
;
}
if
(
hd_info
[
dev
].
head
>
16
)
{
printk
(
"hd%c: cannot handle device with more than 16 heads - giving up
\n
"
,
dev
+
'a'
);
end_request
(
0
);
}
special_op
[
dev
]
=
0
;
return
1
;
}
}
/*
/*
* The driver has been modified to enable interrupts a bit more: in order to
* The driver enables interrupts as much as possible. In order to do this,
* do this we first (a) disable the timeout-interrupt and (b) clear the
* (a) the device-interrupt is disabled before entering hd_request(),
* device-interrupt. This way the interrupts won't mess with out code (the
* and (b) the timeout-interrupt is disabled before the sti().
* worst that can happen is that an unexpected HD-interrupt comes in and
*
* sets the "reset" variable and starts the timer)
* Interrupts are still masked (by default) whenever we are exchanging
* data/cmds with a drive, because some drives seem to have very poor
* tolerance for latency during I/O. For devices which don't suffer from
* that problem (most don't), the unmask_intr[] flag can be set to unmask
* other interrupts during data/cmd transfers (by defining DEFAULT_UNMASK_INTR
* to 1, or by using "hdparm -u1 /dev/hd?" from the shell).
*/
*/
static
void
hd_request
(
void
)
static
void
hd_request
(
void
)
{
{
unsigned
int
block
,
dev
;
unsigned
int
dev
,
block
,
nsect
,
sec
,
track
,
head
,
cyl
;
unsigned
int
sec
,
head
,
cyl
,
track
;
unsigned
int
nsect
;
if
(
CURRENT
&&
CURRENT
->
dev
<
0
)
return
;
if
(
CURRENT
&&
CURRENT
->
dev
<
0
)
return
;
if
(
DEVICE_INTR
)
if
(
DEVICE_INTR
)
return
;
return
;
repeat:
repeat:
timer_active
&=
~
(
1
<<
HD_TIMER
);
timer_active
&=
~
(
1
<<
HD_TIMER
);
sti
();
sti
();
INIT_REQUEST
;
INIT_REQUEST
;
if
(
reset
)
{
cli
();
reset_hd
();
return
;
}
dev
=
MINOR
(
CURRENT
->
dev
);
dev
=
MINOR
(
CURRENT
->
dev
);
block
=
CURRENT
->
sector
;
block
=
CURRENT
->
sector
;
nsect
=
CURRENT
->
nr_sectors
;
nsect
=
CURRENT
->
nr_sectors
;
if
(
dev
>=
(
NR_HD
<<
6
)
||
block
>=
hd
[
dev
].
nr_sects
)
{
if
(
dev
>=
(
NR_HD
<<
6
)
||
block
>=
hd
[
dev
].
nr_sects
||
((
block
+
nsect
)
>
hd
[
dev
].
nr_sects
)
)
{
#ifdef DEBUG
#ifdef DEBUG
printk
(
"hd : attempted read for sector %d past end of device at sector %d.
\n
"
,
if
(
dev
>=
(
NR_HD
<<
6
))
block
,
hd
[
dev
].
nr_sects
);
printk
(
"hd: bad minor number: device=0x%04x
\n
"
,
CURRENT
->
dev
);
else
printk
(
"hd%c: bad access: block=%d, count=%d
\n
"
,
(
CURRENT
->
dev
>>
6
)
+
'a'
,
block
,
nsect
);
#endif
#endif
end_request
(
0
);
end_request
(
0
);
goto
repeat
;
goto
repeat
;
}
}
block
+=
hd
[
dev
].
start_sect
;
block
+=
hd
[
dev
].
start_sect
;
dev
>>=
6
;
dev
>>=
6
;
if
(
special_op
[
dev
])
{
if
(
do_special_op
(
dev
))
goto
repeat
;
return
;
}
sec
=
block
%
hd_info
[
dev
].
sect
+
1
;
sec
=
block
%
hd_info
[
dev
].
sect
+
1
;
track
=
block
/
hd_info
[
dev
].
sect
;
track
=
block
/
hd_info
[
dev
].
sect
;
head
=
track
%
hd_info
[
dev
].
head
;
head
=
track
%
hd_info
[
dev
].
head
;
cyl
=
track
/
hd_info
[
dev
].
head
;
cyl
=
track
/
hd_info
[
dev
].
head
;
#ifdef DEBUG
#ifdef DEBUG
printk
(
"hd%c : cyl = %d, head = %d, sector = %d, buffer = %08x
\n
"
,
printk
(
"hd%c: %sing: CHS=%d/%d/%d, sectors=%d, buffer=0x%08lx
\n
"
,
dev
+
'a'
,
cyl
,
head
,
sec
,
CURRENT
->
buffer
);
dev
+
'a'
,
(
CURRENT
->
cmd
==
READ
)
?
"read"
:
"writ"
,
cyl
,
head
,
sec
,
nsect
,
(
unsigned
long
)
CURRENT
->
buffer
);
#endif
#endif
if
(
!
unmask_intr
[
dev
])
if
(
!
unmask_intr
[
dev
])
cli
();
cli
();
if
(
reset
)
{
int
i
;
for
(
i
=
0
;
i
<
NR_HD
;
i
++
)
special_op
[
i
]
=
recalibrate
[
i
]
=
1
;
cli
();
/* better play it safe, as resets are the last resort */
reset_hd
();
return
;
}
if
(
special_op
[
dev
])
{
/* we use "special_op" to reduce overhead on r/w */
if
(
recalibrate
[
dev
])
{
recalibrate
[
dev
]
=
0
;
hd_out
(
dev
,
hd_info
[
dev
].
sect
,
0
,
0
,
0
,
WIN_RESTORE
,
&
recal_intr
);
if
(
reset
)
goto
repeat
;
return
;
}
if
(
!
identified
[
dev
])
{
identified
[
dev
]
=
1
;
unmask_intr
[
dev
]
=
DEFAULT_UNMASK_INTR
;
mult_req
[
dev
]
=
DEFAULT_MULT_COUNT
;
hd_out
(
dev
,
0
,
0
,
0
,
0
,
WIN_IDENTIFY
,
&
identify_intr
);
if
(
reset
)
goto
repeat
;
return
;
}
if
(
mult_req
[
dev
]
!=
mult_count
[
dev
])
{
hd_out
(
dev
,
mult_req
[
dev
],
0
,
0
,
0
,
WIN_SETMULT
,
&
set_multmode_intr
);
if
(
reset
)
goto
repeat
;
return
;
}
if
(
hd_info
[
dev
].
head
>
16
)
{
printk
(
"hd%c: cannot handle device with more than 16 heads - giving up
\n
"
,
dev
+
'a'
);
end_request
(
0
);
goto
repeat
;
}
--
special_op
[
dev
];
}
/* special_op[dev] */
if
(
CURRENT
->
cmd
==
READ
)
{
if
(
CURRENT
->
cmd
==
READ
)
{
unsigned
int
cmd
=
mult_count
[
dev
]
>
1
?
WIN_MULTREAD
:
WIN_READ
;
unsigned
int
cmd
=
mult_count
[
dev
]
>
1
?
WIN_MULTREAD
:
WIN_READ
;
hd_out
(
dev
,
nsect
,
sec
,
head
,
cyl
,
cmd
,
&
read_intr
);
hd_out
(
dev
,
nsect
,
sec
,
head
,
cyl
,
cmd
,
&
read_intr
);
if
(
reset
)
if
(
reset
)
goto
repeat
;
goto
repeat
;
#ifdef DEBUG
printk
(
"hd%c: reading %d sectors(%ld-%ld), buffer=%08lx
\n
"
,
dev
+
'a'
,
nsect
,
CURRENT
->
sector
,
CURRENT
->
sector
+
nsect
-
1
,
(
long
)
CURRENT
->
buffer
);
#endif
return
;
return
;
}
}
if
(
CURRENT
->
cmd
==
WRITE
)
{
if
(
CURRENT
->
cmd
==
WRITE
)
{
...
@@ -783,11 +770,6 @@ static void hd_request(void)
...
@@ -783,11 +770,6 @@ static void hd_request(void)
hd_out
(
dev
,
nsect
,
sec
,
head
,
cyl
,
WIN_WRITE
,
&
write_intr
);
hd_out
(
dev
,
nsect
,
sec
,
head
,
cyl
,
WIN_WRITE
,
&
write_intr
);
if
(
reset
)
if
(
reset
)
goto
repeat
;
goto
repeat
;
#ifdef DEBUG
printk
(
"hd%c: writing %d sectors(%ld-%ld), buffer=%08lx
\n
"
,
dev
+
'a'
,
nsect
,
CURRENT
->
sector
,
CURRENT
->
sector
+
nsect
-
1
,
(
long
)
CURRENT
->
buffer
);
#endif
if
(
wait_DRQ
())
{
if
(
wait_DRQ
())
{
bad_rw_intr
();
bad_rw_intr
();
goto
repeat
;
goto
repeat
;
...
@@ -795,9 +777,8 @@ static void hd_request(void)
...
@@ -795,9 +777,8 @@ static void hd_request(void)
if
(
mult_count
[
dev
])
{
if
(
mult_count
[
dev
])
{
WCURRENT
=
*
CURRENT
;
WCURRENT
=
*
CURRENT
;
multwrite
(
dev
);
multwrite
(
dev
);
}
else
{
}
else
outsw
(
HD_DATA
,
CURRENT
->
buffer
,
256
);
outsw
(
HD_DATA
,
CURRENT
->
buffer
,
256
);
}
return
;
return
;
}
}
panic
(
"unknown hd-command"
);
panic
(
"unknown hd-command"
);
...
@@ -905,11 +886,11 @@ static int hd_ioctl(struct inode * inode, struct file * file,
...
@@ -905,11 +886,11 @@ static int hd_ioctl(struct inode * inode, struct file * file,
if
(
arg
>
max_mult
[
dev
])
if
(
arg
>
max_mult
[
dev
])
err
=
-
EINVAL
;
/* out of range for device */
err
=
-
EINVAL
;
/* out of range for device */
else
if
(
mult_req
[
dev
]
!=
mult_count
[
dev
])
{
else
if
(
mult_req
[
dev
]
!=
mult_count
[
dev
])
{
++
special_op
[
dev
]
;
special_op
[
dev
]
=
1
;
err
=
-
EBUSY
;
/* busy, try again */
err
=
-
EBUSY
;
/* busy, try again */
}
else
{
}
else
{
mult_req
[
dev
]
=
arg
;
mult_req
[
dev
]
=
arg
;
++
special_op
[
dev
]
;
special_op
[
dev
]
=
1
;
err
=
0
;
err
=
0
;
}
}
restore_flags
(
flags
);
restore_flags
(
flags
);
...
@@ -1045,8 +1026,6 @@ static void hd_geninit(void)
...
@@ -1045,8 +1026,6 @@ static void hd_geninit(void)
}
}
i
=
NR_HD
;
i
=
NR_HD
;
while
(
i
--
>
0
)
{
while
(
i
--
>
0
)
{
hd
[
i
<<
6
].
nr_sects
=
0
;
if
(
bios_info
[
i
].
head
>
16
)
{
/*
/*
* The newer E-IDE BIOSs handle drives larger than 1024
* The newer E-IDE BIOSs handle drives larger than 1024
* cylinders by increasing the number of logical heads
* cylinders by increasing the number of logical heads
...
@@ -1055,12 +1034,6 @@ static void hd_geninit(void)
...
@@ -1055,12 +1034,6 @@ static void hd_geninit(void)
* what's happening here, we'll find out and correct
* what's happening here, we'll find out and correct
* it later when "identifying" the drive.
* it later when "identifying" the drive.
*/
*/
printk
(
"hd.c: IDE/ST-506 disk with more than 16 heads detected.
\n
"
);
printk
(
" (hd%c: cyl=%d, sect=%d, head=%d)
\n
"
,
i
+
'a'
,
bios_info
[
i
].
cyl
,
bios_info
[
i
].
sect
,
bios_info
[
i
].
head
);
}
hd
[
i
<<
6
].
nr_sects
=
bios_info
[
i
].
head
*
hd
[
i
<<
6
].
nr_sects
=
bios_info
[
i
].
head
*
bios_info
[
i
].
sect
*
bios_info
[
i
].
cyl
;
bios_info
[
i
].
sect
*
bios_info
[
i
].
cyl
;
hd_ident_info
[
i
]
=
(
struct
hd_driveid
*
)
kmalloc
(
512
,
GFP_KERNEL
);
hd_ident_info
[
i
]
=
(
struct
hd_driveid
*
)
kmalloc
(
512
,
GFP_KERNEL
);
...
@@ -1068,7 +1041,7 @@ static void hd_geninit(void)
...
@@ -1068,7 +1041,7 @@ static void hd_geninit(void)
}
}
if
(
NR_HD
)
{
if
(
NR_HD
)
{
if
(
request_irq
(
HD_IRQ
,
hd_interrupt
,
SA_INTERRUPT
,
"hd"
))
{
if
(
request_irq
(
HD_IRQ
,
hd_interrupt
,
SA_INTERRUPT
,
"hd"
))
{
printk
(
"hd
.c
: unable to get IRQ%d for the harddisk driver
\n
"
,
HD_IRQ
);
printk
(
"hd: unable to get IRQ%d for the harddisk driver
\n
"
,
HD_IRQ
);
NR_HD
=
0
;
NR_HD
=
0
;
}
}
}
}
...
@@ -1094,7 +1067,7 @@ static struct file_operations hd_fops = {
...
@@ -1094,7 +1067,7 @@ static struct file_operations hd_fops = {
unsigned
long
hd_init
(
unsigned
long
mem_start
,
unsigned
long
mem_end
)
unsigned
long
hd_init
(
unsigned
long
mem_start
,
unsigned
long
mem_end
)
{
{
if
(
register_blkdev
(
MAJOR_NR
,
"hd"
,
&
hd_fops
))
{
if
(
register_blkdev
(
MAJOR_NR
,
"hd"
,
&
hd_fops
))
{
printk
(
"
U
nable to get major %d for harddisk
\n
"
,
MAJOR_NR
);
printk
(
"
hd: u
nable to get major %d for harddisk
\n
"
,
MAJOR_NR
);
return
mem_start
;
return
mem_start
;
}
}
blk_dev
[
MAJOR_NR
].
request_fn
=
DEVICE_REQUEST
;
blk_dev
[
MAJOR_NR
].
request_fn
=
DEVICE_REQUEST
;
...
...
drivers/char/lp.c
View file @
f4a818e0
...
@@ -514,4 +514,12 @@ int init_module(void)
...
@@ -514,4 +514,12 @@ int init_module(void)
return
0
;
return
0
;
}
}
void
cleanup_module
(
void
)
{
if
(
MOD_IN_USE
)
printk
(
"lp: busy - remove delayed
\n
"
);
else
unregister_chrdev
(
LP_MAJOR
,
"lp"
);
}
#endif
#endif
drivers/sound/Makefile
View file @
f4a818e0
...
@@ -54,7 +54,11 @@ config: configure /usr/include/sys/soundcard.h
...
@@ -54,7 +54,11 @@ config: configure /usr/include/sys/soundcard.h
@
echo
\#
define SOUND_CONFIG_DATE
\"
`
date
`
\"
>>
local.h
@
echo
\#
define SOUND_CONFIG_DATE
\"
`
date
`
\"
>>
local.h
@
echo
\#
define SOUND_CONFIG_BY
\"
`
whoami
`
\"
>>
local.h
@
echo
\#
define SOUND_CONFIG_BY
\"
`
whoami
`
\"
>>
local.h
@
echo
\#
define SOUND_CONFIG_HOST
\"
`
hostname
`
\"
>>
local.h
@
echo
\#
define SOUND_CONFIG_HOST
\"
`
hostname
`
\"
>>
local.h
@
echo
\#
define SOUND_CONFIG_DOMAIN
\"
`
domainname
`
\"
>>
local.h
@
if
[
-x
/bin/dnsdomainname
]
;
then
\
echo
\#
define SOUND_CONFIG_DOMAIN
\"
`
dnsdomainname
`
\"
;
\
else
\
echo
\#
define SOUND_CONFIG_DOMAIN
\"
`
domainname
`
\"
;
\
fi
>>
local.h
clrconf
:
clrconf
:
rm
-f
local.h .depend
rm
-f
local.h .depend
...
...
fs/binfmt_elf.c
View file @
f4a818e0
...
@@ -612,8 +612,8 @@ load_elf_binary(struct linux_binprm * bprm, struct pt_regs * regs)
...
@@ -612,8 +612,8 @@ load_elf_binary(struct linux_binprm * bprm, struct pt_regs * regs)
current
->
mm
->
start_code
=
start_code
;
current
->
mm
->
start_code
=
start_code
;
current
->
mm
->
end_data
=
end_data
;
current
->
mm
->
end_data
=
end_data
;
current
->
mm
->
start_stack
=
bprm
->
p
;
current
->
mm
->
start_stack
=
bprm
->
p
;
current
->
suid
=
current
->
euid
=
bprm
->
e_uid
;
current
->
suid
=
current
->
euid
=
current
->
fsuid
=
bprm
->
e_uid
;
current
->
sgid
=
current
->
egid
=
bprm
->
e_gid
;
current
->
sgid
=
current
->
egid
=
current
->
fsgid
=
bprm
->
e_gid
;
/* Calling sys_brk effectively mmaps the pages that we need for the bss and break
/* Calling sys_brk effectively mmaps the pages that we need for the bss and break
sections */
sections */
...
...
fs/ext/dir.c
View file @
f4a818e0
...
@@ -20,6 +20,9 @@
...
@@ -20,6 +20,9 @@
#include <linux/ext_fs.h>
#include <linux/ext_fs.h>
#include <linux/stat.h>
#include <linux/stat.h>
#define NAME_OFFSET(de) ((int) ((de)->d_name - (char *) (de)))
#define ROUND_UP(x) (((x)+3) & ~3)
static
int
ext_dir_read
(
struct
inode
*
inode
,
struct
file
*
filp
,
char
*
buf
,
int
count
)
static
int
ext_dir_read
(
struct
inode
*
inode
,
struct
file
*
filp
,
char
*
buf
,
int
count
)
{
{
return
-
EISDIR
;
return
-
EISDIR
;
...
@@ -65,6 +68,7 @@ static int ext_readdir(struct inode * inode, struct file * filp,
...
@@ -65,6 +68,7 @@ static int ext_readdir(struct inode * inode, struct file * filp,
struct
dirent
*
dirent
,
int
count
)
struct
dirent
*
dirent
,
int
count
)
{
{
unsigned
int
i
;
unsigned
int
i
;
unsigned
int
ret
;
off_t
offset
;
off_t
offset
;
char
c
;
char
c
;
struct
buffer_head
*
bh
;
struct
buffer_head
*
bh
;
...
@@ -74,15 +78,23 @@ static int ext_readdir(struct inode * inode, struct file * filp,
...
@@ -74,15 +78,23 @@ static int ext_readdir(struct inode * inode, struct file * filp,
return
-
EBADF
;
return
-
EBADF
;
if
((
filp
->
f_pos
&
7
)
!=
0
)
if
((
filp
->
f_pos
&
7
)
!=
0
)
return
-
EBADF
;
return
-
EBADF
;
while
(
filp
->
f_pos
<
inode
->
i_size
)
{
ret
=
0
;
while
(
!
ret
&&
filp
->
f_pos
<
inode
->
i_size
)
{
offset
=
filp
->
f_pos
&
1023
;
offset
=
filp
->
f_pos
&
1023
;
bh
=
ext_bread
(
inode
,(
filp
->
f_pos
)
>>
BLOCK_SIZE_BITS
,
0
);
bh
=
ext_bread
(
inode
,(
filp
->
f_pos
)
>>
BLOCK_SIZE_BITS
,
0
);
if
(
!
bh
)
{
if
(
!
bh
)
{
filp
->
f_pos
+=
1024
-
offset
;
filp
->
f_pos
+=
1024
-
offset
;
continue
;
continue
;
}
}
for
(
i
=
0
;
i
<
1024
&&
i
<
offset
;
)
{
de
=
(
struct
ext_dir_entry
*
)
(
bh
->
b_data
+
i
);
if
(
!
de
->
rec_len
)
break
;
i
+=
de
->
rec_len
;
}
offset
=
i
;
de
=
(
struct
ext_dir_entry
*
)
(
offset
+
bh
->
b_data
);
de
=
(
struct
ext_dir_entry
*
)
(
offset
+
bh
->
b_data
);
while
(
offset
<
1024
&&
filp
->
f_pos
<
inode
->
i_size
)
{
while
(
!
ret
&&
offset
<
1024
&&
filp
->
f_pos
<
inode
->
i_size
)
{
if
(
de
->
rec_len
<
8
||
de
->
rec_len
%
8
!=
0
||
if
(
de
->
rec_len
<
8
||
de
->
rec_len
%
8
!=
0
||
de
->
rec_len
<
de
->
name_len
+
8
||
de
->
rec_len
<
de
->
name_len
+
8
||
(
de
->
rec_len
+
(
off_t
)
filp
->
f_pos
-
1
)
/
1024
>
((
off_t
)
filp
->
f_pos
/
1024
))
{
(
de
->
rec_len
+
(
off_t
)
filp
->
f_pos
-
1
)
/
1024
>
((
off_t
)
filp
->
f_pos
/
1024
))
{
...
@@ -106,8 +118,8 @@ static int ext_readdir(struct inode * inode, struct file * filp,
...
@@ -106,8 +118,8 @@ static int ext_readdir(struct inode * inode, struct file * filp,
put_fs_long
(
de
->
inode
,
&
dirent
->
d_ino
);
put_fs_long
(
de
->
inode
,
&
dirent
->
d_ino
);
put_fs_byte
(
0
,
i
+
dirent
->
d_name
);
put_fs_byte
(
0
,
i
+
dirent
->
d_name
);
put_fs_word
(
i
,
&
dirent
->
d_reclen
);
put_fs_word
(
i
,
&
dirent
->
d_reclen
);
brelse
(
bh
);
ret
=
ROUND_UP
(
NAME_OFFSET
(
dirent
)
+
i
+
1
);
return
i
;
break
;
}
}
}
}
de
=
(
struct
ext_dir_entry
*
)
((
char
*
)
de
de
=
(
struct
ext_dir_entry
*
)
((
char
*
)
de
...
@@ -115,5 +127,5 @@ static int ext_readdir(struct inode * inode, struct file * filp,
...
@@ -115,5 +127,5 @@ static int ext_readdir(struct inode * inode, struct file * filp,
}
}
brelse
(
bh
);
brelse
(
bh
);
}
}
return
0
;
return
ret
;
}
}
fs/ext2/dir.c
View file @
f4a818e0
...
@@ -22,6 +22,9 @@
...
@@ -22,6 +22,9 @@
#include <linux/sched.h>
#include <linux/sched.h>
#include <linux/stat.h>
#include <linux/stat.h>
#define NAME_OFFSET(de) ((int) ((de)->d_name - (char *) (de)))
#define ROUND_UP(x) (((x)+3) & ~3)
static
int
ext2_dir_read
(
struct
inode
*
inode
,
struct
file
*
filp
,
static
int
ext2_dir_read
(
struct
inode
*
inode
,
struct
file
*
filp
,
char
*
buf
,
int
count
)
char
*
buf
,
int
count
)
{
{
...
@@ -98,17 +101,21 @@ static int ext2_readdir (struct inode * inode, struct file * filp,
...
@@ -98,17 +101,21 @@ static int ext2_readdir (struct inode * inode, struct file * filp,
struct
dirent
*
dirent
,
int
count
)
struct
dirent
*
dirent
,
int
count
)
{
{
unsigned
long
offset
,
blk
;
unsigned
long
offset
,
blk
;
int
i
,
num
;
int
i
,
num
,
stored
,
dlen
;
struct
buffer_head
*
bh
,
*
tmp
,
*
bha
[
16
];
struct
buffer_head
*
bh
,
*
tmp
,
*
bha
[
16
];
struct
ext2_dir_entry
*
de
;
struct
ext2_dir_entry
*
de
;
struct
super_block
*
sb
;
struct
super_block
*
sb
;
int
err
;
int
err
,
version
;
if
(
!
inode
||
!
S_ISDIR
(
inode
->
i_mode
))
if
(
!
inode
||
!
S_ISDIR
(
inode
->
i_mode
))
return
-
EBADF
;
return
-
EBADF
;
sb
=
inode
->
i_sb
;
sb
=
inode
->
i_sb
;
while
(
filp
->
f_pos
<
inode
->
i_size
)
{
stored
=
0
;
bh
=
NULL
;
offset
=
filp
->
f_pos
&
(
sb
->
s_blocksize
-
1
);
offset
=
filp
->
f_pos
&
(
sb
->
s_blocksize
-
1
);
while
(
count
>
0
&&
!
stored
&&
filp
->
f_pos
<
inode
->
i_size
)
{
blk
=
(
filp
->
f_pos
)
>>
EXT2_BLOCK_SIZE_BITS
(
sb
);
blk
=
(
filp
->
f_pos
)
>>
EXT2_BLOCK_SIZE_BITS
(
sb
);
bh
=
ext2_bread
(
inode
,
blk
,
0
,
&
err
);
bh
=
ext2_bread
(
inode
,
blk
,
0
,
&
err
);
if
(
!
bh
)
{
if
(
!
bh
)
{
...
@@ -135,39 +142,78 @@ static int ext2_readdir (struct inode * inode, struct file * filp,
...
@@ -135,39 +142,78 @@ static int ext2_readdir (struct inode * inode, struct file * filp,
}
}
}
}
de
=
(
struct
ext2_dir_entry
*
)
(
offset
+
bh
->
b_data
);
revalidate:
while
(
offset
<
sb
->
s_blocksize
&&
filp
->
f_pos
<
inode
->
i_size
)
{
/* If the dir block has changed since the last call to
readdir(2), then we might be pointing to an invalid dirent
right now. Scan from the start of the block to make
sure. */
for
(
i
=
0
;
i
<
sb
->
s_blocksize
&&
i
<
offset
;
)
{
de
=
(
struct
ext2_dir_entry
*
)
(
bh
->
b_data
+
i
);
/* It's too expensive to do a full dirent test
* each time round this loop, but we do have
* to test at least that it is non-zero. A
* failure will be detected in the dirent test
* below. */
if
(
de
->
rec_len
<
EXT2_DIR_REC_LEN
(
1
))
break
;
i
+=
de
->
rec_len
;
}
offset
=
i
;
filp
->
f_pos
=
(
filp
->
f_pos
&
~
(
sb
->
s_blocksize
-
1
))
|
offset
;
while
(
count
>
0
&&
filp
->
f_pos
<
inode
->
i_size
&&
offset
<
sb
->
s_blocksize
)
{
de
=
(
struct
ext2_dir_entry
*
)
(
bh
->
b_data
+
offset
);
if
(
!
ext2_check_dir_entry
(
"ext2_readdir"
,
inode
,
de
,
if
(
!
ext2_check_dir_entry
(
"ext2_readdir"
,
inode
,
de
,
bh
,
offset
))
{
bh
,
offset
))
{
/* On error, skip the f_pos to the next block. */
filp
->
f_pos
=
(
filp
->
f_pos
&
(
sb
->
s_blocksize
-
1
))
+
sb
->
s_blocksize
;
brelse
(
bh
);
brelse
(
bh
);
return
0
;
return
stored
;
}
}
offset
+=
de
->
rec_len
;
filp
->
f_pos
+=
de
->
rec_len
;
if
(
de
->
inode
)
{
if
(
de
->
inode
)
{
memcpy_tofs
(
dirent
->
d_name
,
de
->
name
,
dlen
=
ROUND_UP
(
NAME_OFFSET
(
dirent
)
de
->
name_len
);
+
de
->
name_len
+
1
);
/* Old libc libraries always use a count of 1. */
if
(
count
==
1
&&
!
stored
)
count
=
dlen
;
if
(
count
<
dlen
)
{
count
=
0
;
break
;
}
/* We might block in the next section
* if the data destination is
* currently swapped out. So, use a
* version stamp to detect whether or
* not the directory has been modified
* during the copy operation. */
version
=
inode
->
i_version
;
i
=
de
->
name_len
;
memcpy_tofs
(
dirent
->
d_name
,
de
->
name
,
i
);
put_fs_long
(
de
->
inode
,
&
dirent
->
d_ino
);
put_fs_long
(
de
->
inode
,
&
dirent
->
d_ino
);
put_fs_byte
(
0
,
de
->
name_len
+
dirent
->
d_name
);
put_fs_byte
(
0
,
dirent
->
d_name
+
i
);
put_fs_word
(
de
->
name_len
,
&
dirent
->
d_reclen
);
put_fs_word
(
i
,
&
dirent
->
d_reclen
);
put_fs_long
(
dlen
,
&
dirent
->
d_off
);
if
(
version
!=
inode
->
i_version
)
goto
revalidate
;
dcache_add
(
inode
,
de
->
name
,
de
->
name_len
,
dcache_add
(
inode
,
de
->
name
,
de
->
name_len
,
de
->
inode
);
de
->
inode
);
i
=
de
->
name_len
;
brelse
(
bh
);
stored
+=
dlen
;
if
(
!
IS_RDONLY
(
inode
))
{
count
-=
dlen
;
inode
->
i_atime
=
CURRENT_TIME
;
((
char
*
)
dirent
)
+=
dlen
;
inode
->
i_dirt
=
1
;
}
return
i
;
}
}
de
=
(
struct
ext2_dir_entry
*
)
((
char
*
)
de
+
offset
+=
de
->
rec_len
;
de
->
rec_len
)
;
filp
->
f_pos
+=
de
->
rec_len
;
}
}
offset
=
0
;
brelse
(
bh
);
brelse
(
bh
);
}
}
if
(
!
IS_RDONLY
(
inode
))
{
if
(
!
IS_RDONLY
(
inode
))
{
inode
->
i_atime
=
CURRENT_TIME
;
inode
->
i_atime
=
CURRENT_TIME
;
inode
->
i_dirt
=
1
;
inode
->
i_dirt
=
1
;
}
}
return
0
;
return
stored
;
}
}
fs/ext2/namei.c
View file @
f4a818e0
...
@@ -318,6 +318,7 @@ static struct buffer_head * ext2_add_entry (struct inode * dir,
...
@@ -318,6 +318,7 @@ static struct buffer_head * ext2_add_entry (struct inode * dir,
*/
*/
dir
->
i_mtime
=
dir
->
i_ctime
=
CURRENT_TIME
;
dir
->
i_mtime
=
dir
->
i_ctime
=
CURRENT_TIME
;
dir
->
i_dirt
=
1
;
dir
->
i_dirt
=
1
;
dir
->
i_version
=
++
event
;
mark_buffer_dirty
(
bh
,
1
);
mark_buffer_dirty
(
bh
,
1
);
*
res_dir
=
de
;
*
res_dir
=
de
;
*
err
=
0
;
*
err
=
0
;
...
@@ -923,8 +924,10 @@ static int do_ext2_rename (struct inode * old_dir, const char * old_name,
...
@@ -923,8 +924,10 @@ static int do_ext2_rename (struct inode * old_dir, const char * old_name,
goto
start_up
;
goto
start_up
;
try_again:
try_again:
if
(
new_bh
&&
new_de
)
if
(
new_bh
&&
new_de
)
{
ext2_delete_entry
(
new_de
,
new_bh
);
ext2_delete_entry
(
new_de
,
new_bh
);
new_dir
->
i_version
=
++
event
;
}
brelse
(
old_bh
);
brelse
(
old_bh
);
brelse
(
new_bh
);
brelse
(
new_bh
);
brelse
(
dir_bh
);
brelse
(
dir_bh
);
...
@@ -1002,6 +1005,7 @@ static int do_ext2_rename (struct inode * old_dir, const char * old_name,
...
@@ -1002,6 +1005,7 @@ static int do_ext2_rename (struct inode * old_dir, const char * old_name,
&
retval
);
&
retval
);
if
(
!
new_bh
)
if
(
!
new_bh
)
goto
end_rename
;
goto
end_rename
;
new_dir
->
i_version
=
++
event
;
/*
/*
* sanity checking before doing the rename - avoid races
* sanity checking before doing the rename - avoid races
*/
*/
...
@@ -1015,7 +1019,6 @@ static int do_ext2_rename (struct inode * old_dir, const char * old_name,
...
@@ -1015,7 +1019,6 @@ static int do_ext2_rename (struct inode * old_dir, const char * old_name,
* ok, that's it
* ok, that's it
*/
*/
new_de
->
inode
=
old_inode
->
i_ino
;
new_de
->
inode
=
old_inode
->
i_ino
;
new_dir
->
i_version
=
++
event
;
dcache_add
(
new_dir
,
new_de
->
name
,
new_de
->
name_len
,
new_de
->
inode
);
dcache_add
(
new_dir
,
new_de
->
name
,
new_de
->
name_len
,
new_de
->
inode
);
retval
=
ext2_delete_entry
(
old_de
,
old_bh
);
retval
=
ext2_delete_entry
(
old_de
,
old_bh
);
if
(
retval
==
-
ENOENT
)
if
(
retval
==
-
ENOENT
)
...
...
fs/hpfs/hpfs_fs.c
View file @
f4a818e0
...
@@ -117,6 +117,9 @@
...
@@ -117,6 +117,9 @@
/* notation */
/* notation */
#define NAME_OFFSET(de) ((int) ((de)->d_name - (char *) (de)))
#define ROUND_UP(x) (((x)+3) & ~3)
#define little_ushort(x) (*(unsigned short *) &(x))
#define little_ushort(x) (*(unsigned short *) &(x))
typedef
void
nonconst
;
typedef
void
nonconst
;
...
@@ -1345,13 +1348,13 @@ static int hpfs_readdir(struct inode *inode, struct file *filp,
...
@@ -1345,13 +1348,13 @@ static int hpfs_readdir(struct inode *inode, struct file *filp,
case
0
:
case
0
:
write_one_dirent
(
dirent
,
"."
,
1
,
inode
->
i_ino
,
lc
);
write_one_dirent
(
dirent
,
"."
,
1
,
inode
->
i_ino
,
lc
);
filp
->
f_pos
=
-
1
;
filp
->
f_pos
=
-
1
;
return
1
;
return
ROUND_UP
(
NAME_OFFSET
(
dirent
)
+
2
)
;
case
-
1
:
case
-
1
:
write_one_dirent
(
dirent
,
".."
,
2
,
write_one_dirent
(
dirent
,
".."
,
2
,
inode
->
i_hpfs_parent_dir
,
lc
);
inode
->
i_hpfs_parent_dir
,
lc
);
filp
->
f_pos
=
1
;
filp
->
f_pos
=
1
;
return
2
;
return
ROUND_UP
(
NAME_OFFSET
(
dirent
)
+
3
)
;
case
-
2
:
case
-
2
:
return
0
;
return
0
;
...
@@ -1371,7 +1374,7 @@ static int hpfs_readdir(struct inode *inode, struct file *filp,
...
@@ -1371,7 +1374,7 @@ static int hpfs_readdir(struct inode *inode, struct file *filp,
write_one_dirent
(
dirent
,
de
->
name
,
namelen
,
ino
,
lc
);
write_one_dirent
(
dirent
,
de
->
name
,
namelen
,
ino
,
lc
);
brelse4
(
&
qbh
);
brelse4
(
&
qbh
);
return
namelen
;
return
ROUND_UP
(
NAME_OFFSET
(
dirent
)
+
namelen
+
1
)
;
}
}
}
}
...
...
fs/isofs/dir.c
View file @
f4a818e0
...
@@ -22,6 +22,9 @@
...
@@ -22,6 +22,9 @@
#include <linux/sched.h>
#include <linux/sched.h>
#include <linux/locks.h>
#include <linux/locks.h>
#define NAME_OFFSET(de) ((int) ((de)->d_name - (char *) (de)))
#define ROUND_UP(x) (((x)+3) & ~3)
static
int
isofs_readdir
(
struct
inode
*
,
struct
file
*
,
struct
dirent
*
,
int
);
static
int
isofs_readdir
(
struct
inode
*
,
struct
file
*
,
struct
dirent
*
,
int
);
static
struct
file_operations
isofs_dir_operations
=
{
static
struct
file_operations
isofs_dir_operations
=
{
...
@@ -228,7 +231,7 @@ static int isofs_readdir(struct inode * inode, struct file * filp,
...
@@ -228,7 +231,7 @@ static int isofs_readdir(struct inode * inode, struct file * filp,
put_fs_byte
(
0
,
i
+
dirent
->
d_name
);
put_fs_byte
(
0
,
i
+
dirent
->
d_name
);
put_fs_word
(
i
,
&
dirent
->
d_reclen
);
put_fs_word
(
i
,
&
dirent
->
d_reclen
);
brelse
(
bh
);
brelse
(
bh
);
return
i
;
return
ROUND_UP
(
NAME_OFFSET
(
dirent
)
+
i
+
1
)
;
}
}
}
}
/* We go here for any condition we cannot handle. We also drop through
/* We go here for any condition we cannot handle. We also drop through
...
...
fs/minix/dir.c
View file @
f4a818e0
...
@@ -13,6 +13,9 @@
...
@@ -13,6 +13,9 @@
#include <linux/minix_fs.h>
#include <linux/minix_fs.h>
#include <linux/stat.h>
#include <linux/stat.h>
#define NAME_OFFSET(de) ((int) ((de)->d_name - (char *) (de)))
#define ROUND_UP(x) (((x)+3) & ~3)
static
int
minix_dir_read
(
struct
inode
*
inode
,
struct
file
*
filp
,
char
*
buf
,
int
count
)
static
int
minix_dir_read
(
struct
inode
*
inode
,
struct
file
*
filp
,
char
*
buf
,
int
count
)
{
{
return
-
EISDIR
;
return
-
EISDIR
;
...
@@ -57,7 +60,8 @@ struct inode_operations minix_dir_inode_operations = {
...
@@ -57,7 +60,8 @@ struct inode_operations minix_dir_inode_operations = {
static
int
minix_readdir
(
struct
inode
*
inode
,
struct
file
*
filp
,
static
int
minix_readdir
(
struct
inode
*
inode
,
struct
file
*
filp
,
struct
dirent
*
dirent
,
int
count
)
struct
dirent
*
dirent
,
int
count
)
{
{
unsigned
int
offset
,
i
;
unsigned
int
offset
,
i
,
ret
;
int
version
;
char
c
;
char
c
;
struct
buffer_head
*
bh
;
struct
buffer_head
*
bh
;
struct
minix_dir_entry
*
de
;
struct
minix_dir_entry
*
de
;
...
@@ -68,17 +72,19 @@ static int minix_readdir(struct inode * inode, struct file * filp,
...
@@ -68,17 +72,19 @@ static int minix_readdir(struct inode * inode, struct file * filp,
info
=
&
inode
->
i_sb
->
u
.
minix_sb
;
info
=
&
inode
->
i_sb
->
u
.
minix_sb
;
if
(
filp
->
f_pos
&
(
info
->
s_dirsize
-
1
))
if
(
filp
->
f_pos
&
(
info
->
s_dirsize
-
1
))
return
-
EBADF
;
return
-
EBADF
;
while
(
filp
->
f_pos
<
inode
->
i_size
)
{
ret
=
0
;
while
(
!
ret
&&
filp
->
f_pos
<
inode
->
i_size
)
{
offset
=
filp
->
f_pos
&
1023
;
offset
=
filp
->
f_pos
&
1023
;
bh
=
minix_bread
(
inode
,(
filp
->
f_pos
)
>>
BLOCK_SIZE_BITS
,
0
);
bh
=
minix_bread
(
inode
,(
filp
->
f_pos
)
>>
BLOCK_SIZE_BITS
,
0
);
if
(
!
bh
)
{
if
(
!
bh
)
{
filp
->
f_pos
+=
1024
-
offset
;
filp
->
f_pos
+=
1024
-
offset
;
continue
;
continue
;
}
}
while
(
offset
<
1024
&&
filp
->
f_pos
<
inode
->
i_size
)
{
while
(
!
ret
&&
offset
<
1024
&&
filp
->
f_pos
<
inode
->
i_size
)
{
de
=
(
struct
minix_dir_entry
*
)
(
offset
+
bh
->
b_data
);
de
=
(
struct
minix_dir_entry
*
)
(
offset
+
bh
->
b_data
);
offset
+=
info
->
s_dirsize
;
offset
+=
info
->
s_dirsize
;
filp
->
f_pos
+=
info
->
s_dirsize
;
filp
->
f_pos
+=
info
->
s_dirsize
;
retry:
if
(
de
->
inode
)
{
if
(
de
->
inode
)
{
for
(
i
=
0
;
i
<
info
->
s_namelen
;
i
++
)
for
(
i
=
0
;
i
<
info
->
s_namelen
;
i
++
)
if
((
c
=
de
->
name
[
i
])
!=
0
)
if
((
c
=
de
->
name
[
i
])
!=
0
)
...
@@ -86,15 +92,17 @@ static int minix_readdir(struct inode * inode, struct file * filp,
...
@@ -86,15 +92,17 @@ static int minix_readdir(struct inode * inode, struct file * filp,
else
else
break
;
break
;
if
(
i
)
{
if
(
i
)
{
version
=
inode
->
i_version
;
put_fs_long
(
de
->
inode
,
&
dirent
->
d_ino
);
put_fs_long
(
de
->
inode
,
&
dirent
->
d_ino
);
put_fs_byte
(
0
,
i
+
dirent
->
d_name
);
put_fs_byte
(
0
,
i
+
dirent
->
d_name
);
put_fs_word
(
i
,
&
dirent
->
d_reclen
);
put_fs_word
(
i
,
&
dirent
->
d_reclen
);
brelse
(
bh
);
if
(
version
!=
inode
->
i_version
)
return
i
;
goto
retry
;
ret
=
ROUND_UP
(
NAME_OFFSET
(
dirent
)
+
i
+
1
);
}
}
}
}
}
}
brelse
(
bh
);
brelse
(
bh
);
}
}
return
0
;
return
ret
;
}
}
fs/minix/namei.c
View file @
f4a818e0
...
@@ -195,6 +195,7 @@ static int minix_add_entry(struct inode * dir,
...
@@ -195,6 +195,7 @@ static int minix_add_entry(struct inode * dir,
dir
->
i_mtime
=
dir
->
i_ctime
=
CURRENT_TIME
;
dir
->
i_mtime
=
dir
->
i_ctime
=
CURRENT_TIME
;
for
(
i
=
0
;
i
<
info
->
s_namelen
;
i
++
)
for
(
i
=
0
;
i
<
info
->
s_namelen
;
i
++
)
de
->
name
[
i
]
=
(
i
<
namelen
)
?
name
[
i
]
:
0
;
de
->
name
[
i
]
=
(
i
<
namelen
)
?
name
[
i
]
:
0
;
dir
->
i_version
=
++
event
;
mark_buffer_dirty
(
bh
,
1
);
mark_buffer_dirty
(
bh
,
1
);
*
res_dir
=
de
;
*
res_dir
=
de
;
break
;
break
;
...
@@ -470,11 +471,12 @@ int minix_rmdir(struct inode * dir, const char * name, int len)
...
@@ -470,11 +471,12 @@ int minix_rmdir(struct inode * dir, const char * name, int len)
if
(
inode
->
i_nlink
!=
2
)
if
(
inode
->
i_nlink
!=
2
)
printk
(
"empty directory has nlink!=2 (%d)
\n
"
,
inode
->
i_nlink
);
printk
(
"empty directory has nlink!=2 (%d)
\n
"
,
inode
->
i_nlink
);
de
->
inode
=
0
;
de
->
inode
=
0
;
dir
->
i_version
=
++
event
;
mark_buffer_dirty
(
bh
,
1
);
mark_buffer_dirty
(
bh
,
1
);
inode
->
i_nlink
=
0
;
inode
->
i_nlink
=
0
;
inode
->
i_dirt
=
1
;
inode
->
i_dirt
=
1
;
dir
->
i_nlink
--
;
inode
->
i_ctime
=
dir
->
i_ctime
=
dir
->
i_mtime
=
CURRENT_TIME
;
inode
->
i_ctime
=
dir
->
i_ctime
=
dir
->
i_mtime
=
CURRENT_TIME
;
dir
->
i_nlink
--
;
dir
->
i_dirt
=
1
;
dir
->
i_dirt
=
1
;
retval
=
0
;
retval
=
0
;
end_rmdir:
end_rmdir:
...
@@ -523,6 +525,7 @@ int minix_unlink(struct inode * dir, const char * name, int len)
...
@@ -523,6 +525,7 @@ int minix_unlink(struct inode * dir, const char * name, int len)
inode
->
i_nlink
=
1
;
inode
->
i_nlink
=
1
;
}
}
de
->
inode
=
0
;
de
->
inode
=
0
;
dir
->
i_version
=
++
event
;
mark_buffer_dirty
(
bh
,
1
);
mark_buffer_dirty
(
bh
,
1
);
dir
->
i_ctime
=
dir
->
i_mtime
=
CURRENT_TIME
;
dir
->
i_ctime
=
dir
->
i_mtime
=
CURRENT_TIME
;
dir
->
i_dirt
=
1
;
dir
->
i_dirt
=
1
;
...
@@ -768,8 +771,10 @@ static int do_minix_rename(struct inode * old_dir, const char * old_name, int ol
...
@@ -768,8 +771,10 @@ static int do_minix_rename(struct inode * old_dir, const char * old_name, int ol
new_de
->
inode
=
old_inode
->
i_ino
;
new_de
->
inode
=
old_inode
->
i_ino
;
old_dir
->
i_ctime
=
old_dir
->
i_mtime
=
CURRENT_TIME
;
old_dir
->
i_ctime
=
old_dir
->
i_mtime
=
CURRENT_TIME
;
old_dir
->
i_dirt
=
1
;
old_dir
->
i_dirt
=
1
;
old_dir
->
i_version
=
++
event
;
new_dir
->
i_ctime
=
new_dir
->
i_mtime
=
CURRENT_TIME
;
new_dir
->
i_ctime
=
new_dir
->
i_mtime
=
CURRENT_TIME
;
new_dir
->
i_dirt
=
1
;
new_dir
->
i_dirt
=
1
;
new_dir
->
i_version
=
++
event
;
if
(
new_inode
)
{
if
(
new_inode
)
{
new_inode
->
i_nlink
--
;
new_inode
->
i_nlink
--
;
new_inode
->
i_ctime
=
CURRENT_TIME
;
new_inode
->
i_ctime
=
CURRENT_TIME
;
...
...
fs/msdos/dir.c
View file @
f4a818e0
...
@@ -14,6 +14,9 @@
...
@@ -14,6 +14,9 @@
#include <linux/stat.h>
#include <linux/stat.h>
#include <linux/string.h>
#include <linux/string.h>
#define NAME_OFFSET(de) ((int) ((de)->d_name - (char *) (de)))
#define ROUND_UP(x) (((x)+3) & ~3)
static
int
msdos_dir_read
(
struct
inode
*
inode
,
struct
file
*
filp
,
char
*
buf
,
int
count
)
static
int
msdos_dir_read
(
struct
inode
*
inode
,
struct
file
*
filp
,
char
*
buf
,
int
count
)
{
{
...
@@ -73,7 +76,7 @@ int msdos_readdir(
...
@@ -73,7 +76,7 @@ int msdos_readdir(
put_fs_long
(
MSDOS_ROOT_INO
,
&
dirent
->
d_ino
);
put_fs_long
(
MSDOS_ROOT_INO
,
&
dirent
->
d_ino
);
put_fs_byte
(
0
,
dirent
->
d_name
+
i
);
put_fs_byte
(
0
,
dirent
->
d_name
+
i
);
put_fs_word
(
i
,
&
dirent
->
d_reclen
);
put_fs_word
(
i
,
&
dirent
->
d_reclen
);
return
i
;
return
ROUND_UP
(
NAME_OFFSET
(
dirent
)
+
i
+
1
)
;
}
}
}
}
if
(
filp
->
f_pos
&
(
sizeof
(
struct
msdos_dir_entry
)
-
1
))
return
-
ENOENT
;
if
(
filp
->
f_pos
&
(
sizeof
(
struct
msdos_dir_entry
)
-
1
))
return
-
ENOENT
;
...
@@ -110,7 +113,7 @@ int msdos_readdir(
...
@@ -110,7 +113,7 @@ int msdos_readdir(
memcpy_tofs
(
dirent
->
d_name
,
bufname
,
i
+
1
);
memcpy_tofs
(
dirent
->
d_name
,
bufname
,
i
+
1
);
put_fs_word
(
i
,
&
dirent
->
d_reclen
);
put_fs_word
(
i
,
&
dirent
->
d_reclen
);
brelse
(
bh
);
brelse
(
bh
);
return
i
;
return
ROUND_UP
(
NAME_OFFSET
(
dirent
)
+
i
+
1
)
;
}
}
}
}
}
}
...
...
fs/nfs/dir.c
View file @
f4a818e0
...
@@ -18,6 +18,9 @@
...
@@ -18,6 +18,9 @@
#include <asm/segment.h>
/* for fs functions */
#include <asm/segment.h>
/* for fs functions */
#define NAME_OFFSET(de) ((int) ((de)->d_name - (char *) (de)))
#define ROUND_UP(x) (((x)+3) & ~3)
static
int
nfs_dir_read
(
struct
inode
*
,
struct
file
*
filp
,
char
*
buf
,
static
int
nfs_dir_read
(
struct
inode
*
,
struct
file
*
filp
,
char
*
buf
,
int
count
);
int
count
);
static
int
nfs_readdir
(
struct
inode
*
,
struct
file
*
,
struct
dirent
*
,
int
);
static
int
nfs_readdir
(
struct
inode
*
,
struct
file
*
,
struct
dirent
*
,
int
);
...
@@ -153,7 +156,7 @@ static int nfs_readdir(struct inode *inode, struct file *filp,
...
@@ -153,7 +156,7 @@ static int nfs_readdir(struct inode *inode, struct file *filp,
put_fs_long
(
entry
->
fileid
,
&
dirent
->
d_ino
);
put_fs_long
(
entry
->
fileid
,
&
dirent
->
d_ino
);
put_fs_word
(
i
,
&
dirent
->
d_reclen
);
put_fs_word
(
i
,
&
dirent
->
d_reclen
);
filp
->
f_pos
=
entry
->
cookie
;
filp
->
f_pos
=
entry
->
cookie
;
return
i
;
return
ROUND_UP
(
NAME_OFFSET
(
dirent
)
+
i
+
1
)
;
}
}
return
0
;
return
0
;
}
}
...
...
fs/proc/link.c
View file @
f4a818e0
...
@@ -92,7 +92,8 @@ static int proc_fd_dupf(struct inode * inode, struct file * f)
...
@@ -92,7 +92,8 @@ static int proc_fd_dupf(struct inode * inode, struct file * f)
new_f
->
f_count
++
;
new_f
->
f_count
++
;
current
->
files
->
fd
[
fd
]
=
new_f
;
current
->
files
->
fd
[
fd
]
=
new_f
;
f
->
f_count
--
;
if
(
!--
f
->
f_count
)
iput
(
f
->
f_inode
);
return
0
;
return
0
;
}
}
...
...
fs/read_write.c
View file @
f4a818e0
...
@@ -13,8 +13,14 @@
...
@@ -13,8 +13,14 @@
#include <asm/segment.h>
#include <asm/segment.h>
/*
/*
* Count is not yet used: but we'll probably support reading several entries
* Count is now a supported feature, but currently only the ext2fs
* at once in the future. Use count=1 in the library for future expansions.
* uses it. A count value of 1 is supported for compatibility with
* earlier libraries, but larger values are supported: count should
* indicate the total buffer space available for filling with dirents.
* The d_off entry in the dirents will then indicate the offset from
* each dirent to the next, and the return value will indicate the
* number of bytes written. All dirents will be written at
* word-aligned addresses. [sct Oct 1994]
*/
*/
asmlinkage
int
sys_readdir
(
unsigned
int
fd
,
struct
dirent
*
dirent
,
unsigned
int
count
)
asmlinkage
int
sys_readdir
(
unsigned
int
fd
,
struct
dirent
*
dirent
,
unsigned
int
count
)
{
{
...
@@ -62,8 +68,10 @@ asmlinkage int sys_lseek(unsigned int fd, off_t offset, unsigned int origin)
...
@@ -62,8 +68,10 @@ asmlinkage int sys_lseek(unsigned int fd, off_t offset, unsigned int origin)
}
}
if
(
tmp
<
0
)
if
(
tmp
<
0
)
return
-
EINVAL
;
return
-
EINVAL
;
if
(
tmp
!=
file
->
f_pos
)
{
file
->
f_pos
=
tmp
;
file
->
f_pos
=
tmp
;
file
->
f_reada
=
0
;
file
->
f_reada
=
0
;
}
return
file
->
f_pos
;
return
file
->
f_pos
;
}
}
...
...
fs/sysv/dir.c
View file @
f4a818e0
...
@@ -20,6 +20,9 @@
...
@@ -20,6 +20,9 @@
#include <linux/sysv_fs.h>
#include <linux/sysv_fs.h>
#include <linux/stat.h>
#include <linux/stat.h>
#define NAME_OFFSET(de) ((int) ((de)->d_name - (char *) (de)))
#define ROUND_UP(x) (((x)+3) & ~3)
static
int
sysv_dir_read
(
struct
inode
*
inode
,
struct
file
*
filp
,
char
*
buf
,
int
count
)
static
int
sysv_dir_read
(
struct
inode
*
inode
,
struct
file
*
filp
,
char
*
buf
,
int
count
)
{
{
return
-
EISDIR
;
return
-
EISDIR
;
...
@@ -100,7 +103,7 @@ static int sysv_readdir(struct inode * inode, struct file * filp,
...
@@ -100,7 +103,7 @@ static int sysv_readdir(struct inode * inode, struct file * filp,
put_fs_byte
(
0
,
i
+
dirent
->
d_name
);
put_fs_byte
(
0
,
i
+
dirent
->
d_name
);
put_fs_word
(
i
,
&
dirent
->
d_reclen
);
put_fs_word
(
i
,
&
dirent
->
d_reclen
);
brelse
(
bh
);
brelse
(
bh
);
return
i
;
return
ROUND_UP
(
NAME_OFFSET
(
dirent
)
+
i
+
1
)
;
}
}
}
}
}
}
...
...
fs/xiafs/dir.c
View file @
f4a818e0
...
@@ -19,6 +19,9 @@
...
@@ -19,6 +19,9 @@
#include "xiafs_mac.h"
#include "xiafs_mac.h"
#define NAME_OFFSET(de) ((int) ((de)->d_name - (char *) (de)))
#define ROUND_UP(x) (((x)+3) & ~3)
static
int
xiafs_dir_read
(
struct
inode
*
,
struct
file
*
,
char
*
,
int
);
static
int
xiafs_dir_read
(
struct
inode
*
,
struct
file
*
,
char
*
,
int
);
static
int
xiafs_readdir
(
struct
inode
*
,
struct
file
*
,
struct
dirent
*
,
int
);
static
int
xiafs_readdir
(
struct
inode
*
,
struct
file
*
,
struct
dirent
*
,
int
);
...
@@ -65,7 +68,7 @@ static int xiafs_dir_read(struct inode * inode,
...
@@ -65,7 +68,7 @@ static int xiafs_dir_read(struct inode * inode,
static
int
xiafs_readdir
(
struct
inode
*
inode
,
static
int
xiafs_readdir
(
struct
inode
*
inode
,
struct
file
*
filp
,
struct
dirent
*
dirent
,
int
count
)
struct
file
*
filp
,
struct
dirent
*
dirent
,
int
count
)
{
{
u_int
offset
,
i
;
u_int
offset
,
i
,
ret
;
struct
buffer_head
*
bh
;
struct
buffer_head
*
bh
;
struct
xiafs_direct
*
de
;
struct
xiafs_direct
*
de
;
...
@@ -73,15 +76,24 @@ static int xiafs_readdir(struct inode * inode,
...
@@ -73,15 +76,24 @@ static int xiafs_readdir(struct inode * inode,
return
-
EBADF
;
return
-
EBADF
;
if
(
inode
->
i_size
&
(
XIAFS_ZSIZE
(
inode
->
i_sb
)
-
1
)
)
if
(
inode
->
i_size
&
(
XIAFS_ZSIZE
(
inode
->
i_sb
)
-
1
)
)
return
-
EBADF
;
return
-
EBADF
;
while
(
filp
->
f_pos
<
inode
->
i_size
)
{
ret
=
0
;
while
(
!
ret
&&
filp
->
f_pos
<
inode
->
i_size
)
{
offset
=
filp
->
f_pos
&
(
XIAFS_ZSIZE
(
inode
->
i_sb
)
-
1
);
offset
=
filp
->
f_pos
&
(
XIAFS_ZSIZE
(
inode
->
i_sb
)
-
1
);
bh
=
xiafs_bread
(
inode
,
filp
->
f_pos
>>
XIAFS_ZSIZE_BITS
(
inode
->
i_sb
),
0
);
bh
=
xiafs_bread
(
inode
,
filp
->
f_pos
>>
XIAFS_ZSIZE_BITS
(
inode
->
i_sb
),
0
);
if
(
!
bh
)
{
if
(
!
bh
)
{
filp
->
f_pos
+=
XIAFS_ZSIZE
(
inode
->
i_sb
)
-
offset
;
filp
->
f_pos
+=
XIAFS_ZSIZE
(
inode
->
i_sb
)
-
offset
;
continue
;
continue
;
}
}
for
(
i
=
0
;
i
<
XIAFS_ZSIZE
(
inode
->
i_sb
)
&&
i
<
offset
;
)
{
de
=
(
struct
xiafs_direct
*
)
(
bh
->
b_data
+
i
);
if
(
!
de
->
d_rec_len
)
break
;
i
+=
de
->
d_rec_len
;
}
offset
=
i
;
de
=
(
struct
xiafs_direct
*
)
(
offset
+
bh
->
b_data
);
de
=
(
struct
xiafs_direct
*
)
(
offset
+
bh
->
b_data
);
while
(
offset
<
XIAFS_ZSIZE
(
inode
->
i_sb
)
&&
filp
->
f_pos
<
inode
->
i_size
)
{
while
(
!
ret
&&
offset
<
XIAFS_ZSIZE
(
inode
->
i_sb
)
&&
filp
->
f_pos
<
inode
->
i_size
)
{
if
(
de
->
d_ino
>
inode
->
i_sb
->
u
.
xiafs_sb
.
s_ninodes
||
if
(
de
->
d_ino
>
inode
->
i_sb
->
u
.
xiafs_sb
.
s_ninodes
||
de
->
d_rec_len
<
12
||
de
->
d_rec_len
<
12
||
(
char
*
)
de
+
de
->
d_rec_len
>
XIAFS_ZSIZE
(
inode
->
i_sb
)
+
bh
->
b_data
||
(
char
*
)
de
+
de
->
d_rec_len
>
XIAFS_ZSIZE
(
inode
->
i_sb
)
+
bh
->
b_data
||
...
@@ -100,12 +112,12 @@ static int xiafs_readdir(struct inode * inode,
...
@@ -100,12 +112,12 @@ static int xiafs_readdir(struct inode * inode,
put_fs_byte
(
0
,
i
+
dirent
->
d_name
);
put_fs_byte
(
0
,
i
+
dirent
->
d_name
);
put_fs_long
(
de
->
d_ino
,
&
dirent
->
d_ino
);
put_fs_long
(
de
->
d_ino
,
&
dirent
->
d_ino
);
put_fs_word
(
i
,
&
dirent
->
d_reclen
);
put_fs_word
(
i
,
&
dirent
->
d_reclen
);
brelse
(
bh
);
if
(
!
IS_RDONLY
(
inode
))
{
if
(
!
IS_RDONLY
(
inode
))
{
inode
->
i_atime
=
CURRENT_TIME
;
inode
->
i_atime
=
CURRENT_TIME
;
inode
->
i_dirt
=
1
;
inode
->
i_dirt
=
1
;
}
}
return
i
;
ret
=
ROUND_UP
(
NAME_OFFSET
(
dirent
)
+
i
+
1
);
break
;
}
}
de
=
(
struct
xiafs_direct
*
)
(
offset
+
bh
->
b_data
);
de
=
(
struct
xiafs_direct
*
)
(
offset
+
bh
->
b_data
);
}
}
...
...
include/linux/hdreg.h
View file @
f4a818e0
...
@@ -71,7 +71,6 @@ struct hd_geometry {
...
@@ -71,7 +71,6 @@ struct hd_geometry {
#define HDIO_SETUNMASKINTR 0x303
#define HDIO_SETUNMASKINTR 0x303
#define HDIO_GETMULTCOUNT 0x304
#define HDIO_GETMULTCOUNT 0x304
#define HDIO_SETMULTCOUNT 0x305
#define HDIO_SETMULTCOUNT 0x305
#define HDIO_SETFEATURE 0x306
#define HDIO_GETIDENTITY 0x307
#define HDIO_GETIDENTITY 0x307
#endif
#endif
...
...
include/linux/tty.h
View file @
f4a818e0
...
@@ -289,6 +289,7 @@ extern void stop_tty(struct tty_struct * tty);
...
@@ -289,6 +289,7 @@ extern void stop_tty(struct tty_struct * tty);
extern
void
start_tty
(
struct
tty_struct
*
tty
);
extern
void
start_tty
(
struct
tty_struct
*
tty
);
extern
int
tty_register_ldisc
(
int
disc
,
struct
tty_ldisc
*
new_ldisc
);
extern
int
tty_register_ldisc
(
int
disc
,
struct
tty_ldisc
*
new_ldisc
);
extern
int
tty_register_driver
(
struct
tty_driver
*
driver
);
extern
int
tty_register_driver
(
struct
tty_driver
*
driver
);
extern
int
tty_unregister_driver
(
struct
tty_driver
*
driver
);
extern
int
tty_read_raw_data
(
struct
tty_struct
*
tty
,
unsigned
char
*
bufp
,
extern
int
tty_read_raw_data
(
struct
tty_struct
*
tty
,
unsigned
char
*
bufp
,
int
buflen
);
int
buflen
);
...
...
kernel/ksyms.c
View file @
f4a818e0
...
@@ -25,6 +25,7 @@
...
@@ -25,6 +25,7 @@
#include <linux/module.h>
#include <linux/module.h>
#include <linux/termios.h>
#include <linux/termios.h>
#include <linux/tqueue.h>
#include <linux/tqueue.h>
#include <linux/tty.h>
#include <linux/serial.h>
#include <linux/serial.h>
#ifdef CONFIG_INET
#ifdef CONFIG_INET
#include <linux/netdevice.h>
#include <linux/netdevice.h>
...
@@ -91,6 +92,9 @@ struct symbol_table symbol_table = { 0, 0, 0, /* for stacked module support */
...
@@ -91,6 +92,9 @@ struct symbol_table symbol_table = { 0, 0, 0, /* for stacked module support */
X
(
unregister_chrdev
),
X
(
unregister_chrdev
),
X
(
register_blkdev
),
X
(
register_blkdev
),
X
(
unregister_blkdev
),
X
(
unregister_blkdev
),
X
(
tty_register_driver
),
X
(
tty_unregister_driver
),
X
(
tty_std_termios
),
/* block device driver support */
/* block device driver support */
X
(
block_read
),
X
(
block_read
),
...
...
net/inet/tcp.c
View file @
f4a818e0
...
@@ -83,6 +83,7 @@
...
@@ -83,6 +83,7 @@
* Matt Dillon : Yet more small nasties remove from the TCP code
* Matt Dillon : Yet more small nasties remove from the TCP code
* (Be very nice to this man if tcp finally works 100%) 8)
* (Be very nice to this man if tcp finally works 100%) 8)
* Alan Cox : BSD accept semantics.
* Alan Cox : BSD accept semantics.
* Peter De Schrijver : ENOTCONN check missing in tcp_sendto().
*
*
*
*
* To Fix:
* To Fix:
...
@@ -1221,15 +1222,17 @@ static int tcp_sendto(struct sock *sk, unsigned char *from,
...
@@ -1221,15 +1222,17 @@ static int tcp_sendto(struct sock *sk, unsigned char *from,
{
{
if
(
flags
&
~
(
MSG_OOB
|
MSG_DONTROUTE
))
if
(
flags
&
~
(
MSG_OOB
|
MSG_DONTROUTE
))
return
-
EINVAL
;
return
-
EINVAL
;
if
(
!
tcp_connected
(
sk
->
state
))
return
-
ENOTCONN
;
if
(
addr_len
<
sizeof
(
*
addr
))
if
(
addr_len
<
sizeof
(
*
addr
))
return
(
-
EINVAL
)
;
return
-
EINVAL
;
if
(
addr
->
sin_family
&&
addr
->
sin_family
!=
AF_INET
)
if
(
addr
->
sin_family
&&
addr
->
sin_family
!=
AF_INET
)
return
(
-
EINVAL
)
;
return
-
EINVAL
;
if
(
addr
->
sin_port
!=
sk
->
dummy_th
.
dest
)
if
(
addr
->
sin_port
!=
sk
->
dummy_th
.
dest
)
return
(
-
EISCONN
)
;
return
-
EISCONN
;
if
(
addr
->
sin_addr
.
s_addr
!=
sk
->
daddr
)
if
(
addr
->
sin_addr
.
s_addr
!=
sk
->
daddr
)
return
(
-
EISCONN
)
;
return
-
EISCONN
;
return
(
tcp_write
(
sk
,
from
,
len
,
nonblock
,
flags
)
);
return
tcp_write
(
sk
,
from
,
len
,
nonblock
,
flags
);
}
}
...
@@ -2238,7 +2241,7 @@ static void tcp_close(struct sock *sk, int timeout)
...
@@ -2238,7 +2241,7 @@ static void tcp_close(struct sock *sk, int timeout)
/* The +1 is not needed because the FIN takes up seq
/* The +1 is not needed because the FIN takes up seq
is not read!!! */
is not read!!! */
if
(
skb
->
len
>
0
&&
after
(
skb
->
h
.
th
->
seq
+
skb
->
len
,
sk
->
copied_seq
))
if
(
skb
->
len
>
0
&&
after
(
skb
->
h
.
th
->
seq
+
skb
->
len
,
sk
->
copied_seq
))
need_reset
=
1
;
need_reset
=
0
;
kfree_skb
(
skb
,
FREE_READ
);
kfree_skb
(
skb
,
FREE_READ
);
}
}
if
(
sk
->
debug
)
if
(
sk
->
debug
)
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment