diff options
483 files changed, 10771 insertions, 5040 deletions
@@ -966,6 +966,7 @@ N: Pekka Enberg | |||
966 | E: penberg@cs.helsinki.fi | 966 | E: penberg@cs.helsinki.fi |
967 | W: http://www.cs.helsinki.fi/u/penberg/ | 967 | W: http://www.cs.helsinki.fi/u/penberg/ |
968 | D: Various kernel hacks, fixes, and cleanups. | 968 | D: Various kernel hacks, fixes, and cleanups. |
969 | D: Slab allocators | ||
969 | S: Finland | 970 | S: Finland |
970 | 971 | ||
971 | N: David Engebretsen | 972 | N: David Engebretsen |
@@ -1939,8 +1940,8 @@ D: for Menuconfig's lxdialog. | |||
1939 | N: Christoph Lameter | 1940 | N: Christoph Lameter |
1940 | E: christoph@lameter.com | 1941 | E: christoph@lameter.com |
1941 | D: Digiboard PC/Xe and PC/Xi, Digiboard EPCA | 1942 | D: Digiboard PC/Xe and PC/Xi, Digiboard EPCA |
1942 | D: Early protocol filter for bridging code | 1943 | D: NUMA support, Slab allocators, Page migration |
1943 | D: Bug fixes | 1944 | D: Scalability, Time subsystem |
1944 | 1945 | ||
1945 | N: Paul Laufer | 1946 | N: Paul Laufer |
1946 | E: paul@laufernet.com | 1947 | E: paul@laufernet.com |
diff --git a/Documentation/block/barrier.txt b/Documentation/block/barrier.txt index 7d279f2f5bb2..2c2f24f634e4 100644 --- a/Documentation/block/barrier.txt +++ b/Documentation/block/barrier.txt | |||
@@ -79,9 +79,9 @@ and how to prepare flush requests. Note that the term 'ordered' is | |||
79 | used to indicate the whole sequence of performing barrier requests | 79 | used to indicate the whole sequence of performing barrier requests |
80 | including draining and flushing. | 80 | including draining and flushing. |
81 | 81 | ||
82 | typedef void (prepare_flush_fn)(request_queue_t *q, struct request *rq); | 82 | typedef void (prepare_flush_fn)(struct request_queue *q, struct request *rq); |
83 | 83 | ||
84 | int blk_queue_ordered(request_queue_t *q, unsigned ordered, | 84 | int blk_queue_ordered(struct request_queue *q, unsigned ordered, |
85 | prepare_flush_fn *prepare_flush_fn); | 85 | prepare_flush_fn *prepare_flush_fn); |
86 | 86 | ||
87 | @q : the queue in question | 87 | @q : the queue in question |
@@ -92,7 +92,7 @@ int blk_queue_ordered(request_queue_t *q, unsigned ordered, | |||
92 | For example, SCSI disk driver's prepare_flush_fn looks like the | 92 | For example, SCSI disk driver's prepare_flush_fn looks like the |
93 | following. | 93 | following. |
94 | 94 | ||
95 | static void sd_prepare_flush(request_queue_t *q, struct request *rq) | 95 | static void sd_prepare_flush(struct request_queue *q, struct request *rq) |
96 | { | 96 | { |
97 | memset(rq->cmd, 0, sizeof(rq->cmd)); | 97 | memset(rq->cmd, 0, sizeof(rq->cmd)); |
98 | rq->cmd_type = REQ_TYPE_BLOCK_PC; | 98 | rq->cmd_type = REQ_TYPE_BLOCK_PC; |
diff --git a/Documentation/block/biodoc.txt b/Documentation/block/biodoc.txt index 3adaace328a6..8af392fc6ef0 100644 --- a/Documentation/block/biodoc.txt +++ b/Documentation/block/biodoc.txt | |||
@@ -740,12 +740,12 @@ Block now offers some simple generic functionality to help support command | |||
740 | queueing (typically known as tagged command queueing), ie manage more than | 740 | queueing (typically known as tagged command queueing), ie manage more than |
741 | one outstanding command on a queue at any given time. | 741 | one outstanding command on a queue at any given time. |
742 | 742 | ||
743 | blk_queue_init_tags(request_queue_t *q, int depth) | 743 | blk_queue_init_tags(struct request_queue *q, int depth) |
744 | 744 | ||
745 | Initialize internal command tagging structures for a maximum | 745 | Initialize internal command tagging structures for a maximum |
746 | depth of 'depth'. | 746 | depth of 'depth'. |
747 | 747 | ||
748 | blk_queue_free_tags((request_queue_t *q) | 748 | blk_queue_free_tags((struct request_queue *q) |
749 | 749 | ||
750 | Teardown tag info associated with the queue. This will be done | 750 | Teardown tag info associated with the queue. This will be done |
751 | automatically by block if blk_queue_cleanup() is called on a queue | 751 | automatically by block if blk_queue_cleanup() is called on a queue |
@@ -754,7 +754,7 @@ one outstanding command on a queue at any given time. | |||
754 | The above are initialization and exit management, the main helpers during | 754 | The above are initialization and exit management, the main helpers during |
755 | normal operations are: | 755 | normal operations are: |
756 | 756 | ||
757 | blk_queue_start_tag(request_queue_t *q, struct request *rq) | 757 | blk_queue_start_tag(struct request_queue *q, struct request *rq) |
758 | 758 | ||
759 | Start tagged operation for this request. A free tag number between | 759 | Start tagged operation for this request. A free tag number between |
760 | 0 and 'depth' is assigned to the request (rq->tag holds this number), | 760 | 0 and 'depth' is assigned to the request (rq->tag holds this number), |
@@ -762,7 +762,7 @@ normal operations are: | |||
762 | for this queue is already achieved (or if the tag wasn't started for | 762 | for this queue is already achieved (or if the tag wasn't started for |
763 | some other reason), 1 is returned. Otherwise 0 is returned. | 763 | some other reason), 1 is returned. Otherwise 0 is returned. |
764 | 764 | ||
765 | blk_queue_end_tag(request_queue_t *q, struct request *rq) | 765 | blk_queue_end_tag(struct request_queue *q, struct request *rq) |
766 | 766 | ||
767 | End tagged operation on this request. 'rq' is removed from the internal | 767 | End tagged operation on this request. 'rq' is removed from the internal |
768 | book keeping structures. | 768 | book keeping structures. |
@@ -781,7 +781,7 @@ queue. For instance, on IDE any tagged request error needs to clear both | |||
781 | the hardware and software block queue and enable the driver to sanely restart | 781 | the hardware and software block queue and enable the driver to sanely restart |
782 | all the outstanding requests. There's a third helper to do that: | 782 | all the outstanding requests. There's a third helper to do that: |
783 | 783 | ||
784 | blk_queue_invalidate_tags(request_queue_t *q) | 784 | blk_queue_invalidate_tags(struct request_queue *q) |
785 | 785 | ||
786 | Clear the internal block tag queue and re-add all the pending requests | 786 | Clear the internal block tag queue and re-add all the pending requests |
787 | to the request queue. The driver will receive them again on the | 787 | to the request queue. The driver will receive them again on the |
diff --git a/Documentation/block/request.txt b/Documentation/block/request.txt index 75924e2a6975..fff58acb40a3 100644 --- a/Documentation/block/request.txt +++ b/Documentation/block/request.txt | |||
@@ -83,6 +83,6 @@ struct bio *bio DBI First bio in request | |||
83 | 83 | ||
84 | struct bio *biotail DBI Last bio in request | 84 | struct bio *biotail DBI Last bio in request |
85 | 85 | ||
86 | request_queue_t *q DB Request queue this request belongs to | 86 | struct request_queue *q DB Request queue this request belongs to |
87 | 87 | ||
88 | struct request_list *rl B Request list this request came from | 88 | struct request_list *rl B Request list this request came from |
diff --git a/Documentation/iostats.txt b/Documentation/iostats.txt index 09a1bafe2528..b963c3b4afa5 100644 --- a/Documentation/iostats.txt +++ b/Documentation/iostats.txt | |||
@@ -79,7 +79,7 @@ Field 8 -- # of milliseconds spent writing | |||
79 | measured from __make_request() to end_that_request_last()). | 79 | measured from __make_request() to end_that_request_last()). |
80 | Field 9 -- # of I/Os currently in progress | 80 | Field 9 -- # of I/Os currently in progress |
81 | The only field that should go to zero. Incremented as requests are | 81 | The only field that should go to zero. Incremented as requests are |
82 | given to appropriate request_queue_t and decremented as they finish. | 82 | given to appropriate struct request_queue and decremented as they finish. |
83 | Field 10 -- # of milliseconds spent doing I/Os | 83 | Field 10 -- # of milliseconds spent doing I/Os |
84 | This field is increases so long as field 9 is nonzero. | 84 | This field is increases so long as field 9 is nonzero. |
85 | Field 11 -- weighted # of milliseconds spent doing I/Os | 85 | Field 11 -- weighted # of milliseconds spent doing I/Os |
diff --git a/Documentation/lguest/extract b/Documentation/lguest/extract new file mode 100644 index 000000000000..7730bb6e4b94 --- /dev/null +++ b/Documentation/lguest/extract | |||
@@ -0,0 +1,58 @@ | |||
1 | #! /bin/sh | ||
2 | |||
3 | set -e | ||
4 | |||
5 | PREFIX=$1 | ||
6 | shift | ||
7 | |||
8 | trap 'rm -r $TMPDIR' 0 | ||
9 | TMPDIR=`mktemp -d` | ||
10 | |||
11 | exec 3>/dev/null | ||
12 | for f; do | ||
13 | while IFS=" | ||
14 | " read -r LINE; do | ||
15 | case "$LINE" in | ||
16 | *$PREFIX:[0-9]*:\**) | ||
17 | NUM=`echo "$LINE" | sed "s/.*$PREFIX:\([0-9]*\).*/\1/"` | ||
18 | if [ -f $TMPDIR/$NUM ]; then | ||
19 | echo "$TMPDIR/$NUM already exits prior to $f" | ||
20 | exit 1 | ||
21 | fi | ||
22 | exec 3>>$TMPDIR/$NUM | ||
23 | echo $f | sed 's,\.\./,,g' > $TMPDIR/.$NUM | ||
24 | /bin/echo "$LINE" | sed -e "s/$PREFIX:[0-9]*//" -e "s/:\*/*/" >&3 | ||
25 | ;; | ||
26 | *$PREFIX:[0-9]*) | ||
27 | NUM=`echo "$LINE" | sed "s/.*$PREFIX:\([0-9]*\).*/\1/"` | ||
28 | if [ -f $TMPDIR/$NUM ]; then | ||
29 | echo "$TMPDIR/$NUM already exits prior to $f" | ||
30 | exit 1 | ||
31 | fi | ||
32 | exec 3>>$TMPDIR/$NUM | ||
33 | echo $f | sed 's,\.\./,,g' > $TMPDIR/.$NUM | ||
34 | /bin/echo "$LINE" | sed "s/$PREFIX:[0-9]*//" >&3 | ||
35 | ;; | ||
36 | *:\**) | ||
37 | /bin/echo "$LINE" | sed -e "s/:\*/*/" -e "s,/\*\*/,," >&3 | ||
38 | echo >&3 | ||
39 | exec 3>/dev/null | ||
40 | ;; | ||
41 | *) | ||
42 | /bin/echo "$LINE" >&3 | ||
43 | ;; | ||
44 | esac | ||
45 | done < $f | ||
46 | echo >&3 | ||
47 | exec 3>/dev/null | ||
48 | done | ||
49 | |||
50 | LASTFILE="" | ||
51 | for f in $TMPDIR/*; do | ||
52 | if [ "$LASTFILE" != $(cat $TMPDIR/.$(basename $f) ) ]; then | ||
53 | LASTFILE=$(cat $TMPDIR/.$(basename $f) ) | ||
54 | echo "[ $LASTFILE ]" | ||
55 | fi | ||
56 | cat $f | ||
57 | done | ||
58 | |||
diff --git a/Documentation/lguest/lguest.c b/Documentation/lguest/lguest.c index 62a8133393e1..f7918401a007 100644 --- a/Documentation/lguest/lguest.c +++ b/Documentation/lguest/lguest.c | |||
@@ -1,5 +1,10 @@ | |||
1 | /* Simple program to layout "physical" memory for new lguest guest. | 1 | /*P:100 This is the Launcher code, a simple program which lays out the |
2 | * Linked high to avoid likely physical memory. */ | 2 | * "physical" memory for the new Guest by mapping the kernel image and the |
3 | * virtual devices, then reads repeatedly from /dev/lguest to run the Guest. | ||
4 | * | ||
5 | * The only trick: the Makefile links it at a high address so it will be clear | ||
6 | * of the guest memory region. It means that each Guest cannot have more than | ||
7 | * about 2.5G of memory on a normally configured Host. :*/ | ||
3 | #define _LARGEFILE64_SOURCE | 8 | #define _LARGEFILE64_SOURCE |
4 | #define _GNU_SOURCE | 9 | #define _GNU_SOURCE |
5 | #include <stdio.h> | 10 | #include <stdio.h> |
@@ -29,12 +34,20 @@ | |||
29 | #include <termios.h> | 34 | #include <termios.h> |
30 | #include <getopt.h> | 35 | #include <getopt.h> |
31 | #include <zlib.h> | 36 | #include <zlib.h> |
37 | /*L:110 We can ignore the 28 include files we need for this program, but I do | ||
38 | * want to draw attention to the use of kernel-style types. | ||
39 | * | ||
40 | * As Linus said, "C is a Spartan language, and so should your naming be." I | ||
41 | * like these abbreviations and the header we need uses them, so we define them | ||
42 | * here. | ||
43 | */ | ||
32 | typedef unsigned long long u64; | 44 | typedef unsigned long long u64; |
33 | typedef uint32_t u32; | 45 | typedef uint32_t u32; |
34 | typedef uint16_t u16; | 46 | typedef uint16_t u16; |
35 | typedef uint8_t u8; | 47 | typedef uint8_t u8; |
36 | #include "../../include/linux/lguest_launcher.h" | 48 | #include "../../include/linux/lguest_launcher.h" |
37 | #include "../../include/asm-i386/e820.h" | 49 | #include "../../include/asm-i386/e820.h" |
50 | /*:*/ | ||
38 | 51 | ||
39 | #define PAGE_PRESENT 0x7 /* Present, RW, Execute */ | 52 | #define PAGE_PRESENT 0x7 /* Present, RW, Execute */ |
40 | #define NET_PEERNUM 1 | 53 | #define NET_PEERNUM 1 |
@@ -43,33 +56,52 @@ typedef uint8_t u8; | |||
43 | #define SIOCBRADDIF 0x89a2 /* add interface to bridge */ | 56 | #define SIOCBRADDIF 0x89a2 /* add interface to bridge */ |
44 | #endif | 57 | #endif |
45 | 58 | ||
59 | /*L:120 verbose is both a global flag and a macro. The C preprocessor allows | ||
60 | * this, and although I wouldn't recommend it, it works quite nicely here. */ | ||
46 | static bool verbose; | 61 | static bool verbose; |
47 | #define verbose(args...) \ | 62 | #define verbose(args...) \ |
48 | do { if (verbose) printf(args); } while(0) | 63 | do { if (verbose) printf(args); } while(0) |
64 | /*:*/ | ||
65 | |||
66 | /* The pipe to send commands to the waker process */ | ||
49 | static int waker_fd; | 67 | static int waker_fd; |
68 | /* The top of guest physical memory. */ | ||
50 | static u32 top; | 69 | static u32 top; |
51 | 70 | ||
71 | /* This is our list of devices. */ | ||
52 | struct device_list | 72 | struct device_list |
53 | { | 73 | { |
74 | /* Summary information about the devices in our list: ready to pass to | ||
75 | * select() to ask which need servicing.*/ | ||
54 | fd_set infds; | 76 | fd_set infds; |
55 | int max_infd; | 77 | int max_infd; |
56 | 78 | ||
79 | /* The descriptor page for the devices. */ | ||
57 | struct lguest_device_desc *descs; | 80 | struct lguest_device_desc *descs; |
81 | |||
82 | /* A single linked list of devices. */ | ||
58 | struct device *dev; | 83 | struct device *dev; |
84 | /* ... And an end pointer so we can easily append new devices */ | ||
59 | struct device **lastdev; | 85 | struct device **lastdev; |
60 | }; | 86 | }; |
61 | 87 | ||
88 | /* The device structure describes a single device. */ | ||
62 | struct device | 89 | struct device |
63 | { | 90 | { |
91 | /* The linked-list pointer. */ | ||
64 | struct device *next; | 92 | struct device *next; |
93 | /* The descriptor for this device, as mapped into the Guest. */ | ||
65 | struct lguest_device_desc *desc; | 94 | struct lguest_device_desc *desc; |
95 | /* The memory page(s) of this device, if any. Also mapped in Guest. */ | ||
66 | void *mem; | 96 | void *mem; |
67 | 97 | ||
68 | /* Watch this fd if handle_input non-NULL. */ | 98 | /* If handle_input is set, it wants to be called when this file |
99 | * descriptor is ready. */ | ||
69 | int fd; | 100 | int fd; |
70 | bool (*handle_input)(int fd, struct device *me); | 101 | bool (*handle_input)(int fd, struct device *me); |
71 | 102 | ||
72 | /* Watch DMA to this key if handle_input non-NULL. */ | 103 | /* If handle_output is set, it wants to be called when the Guest sends |
104 | * DMA to this key. */ | ||
73 | unsigned long watch_key; | 105 | unsigned long watch_key; |
74 | u32 (*handle_output)(int fd, const struct iovec *iov, | 106 | u32 (*handle_output)(int fd, const struct iovec *iov, |
75 | unsigned int num, struct device *me); | 107 | unsigned int num, struct device *me); |
@@ -78,6 +110,11 @@ struct device | |||
78 | void *priv; | 110 | void *priv; |
79 | }; | 111 | }; |
80 | 112 | ||
113 | /*L:130 | ||
114 | * Loading the Kernel. | ||
115 | * | ||
116 | * We start with couple of simple helper routines. open_or_die() avoids | ||
117 | * error-checking code cluttering the callers: */ | ||
81 | static int open_or_die(const char *name, int flags) | 118 | static int open_or_die(const char *name, int flags) |
82 | { | 119 | { |
83 | int fd = open(name, flags); | 120 | int fd = open(name, flags); |
@@ -86,26 +123,38 @@ static int open_or_die(const char *name, int flags) | |||
86 | return fd; | 123 | return fd; |
87 | } | 124 | } |
88 | 125 | ||
126 | /* map_zeroed_pages() takes a (page-aligned) address and a number of pages. */ | ||
89 | static void *map_zeroed_pages(unsigned long addr, unsigned int num) | 127 | static void *map_zeroed_pages(unsigned long addr, unsigned int num) |
90 | { | 128 | { |
129 | /* We cache the /dev/zero file-descriptor so we only open it once. */ | ||
91 | static int fd = -1; | 130 | static int fd = -1; |
92 | 131 | ||
93 | if (fd == -1) | 132 | if (fd == -1) |
94 | fd = open_or_die("/dev/zero", O_RDONLY); | 133 | fd = open_or_die("/dev/zero", O_RDONLY); |
95 | 134 | ||
135 | /* We use a private mapping (ie. if we write to the page, it will be | ||
136 | * copied), and obviously we insist that it be mapped where we ask. */ | ||
96 | if (mmap((void *)addr, getpagesize() * num, | 137 | if (mmap((void *)addr, getpagesize() * num, |
97 | PROT_READ|PROT_WRITE|PROT_EXEC, MAP_FIXED|MAP_PRIVATE, fd, 0) | 138 | PROT_READ|PROT_WRITE|PROT_EXEC, MAP_FIXED|MAP_PRIVATE, fd, 0) |
98 | != (void *)addr) | 139 | != (void *)addr) |
99 | err(1, "Mmaping %u pages of /dev/zero @%p", num, (void *)addr); | 140 | err(1, "Mmaping %u pages of /dev/zero @%p", num, (void *)addr); |
141 | |||
142 | /* Returning the address is just a courtesy: can simplify callers. */ | ||
100 | return (void *)addr; | 143 | return (void *)addr; |
101 | } | 144 | } |
102 | 145 | ||
103 | /* Find magic string marking entry point, return entry point. */ | 146 | /* To find out where to start we look for the magic Guest string, which marks |
147 | * the code we see in lguest_asm.S. This is a hack which we are currently | ||
148 | * plotting to replace with the normal Linux entry point. */ | ||
104 | static unsigned long entry_point(void *start, void *end, | 149 | static unsigned long entry_point(void *start, void *end, |
105 | unsigned long page_offset) | 150 | unsigned long page_offset) |
106 | { | 151 | { |
107 | void *p; | 152 | void *p; |
108 | 153 | ||
154 | /* The scan gives us the physical starting address. We want the | ||
155 | * virtual address in this case, and fortunately, we already figured | ||
156 | * out the physical-virtual difference and passed it here in | ||
157 | * "page_offset". */ | ||
109 | for (p = start; p < end; p++) | 158 | for (p = start; p < end; p++) |
110 | if (memcmp(p, "GenuineLguest", strlen("GenuineLguest")) == 0) | 159 | if (memcmp(p, "GenuineLguest", strlen("GenuineLguest")) == 0) |
111 | return (long)p + strlen("GenuineLguest") + page_offset; | 160 | return (long)p + strlen("GenuineLguest") + page_offset; |
@@ -113,7 +162,17 @@ static unsigned long entry_point(void *start, void *end, | |||
113 | err(1, "Is this image a genuine lguest?"); | 162 | err(1, "Is this image a genuine lguest?"); |
114 | } | 163 | } |
115 | 164 | ||
116 | /* Returns the entry point */ | 165 | /* This routine takes an open vmlinux image, which is in ELF, and maps it into |
166 | * the Guest memory. ELF = Embedded Linking Format, which is the format used | ||
167 | * by all modern binaries on Linux including the kernel. | ||
168 | * | ||
169 | * The ELF headers give *two* addresses: a physical address, and a virtual | ||
170 | * address. The Guest kernel expects to be placed in memory at the physical | ||
171 | * address, and the page tables set up so it will correspond to that virtual | ||
172 | * address. We return the difference between the virtual and physical | ||
173 | * addresses in the "page_offset" pointer. | ||
174 | * | ||
175 | * We return the starting address. */ | ||
117 | static unsigned long map_elf(int elf_fd, const Elf32_Ehdr *ehdr, | 176 | static unsigned long map_elf(int elf_fd, const Elf32_Ehdr *ehdr, |
118 | unsigned long *page_offset) | 177 | unsigned long *page_offset) |
119 | { | 178 | { |
@@ -122,40 +181,61 @@ static unsigned long map_elf(int elf_fd, const Elf32_Ehdr *ehdr, | |||
122 | unsigned int i; | 181 | unsigned int i; |
123 | unsigned long start = -1UL, end = 0; | 182 | unsigned long start = -1UL, end = 0; |
124 | 183 | ||
125 | /* Sanity checks. */ | 184 | /* Sanity checks on the main ELF header: an x86 executable with a |
185 | * reasonable number of correctly-sized program headers. */ | ||
126 | if (ehdr->e_type != ET_EXEC | 186 | if (ehdr->e_type != ET_EXEC |
127 | || ehdr->e_machine != EM_386 | 187 | || ehdr->e_machine != EM_386 |
128 | || ehdr->e_phentsize != sizeof(Elf32_Phdr) | 188 | || ehdr->e_phentsize != sizeof(Elf32_Phdr) |
129 | || ehdr->e_phnum < 1 || ehdr->e_phnum > 65536U/sizeof(Elf32_Phdr)) | 189 | || ehdr->e_phnum < 1 || ehdr->e_phnum > 65536U/sizeof(Elf32_Phdr)) |
130 | errx(1, "Malformed elf header"); | 190 | errx(1, "Malformed elf header"); |
131 | 191 | ||
192 | /* An ELF executable contains an ELF header and a number of "program" | ||
193 | * headers which indicate which parts ("segments") of the program to | ||
194 | * load where. */ | ||
195 | |||
196 | /* We read in all the program headers at once: */ | ||
132 | if (lseek(elf_fd, ehdr->e_phoff, SEEK_SET) < 0) | 197 | if (lseek(elf_fd, ehdr->e_phoff, SEEK_SET) < 0) |
133 | err(1, "Seeking to program headers"); | 198 | err(1, "Seeking to program headers"); |
134 | if (read(elf_fd, phdr, sizeof(phdr)) != sizeof(phdr)) | 199 | if (read(elf_fd, phdr, sizeof(phdr)) != sizeof(phdr)) |
135 | err(1, "Reading program headers"); | 200 | err(1, "Reading program headers"); |
136 | 201 | ||
202 | /* We don't know page_offset yet. */ | ||
137 | *page_offset = 0; | 203 | *page_offset = 0; |
138 | /* We map the loadable segments at virtual addresses corresponding | 204 | |
139 | * to their physical addresses (our virtual == guest physical). */ | 205 | /* Try all the headers: there are usually only three. A read-only one, |
206 | * a read-write one, and a "note" section which isn't loadable. */ | ||
140 | for (i = 0; i < ehdr->e_phnum; i++) { | 207 | for (i = 0; i < ehdr->e_phnum; i++) { |
208 | /* If this isn't a loadable segment, we ignore it */ | ||
141 | if (phdr[i].p_type != PT_LOAD) | 209 | if (phdr[i].p_type != PT_LOAD) |
142 | continue; | 210 | continue; |
143 | 211 | ||
144 | verbose("Section %i: size %i addr %p\n", | 212 | verbose("Section %i: size %i addr %p\n", |
145 | i, phdr[i].p_memsz, (void *)phdr[i].p_paddr); | 213 | i, phdr[i].p_memsz, (void *)phdr[i].p_paddr); |
146 | 214 | ||
147 | /* We expect linear address space. */ | 215 | /* We expect a simple linear address space: every segment must |
216 | * have the same difference between virtual (p_vaddr) and | ||
217 | * physical (p_paddr) address. */ | ||
148 | if (!*page_offset) | 218 | if (!*page_offset) |
149 | *page_offset = phdr[i].p_vaddr - phdr[i].p_paddr; | 219 | *page_offset = phdr[i].p_vaddr - phdr[i].p_paddr; |
150 | else if (*page_offset != phdr[i].p_vaddr - phdr[i].p_paddr) | 220 | else if (*page_offset != phdr[i].p_vaddr - phdr[i].p_paddr) |
151 | errx(1, "Page offset of section %i different", i); | 221 | errx(1, "Page offset of section %i different", i); |
152 | 222 | ||
223 | /* We track the first and last address we mapped, so we can | ||
224 | * tell entry_point() where to scan. */ | ||
153 | if (phdr[i].p_paddr < start) | 225 | if (phdr[i].p_paddr < start) |
154 | start = phdr[i].p_paddr; | 226 | start = phdr[i].p_paddr; |
155 | if (phdr[i].p_paddr + phdr[i].p_filesz > end) | 227 | if (phdr[i].p_paddr + phdr[i].p_filesz > end) |
156 | end = phdr[i].p_paddr + phdr[i].p_filesz; | 228 | end = phdr[i].p_paddr + phdr[i].p_filesz; |
157 | 229 | ||
158 | /* We map everything private, writable. */ | 230 | /* We map this section of the file at its physical address. We |
231 | * map it read & write even if the header says this segment is | ||
232 | * read-only. The kernel really wants to be writable: it | ||
233 | * patches its own instructions which would normally be | ||
234 | * read-only. | ||
235 | * | ||
236 | * MAP_PRIVATE means that the page won't be copied until a | ||
237 | * write is done to it. This allows us to share much of the | ||
238 | * kernel memory between Guests. */ | ||
159 | addr = mmap((void *)phdr[i].p_paddr, | 239 | addr = mmap((void *)phdr[i].p_paddr, |
160 | phdr[i].p_filesz, | 240 | phdr[i].p_filesz, |
161 | PROT_READ|PROT_WRITE|PROT_EXEC, | 241 | PROT_READ|PROT_WRITE|PROT_EXEC, |
@@ -169,7 +249,31 @@ static unsigned long map_elf(int elf_fd, const Elf32_Ehdr *ehdr, | |||
169 | return entry_point((void *)start, (void *)end, *page_offset); | 249 | return entry_point((void *)start, (void *)end, *page_offset); |
170 | } | 250 | } |
171 | 251 | ||
172 | /* This is amazingly reliable. */ | 252 | /*L:170 Prepare to be SHOCKED and AMAZED. And possibly a trifle nauseated. |
253 | * | ||
254 | * We know that CONFIG_PAGE_OFFSET sets what virtual address the kernel expects | ||
255 | * to be. We don't know what that option was, but we can figure it out | ||
256 | * approximately by looking at the addresses in the code. I chose the common | ||
257 | * case of reading a memory location into the %eax register: | ||
258 | * | ||
259 | * movl <some-address>, %eax | ||
260 | * | ||
261 | * This gets encoded as five bytes: "0xA1 <4-byte-address>". For example, | ||
262 | * "0xA1 0x18 0x60 0x47 0xC0" reads the address 0xC0476018 into %eax. | ||
263 | * | ||
264 | * In this example can guess that the kernel was compiled with | ||
265 | * CONFIG_PAGE_OFFSET set to 0xC0000000 (it's always a round number). If the | ||
266 | * kernel were larger than 16MB, we might see 0xC1 addresses show up, but our | ||
267 | * kernel isn't that bloated yet. | ||
268 | * | ||
269 | * Unfortunately, x86 has variable-length instructions, so finding this | ||
270 | * particular instruction properly involves writing a disassembler. Instead, | ||
271 | * we rely on statistics. We look for "0xA1" and tally the different bytes | ||
272 | * which occur 4 bytes later (the "0xC0" in our example above). When one of | ||
273 | * those bytes appears three times, we can be reasonably confident that it | ||
274 | * forms the start of CONFIG_PAGE_OFFSET. | ||
275 | * | ||
276 | * This is amazingly reliable. */ | ||
173 | static unsigned long intuit_page_offset(unsigned char *img, unsigned long len) | 277 | static unsigned long intuit_page_offset(unsigned char *img, unsigned long len) |
174 | { | 278 | { |
175 | unsigned int i, possibilities[256] = { 0 }; | 279 | unsigned int i, possibilities[256] = { 0 }; |
@@ -182,30 +286,52 @@ static unsigned long intuit_page_offset(unsigned char *img, unsigned long len) | |||
182 | errx(1, "could not determine page offset"); | 286 | errx(1, "could not determine page offset"); |
183 | } | 287 | } |
184 | 288 | ||
289 | /*L:160 Unfortunately the entire ELF image isn't compressed: the segments | ||
290 | * which need loading are extracted and compressed raw. This denies us the | ||
291 | * information we need to make a fully-general loader. */ | ||
185 | static unsigned long unpack_bzimage(int fd, unsigned long *page_offset) | 292 | static unsigned long unpack_bzimage(int fd, unsigned long *page_offset) |
186 | { | 293 | { |
187 | gzFile f; | 294 | gzFile f; |
188 | int ret, len = 0; | 295 | int ret, len = 0; |
296 | /* A bzImage always gets loaded at physical address 1M. This is | ||
297 | * actually configurable as CONFIG_PHYSICAL_START, but as the comment | ||
298 | * there says, "Don't change this unless you know what you are doing". | ||
299 | * Indeed. */ | ||
189 | void *img = (void *)0x100000; | 300 | void *img = (void *)0x100000; |
190 | 301 | ||
302 | /* gzdopen takes our file descriptor (carefully placed at the start of | ||
303 | * the GZIP header we found) and returns a gzFile. */ | ||
191 | f = gzdopen(fd, "rb"); | 304 | f = gzdopen(fd, "rb"); |
305 | /* We read it into memory in 64k chunks until we hit the end. */ | ||
192 | while ((ret = gzread(f, img + len, 65536)) > 0) | 306 | while ((ret = gzread(f, img + len, 65536)) > 0) |
193 | len += ret; | 307 | len += ret; |
194 | if (ret < 0) | 308 | if (ret < 0) |
195 | err(1, "reading image from bzImage"); | 309 | err(1, "reading image from bzImage"); |
196 | 310 | ||
197 | verbose("Unpacked size %i addr %p\n", len, img); | 311 | verbose("Unpacked size %i addr %p\n", len, img); |
312 | |||
313 | /* Without the ELF header, we can't tell virtual-physical gap. This is | ||
314 | * CONFIG_PAGE_OFFSET, and people do actually change it. Fortunately, | ||
315 | * I have a clever way of figuring it out from the code itself. */ | ||
198 | *page_offset = intuit_page_offset(img, len); | 316 | *page_offset = intuit_page_offset(img, len); |
199 | 317 | ||
200 | return entry_point(img, img + len, *page_offset); | 318 | return entry_point(img, img + len, *page_offset); |
201 | } | 319 | } |
202 | 320 | ||
321 | /*L:150 A bzImage, unlike an ELF file, is not meant to be loaded. You're | ||
322 | * supposed to jump into it and it will unpack itself. We can't do that | ||
323 | * because the Guest can't run the unpacking code, and adding features to | ||
324 | * lguest kills puppies, so we don't want to. | ||
325 | * | ||
326 | * The bzImage is formed by putting the decompressing code in front of the | ||
327 | * compressed kernel code. So we can simple scan through it looking for the | ||
328 | * first "gzip" header, and start decompressing from there. */ | ||
203 | static unsigned long load_bzimage(int fd, unsigned long *page_offset) | 329 | static unsigned long load_bzimage(int fd, unsigned long *page_offset) |
204 | { | 330 | { |
205 | unsigned char c; | 331 | unsigned char c; |
206 | int state = 0; | 332 | int state = 0; |
207 | 333 | ||
208 | /* Ugly brute force search for gzip header. */ | 334 | /* GZIP header is 0x1F 0x8B <method> <flags>... <compressed-by>. */ |
209 | while (read(fd, &c, 1) == 1) { | 335 | while (read(fd, &c, 1) == 1) { |
210 | switch (state) { | 336 | switch (state) { |
211 | case 0: | 337 | case 0: |
@@ -222,8 +348,10 @@ static unsigned long load_bzimage(int fd, unsigned long *page_offset) | |||
222 | state++; | 348 | state++; |
223 | break; | 349 | break; |
224 | case 9: | 350 | case 9: |
351 | /* Seek back to the start of the gzip header. */ | ||
225 | lseek(fd, -10, SEEK_CUR); | 352 | lseek(fd, -10, SEEK_CUR); |
226 | if (c != 0x03) /* Compressed under UNIX. */ | 353 | /* One final check: "compressed under UNIX". */ |
354 | if (c != 0x03) | ||
227 | state = -1; | 355 | state = -1; |
228 | else | 356 | else |
229 | return unpack_bzimage(fd, page_offset); | 357 | return unpack_bzimage(fd, page_offset); |
@@ -232,25 +360,43 @@ static unsigned long load_bzimage(int fd, unsigned long *page_offset) | |||
232 | errx(1, "Could not find kernel in bzImage"); | 360 | errx(1, "Could not find kernel in bzImage"); |
233 | } | 361 | } |
234 | 362 | ||
363 | /*L:140 Loading the kernel is easy when it's a "vmlinux", but most kernels | ||
364 | * come wrapped up in the self-decompressing "bzImage" format. With some funky | ||
365 | * coding, we can load those, too. */ | ||
235 | static unsigned long load_kernel(int fd, unsigned long *page_offset) | 366 | static unsigned long load_kernel(int fd, unsigned long *page_offset) |
236 | { | 367 | { |
237 | Elf32_Ehdr hdr; | 368 | Elf32_Ehdr hdr; |
238 | 369 | ||
370 | /* Read in the first few bytes. */ | ||
239 | if (read(fd, &hdr, sizeof(hdr)) != sizeof(hdr)) | 371 | if (read(fd, &hdr, sizeof(hdr)) != sizeof(hdr)) |
240 | err(1, "Reading kernel"); | 372 | err(1, "Reading kernel"); |
241 | 373 | ||
374 | /* If it's an ELF file, it starts with "\177ELF" */ | ||
242 | if (memcmp(hdr.e_ident, ELFMAG, SELFMAG) == 0) | 375 | if (memcmp(hdr.e_ident, ELFMAG, SELFMAG) == 0) |
243 | return map_elf(fd, &hdr, page_offset); | 376 | return map_elf(fd, &hdr, page_offset); |
244 | 377 | ||
378 | /* Otherwise we assume it's a bzImage, and try to unpack it */ | ||
245 | return load_bzimage(fd, page_offset); | 379 | return load_bzimage(fd, page_offset); |
246 | } | 380 | } |
247 | 381 | ||
382 | /* This is a trivial little helper to align pages. Andi Kleen hated it because | ||
383 | * it calls getpagesize() twice: "it's dumb code." | ||
384 | * | ||
385 | * Kernel guys get really het up about optimization, even when it's not | ||
386 | * necessary. I leave this code as a reaction against that. */ | ||
248 | static inline unsigned long page_align(unsigned long addr) | 387 | static inline unsigned long page_align(unsigned long addr) |
249 | { | 388 | { |
389 | /* Add upwards and truncate downwards. */ | ||
250 | return ((addr + getpagesize()-1) & ~(getpagesize()-1)); | 390 | return ((addr + getpagesize()-1) & ~(getpagesize()-1)); |
251 | } | 391 | } |
252 | 392 | ||
253 | /* initrd gets loaded at top of memory: return length. */ | 393 | /*L:180 An "initial ram disk" is a disk image loaded into memory along with |
394 | * the kernel which the kernel can use to boot from without needing any | ||
395 | * drivers. Most distributions now use this as standard: the initrd contains | ||
396 | * the code to load the appropriate driver modules for the current machine. | ||
397 | * | ||
398 | * Importantly, James Morris works for RedHat, and Fedora uses initrds for its | ||
399 | * kernels. He sent me this (and tells me when I break it). */ | ||
254 | static unsigned long load_initrd(const char *name, unsigned long mem) | 400 | static unsigned long load_initrd(const char *name, unsigned long mem) |
255 | { | 401 | { |
256 | int ifd; | 402 | int ifd; |
@@ -259,21 +405,35 @@ static unsigned long load_initrd(const char *name, unsigned long mem) | |||
259 | void *iaddr; | 405 | void *iaddr; |
260 | 406 | ||
261 | ifd = open_or_die(name, O_RDONLY); | 407 | ifd = open_or_die(name, O_RDONLY); |
408 | /* fstat() is needed to get the file size. */ | ||
262 | if (fstat(ifd, &st) < 0) | 409 | if (fstat(ifd, &st) < 0) |
263 | err(1, "fstat() on initrd '%s'", name); | 410 | err(1, "fstat() on initrd '%s'", name); |
264 | 411 | ||
412 | /* The length needs to be rounded up to a page size: mmap needs the | ||
413 | * address to be page aligned. */ | ||
265 | len = page_align(st.st_size); | 414 | len = page_align(st.st_size); |
415 | /* We map the initrd at the top of memory. */ | ||
266 | iaddr = mmap((void *)mem - len, st.st_size, | 416 | iaddr = mmap((void *)mem - len, st.st_size, |
267 | PROT_READ|PROT_EXEC|PROT_WRITE, | 417 | PROT_READ|PROT_EXEC|PROT_WRITE, |
268 | MAP_FIXED|MAP_PRIVATE, ifd, 0); | 418 | MAP_FIXED|MAP_PRIVATE, ifd, 0); |
269 | if (iaddr != (void *)mem - len) | 419 | if (iaddr != (void *)mem - len) |
270 | err(1, "Mmaping initrd '%s' returned %p not %p", | 420 | err(1, "Mmaping initrd '%s' returned %p not %p", |
271 | name, iaddr, (void *)mem - len); | 421 | name, iaddr, (void *)mem - len); |
422 | /* Once a file is mapped, you can close the file descriptor. It's a | ||
423 | * little odd, but quite useful. */ | ||
272 | close(ifd); | 424 | close(ifd); |
273 | verbose("mapped initrd %s size=%lu @ %p\n", name, st.st_size, iaddr); | 425 | verbose("mapped initrd %s size=%lu @ %p\n", name, st.st_size, iaddr); |
426 | |||
427 | /* We return the initrd size. */ | ||
274 | return len; | 428 | return len; |
275 | } | 429 | } |
276 | 430 | ||
431 | /* Once we know how much memory we have, and the address the Guest kernel | ||
432 | * expects, we can construct simple linear page tables which will get the Guest | ||
433 | * far enough into the boot to create its own. | ||
434 | * | ||
435 | * We lay them out of the way, just below the initrd (which is why we need to | ||
436 | * know its size). */ | ||
277 | static unsigned long setup_pagetables(unsigned long mem, | 437 | static unsigned long setup_pagetables(unsigned long mem, |
278 | unsigned long initrd_size, | 438 | unsigned long initrd_size, |
279 | unsigned long page_offset) | 439 | unsigned long page_offset) |
@@ -282,23 +442,32 @@ static unsigned long setup_pagetables(unsigned long mem, | |||
282 | unsigned int mapped_pages, i, linear_pages; | 442 | unsigned int mapped_pages, i, linear_pages; |
283 | unsigned int ptes_per_page = getpagesize()/sizeof(u32); | 443 | unsigned int ptes_per_page = getpagesize()/sizeof(u32); |
284 | 444 | ||
285 | /* If we can map all of memory above page_offset, we do so. */ | 445 | /* Ideally we map all physical memory starting at page_offset. |
446 | * However, if page_offset is 0xC0000000 we can only map 1G of physical | ||
447 | * (0xC0000000 + 1G overflows). */ | ||
286 | if (mem <= -page_offset) | 448 | if (mem <= -page_offset) |
287 | mapped_pages = mem/getpagesize(); | 449 | mapped_pages = mem/getpagesize(); |
288 | else | 450 | else |
289 | mapped_pages = -page_offset/getpagesize(); | 451 | mapped_pages = -page_offset/getpagesize(); |
290 | 452 | ||
291 | /* Each linear PTE page can map ptes_per_page pages. */ | 453 | /* Each PTE page can map ptes_per_page pages: how many do we need? */ |
292 | linear_pages = (mapped_pages + ptes_per_page-1)/ptes_per_page; | 454 | linear_pages = (mapped_pages + ptes_per_page-1)/ptes_per_page; |
293 | 455 | ||
294 | /* We lay out top-level then linear mapping immediately below initrd */ | 456 | /* We put the toplevel page directory page at the top of memory. */ |
295 | pgdir = (void *)mem - initrd_size - getpagesize(); | 457 | pgdir = (void *)mem - initrd_size - getpagesize(); |
458 | |||
459 | /* Now we use the next linear_pages pages as pte pages */ | ||
296 | linear = (void *)pgdir - linear_pages*getpagesize(); | 460 | linear = (void *)pgdir - linear_pages*getpagesize(); |
297 | 461 | ||
462 | /* Linear mapping is easy: put every page's address into the mapping in | ||
463 | * order. PAGE_PRESENT contains the flags Present, Writable and | ||
464 | * Executable. */ | ||
298 | for (i = 0; i < mapped_pages; i++) | 465 | for (i = 0; i < mapped_pages; i++) |
299 | linear[i] = ((i * getpagesize()) | PAGE_PRESENT); | 466 | linear[i] = ((i * getpagesize()) | PAGE_PRESENT); |
300 | 467 | ||
301 | /* Now set up pgd so that this memory is at page_offset */ | 468 | /* The top level points to the linear page table pages above. The |
469 | * entry representing page_offset points to the first one, and they | ||
470 | * continue from there. */ | ||
302 | for (i = 0; i < mapped_pages; i += ptes_per_page) { | 471 | for (i = 0; i < mapped_pages; i += ptes_per_page) { |
303 | pgdir[(i + page_offset/getpagesize())/ptes_per_page] | 472 | pgdir[(i + page_offset/getpagesize())/ptes_per_page] |
304 | = (((u32)linear + i*sizeof(u32)) | PAGE_PRESENT); | 473 | = (((u32)linear + i*sizeof(u32)) | PAGE_PRESENT); |
@@ -307,9 +476,13 @@ static unsigned long setup_pagetables(unsigned long mem, | |||
307 | verbose("Linear mapping of %u pages in %u pte pages at %p\n", | 476 | verbose("Linear mapping of %u pages in %u pte pages at %p\n", |
308 | mapped_pages, linear_pages, linear); | 477 | mapped_pages, linear_pages, linear); |
309 | 478 | ||
479 | /* We return the top level (guest-physical) address: the kernel needs | ||
480 | * to know where it is. */ | ||
310 | return (unsigned long)pgdir; | 481 | return (unsigned long)pgdir; |
311 | } | 482 | } |
312 | 483 | ||
484 | /* Simple routine to roll all the commandline arguments together with spaces | ||
485 | * between them. */ | ||
313 | static void concat(char *dst, char *args[]) | 486 | static void concat(char *dst, char *args[]) |
314 | { | 487 | { |
315 | unsigned int i, len = 0; | 488 | unsigned int i, len = 0; |
@@ -323,6 +496,10 @@ static void concat(char *dst, char *args[]) | |||
323 | dst[len] = '\0'; | 496 | dst[len] = '\0'; |
324 | } | 497 | } |
325 | 498 | ||
499 | /* This is where we actually tell the kernel to initialize the Guest. We saw | ||
500 | * the arguments it expects when we looked at initialize() in lguest_user.c: | ||
501 | * the top physical page to allow, the top level pagetable, the entry point and | ||
502 | * the page_offset constant for the Guest. */ | ||
326 | static int tell_kernel(u32 pgdir, u32 start, u32 page_offset) | 503 | static int tell_kernel(u32 pgdir, u32 start, u32 page_offset) |
327 | { | 504 | { |
328 | u32 args[] = { LHREQ_INITIALIZE, | 505 | u32 args[] = { LHREQ_INITIALIZE, |
@@ -332,8 +509,11 @@ static int tell_kernel(u32 pgdir, u32 start, u32 page_offset) | |||
332 | fd = open_or_die("/dev/lguest", O_RDWR); | 509 | fd = open_or_die("/dev/lguest", O_RDWR); |
333 | if (write(fd, args, sizeof(args)) < 0) | 510 | if (write(fd, args, sizeof(args)) < 0) |
334 | err(1, "Writing to /dev/lguest"); | 511 | err(1, "Writing to /dev/lguest"); |
512 | |||
513 | /* We return the /dev/lguest file descriptor to control this Guest */ | ||
335 | return fd; | 514 | return fd; |
336 | } | 515 | } |
516 | /*:*/ | ||
337 | 517 | ||
338 | static void set_fd(int fd, struct device_list *devices) | 518 | static void set_fd(int fd, struct device_list *devices) |
339 | { | 519 | { |
@@ -342,61 +522,108 @@ static void set_fd(int fd, struct device_list *devices) | |||
342 | devices->max_infd = fd; | 522 | devices->max_infd = fd; |
343 | } | 523 | } |
344 | 524 | ||
345 | /* When input arrives, we tell the kernel to kick lguest out with -EAGAIN. */ | 525 | /*L:200 |
526 | * The Waker. | ||
527 | * | ||
528 | * With a console and network devices, we can have lots of input which we need | ||
529 | * to process. We could try to tell the kernel what file descriptors to watch, | ||
530 | * but handing a file descriptor mask through to the kernel is fairly icky. | ||
531 | * | ||
532 | * Instead, we fork off a process which watches the file descriptors and writes | ||
533 | * the LHREQ_BREAK command to the /dev/lguest filedescriptor to tell the Host | ||
534 | * loop to stop running the Guest. This causes it to return from the | ||
535 | * /dev/lguest read with -EAGAIN, where it will write to /dev/lguest to reset | ||
536 | * the LHREQ_BREAK and wake us up again. | ||
537 | * | ||
538 | * This, of course, is merely a different *kind* of icky. | ||
539 | */ | ||
346 | static void wake_parent(int pipefd, int lguest_fd, struct device_list *devices) | 540 | static void wake_parent(int pipefd, int lguest_fd, struct device_list *devices) |
347 | { | 541 | { |
542 | /* Add the pipe from the Launcher to the fdset in the device_list, so | ||
543 | * we watch it, too. */ | ||
348 | set_fd(pipefd, devices); | 544 | set_fd(pipefd, devices); |
349 | 545 | ||
350 | for (;;) { | 546 | for (;;) { |
351 | fd_set rfds = devices->infds; | 547 | fd_set rfds = devices->infds; |
352 | u32 args[] = { LHREQ_BREAK, 1 }; | 548 | u32 args[] = { LHREQ_BREAK, 1 }; |
353 | 549 | ||
550 | /* Wait until input is ready from one of the devices. */ | ||
354 | select(devices->max_infd+1, &rfds, NULL, NULL, NULL); | 551 | select(devices->max_infd+1, &rfds, NULL, NULL, NULL); |
552 | /* Is it a message from the Launcher? */ | ||
355 | if (FD_ISSET(pipefd, &rfds)) { | 553 | if (FD_ISSET(pipefd, &rfds)) { |
356 | int ignorefd; | 554 | int ignorefd; |
555 | /* If read() returns 0, it means the Launcher has | ||
556 | * exited. We silently follow. */ | ||
357 | if (read(pipefd, &ignorefd, sizeof(ignorefd)) == 0) | 557 | if (read(pipefd, &ignorefd, sizeof(ignorefd)) == 0) |
358 | exit(0); | 558 | exit(0); |
559 | /* Otherwise it's telling us there's a problem with one | ||
560 | * of the devices, and we should ignore that file | ||
561 | * descriptor from now on. */ | ||
359 | FD_CLR(ignorefd, &devices->infds); | 562 | FD_CLR(ignorefd, &devices->infds); |
360 | } else | 563 | } else /* Send LHREQ_BREAK command. */ |
361 | write(lguest_fd, args, sizeof(args)); | 564 | write(lguest_fd, args, sizeof(args)); |
362 | } | 565 | } |
363 | } | 566 | } |
364 | 567 | ||
568 | /* This routine just sets up a pipe to the Waker process. */ | ||
365 | static int setup_waker(int lguest_fd, struct device_list *device_list) | 569 | static int setup_waker(int lguest_fd, struct device_list *device_list) |
366 | { | 570 | { |
367 | int pipefd[2], child; | 571 | int pipefd[2], child; |
368 | 572 | ||
573 | /* We create a pipe to talk to the waker, and also so it knows when the | ||
574 | * Launcher dies (and closes pipe). */ | ||
369 | pipe(pipefd); | 575 | pipe(pipefd); |
370 | child = fork(); | 576 | child = fork(); |
371 | if (child == -1) | 577 | if (child == -1) |
372 | err(1, "forking"); | 578 | err(1, "forking"); |
373 | 579 | ||
374 | if (child == 0) { | 580 | if (child == 0) { |
581 | /* Close the "writing" end of our copy of the pipe */ | ||
375 | close(pipefd[1]); | 582 | close(pipefd[1]); |
376 | wake_parent(pipefd[0], lguest_fd, device_list); | 583 | wake_parent(pipefd[0], lguest_fd, device_list); |
377 | } | 584 | } |
585 | /* Close the reading end of our copy of the pipe. */ | ||
378 | close(pipefd[0]); | 586 | close(pipefd[0]); |
379 | 587 | ||
588 | /* Here is the fd used to talk to the waker. */ | ||
380 | return pipefd[1]; | 589 | return pipefd[1]; |
381 | } | 590 | } |
382 | 591 | ||
592 | /*L:210 | ||
593 | * Device Handling. | ||
594 | * | ||
595 | * When the Guest sends DMA to us, it sends us an array of addresses and sizes. | ||
596 | * We need to make sure it's not trying to reach into the Launcher itself, so | ||
597 | * we have a convenient routine which check it and exits with an error message | ||
598 | * if something funny is going on: | ||
599 | */ | ||
383 | static void *_check_pointer(unsigned long addr, unsigned int size, | 600 | static void *_check_pointer(unsigned long addr, unsigned int size, |
384 | unsigned int line) | 601 | unsigned int line) |
385 | { | 602 | { |
603 | /* We have to separately check addr and addr+size, because size could | ||
604 | * be huge and addr + size might wrap around. */ | ||
386 | if (addr >= top || addr + size >= top) | 605 | if (addr >= top || addr + size >= top) |
387 | errx(1, "%s:%i: Invalid address %li", __FILE__, line, addr); | 606 | errx(1, "%s:%i: Invalid address %li", __FILE__, line, addr); |
607 | /* We return a pointer for the caller's convenience, now we know it's | ||
608 | * safe to use. */ | ||
388 | return (void *)addr; | 609 | return (void *)addr; |
389 | } | 610 | } |
611 | /* A macro which transparently hands the line number to the real function. */ | ||
390 | #define check_pointer(addr,size) _check_pointer(addr, size, __LINE__) | 612 | #define check_pointer(addr,size) _check_pointer(addr, size, __LINE__) |
391 | 613 | ||
392 | /* Returns pointer to dma->used_len */ | 614 | /* The Guest has given us the address of a "struct lguest_dma". We check it's |
615 | * OK and convert it to an iovec (which is a simple array of ptr/size | ||
616 | * pairs). */ | ||
393 | static u32 *dma2iov(unsigned long dma, struct iovec iov[], unsigned *num) | 617 | static u32 *dma2iov(unsigned long dma, struct iovec iov[], unsigned *num) |
394 | { | 618 | { |
395 | unsigned int i; | 619 | unsigned int i; |
396 | struct lguest_dma *udma; | 620 | struct lguest_dma *udma; |
397 | 621 | ||
622 | /* First we make sure that the array memory itself is valid. */ | ||
398 | udma = check_pointer(dma, sizeof(*udma)); | 623 | udma = check_pointer(dma, sizeof(*udma)); |
624 | /* Now we check each element */ | ||
399 | for (i = 0; i < LGUEST_MAX_DMA_SECTIONS; i++) { | 625 | for (i = 0; i < LGUEST_MAX_DMA_SECTIONS; i++) { |
626 | /* A zero length ends the array. */ | ||
400 | if (!udma->len[i]) | 627 | if (!udma->len[i]) |
401 | break; | 628 | break; |
402 | 629 | ||
@@ -404,9 +631,15 @@ static u32 *dma2iov(unsigned long dma, struct iovec iov[], unsigned *num) | |||
404 | iov[i].iov_len = udma->len[i]; | 631 | iov[i].iov_len = udma->len[i]; |
405 | } | 632 | } |
406 | *num = i; | 633 | *num = i; |
634 | |||
635 | /* We return the pointer to where the caller should write the amount of | ||
636 | * the buffer used. */ | ||
407 | return &udma->used_len; | 637 | return &udma->used_len; |
408 | } | 638 | } |
409 | 639 | ||
640 | /* This routine gets a DMA buffer from the Guest for a given key, and converts | ||
641 | * it to an iovec array. It returns the interrupt the Guest wants when we're | ||
642 | * finished, and a pointer to the "used_len" field to fill in. */ | ||
410 | static u32 *get_dma_buffer(int fd, void *key, | 643 | static u32 *get_dma_buffer(int fd, void *key, |
411 | struct iovec iov[], unsigned int *num, u32 *irq) | 644 | struct iovec iov[], unsigned int *num, u32 *irq) |
412 | { | 645 | { |
@@ -414,16 +647,21 @@ static u32 *get_dma_buffer(int fd, void *key, | |||
414 | unsigned long udma; | 647 | unsigned long udma; |
415 | u32 *res; | 648 | u32 *res; |
416 | 649 | ||
650 | /* Ask the kernel for a DMA buffer corresponding to this key. */ | ||
417 | udma = write(fd, buf, sizeof(buf)); | 651 | udma = write(fd, buf, sizeof(buf)); |
652 | /* They haven't registered any, or they're all used? */ | ||
418 | if (udma == (unsigned long)-1) | 653 | if (udma == (unsigned long)-1) |
419 | return NULL; | 654 | return NULL; |
420 | 655 | ||
421 | /* Kernel stashes irq in ->used_len. */ | 656 | /* Convert it into our iovec array */ |
422 | res = dma2iov(udma, iov, num); | 657 | res = dma2iov(udma, iov, num); |
658 | /* The kernel stashes irq in ->used_len to get it out to us. */ | ||
423 | *irq = *res; | 659 | *irq = *res; |
660 | /* Return a pointer to ((struct lguest_dma *)udma)->used_len. */ | ||
424 | return res; | 661 | return res; |
425 | } | 662 | } |
426 | 663 | ||
664 | /* This is a convenient routine to send the Guest an interrupt. */ | ||
427 | static void trigger_irq(int fd, u32 irq) | 665 | static void trigger_irq(int fd, u32 irq) |
428 | { | 666 | { |
429 | u32 buf[] = { LHREQ_IRQ, irq }; | 667 | u32 buf[] = { LHREQ_IRQ, irq }; |
@@ -431,6 +669,10 @@ static void trigger_irq(int fd, u32 irq) | |||
431 | err(1, "Triggering irq %i", irq); | 669 | err(1, "Triggering irq %i", irq); |
432 | } | 670 | } |
433 | 671 | ||
672 | /* This simply sets up an iovec array where we can put data to be discarded. | ||
673 | * This happens when the Guest doesn't want or can't handle the input: we have | ||
674 | * to get rid of it somewhere, and if we bury it in the ceiling space it will | ||
675 | * start to smell after a week. */ | ||
434 | static void discard_iovec(struct iovec *iov, unsigned int *num) | 676 | static void discard_iovec(struct iovec *iov, unsigned int *num) |
435 | { | 677 | { |
436 | static char discard_buf[1024]; | 678 | static char discard_buf[1024]; |
@@ -439,19 +681,24 @@ static void discard_iovec(struct iovec *iov, unsigned int *num) | |||
439 | iov->iov_len = sizeof(discard_buf); | 681 | iov->iov_len = sizeof(discard_buf); |
440 | } | 682 | } |
441 | 683 | ||
684 | /* Here is the input terminal setting we save, and the routine to restore them | ||
685 | * on exit so the user can see what they type next. */ | ||
442 | static struct termios orig_term; | 686 | static struct termios orig_term; |
443 | static void restore_term(void) | 687 | static void restore_term(void) |
444 | { | 688 | { |
445 | tcsetattr(STDIN_FILENO, TCSANOW, &orig_term); | 689 | tcsetattr(STDIN_FILENO, TCSANOW, &orig_term); |
446 | } | 690 | } |
447 | 691 | ||
692 | /* We associate some data with the console for our exit hack. */ | ||
448 | struct console_abort | 693 | struct console_abort |
449 | { | 694 | { |
695 | /* How many times have they hit ^C? */ | ||
450 | int count; | 696 | int count; |
697 | /* When did they start? */ | ||
451 | struct timeval start; | 698 | struct timeval start; |
452 | }; | 699 | }; |
453 | 700 | ||
454 | /* We DMA input to buffer bound at start of console page. */ | 701 | /* This is the routine which handles console input (ie. stdin). */ |
455 | static bool handle_console_input(int fd, struct device *dev) | 702 | static bool handle_console_input(int fd, struct device *dev) |
456 | { | 703 | { |
457 | u32 irq = 0, *lenp; | 704 | u32 irq = 0, *lenp; |
@@ -460,24 +707,38 @@ static bool handle_console_input(int fd, struct device *dev) | |||
460 | struct iovec iov[LGUEST_MAX_DMA_SECTIONS]; | 707 | struct iovec iov[LGUEST_MAX_DMA_SECTIONS]; |
461 | struct console_abort *abort = dev->priv; | 708 | struct console_abort *abort = dev->priv; |
462 | 709 | ||
710 | /* First we get the console buffer from the Guest. The key is dev->mem | ||
711 | * which was set to 0 in setup_console(). */ | ||
463 | lenp = get_dma_buffer(fd, dev->mem, iov, &num, &irq); | 712 | lenp = get_dma_buffer(fd, dev->mem, iov, &num, &irq); |
464 | if (!lenp) { | 713 | if (!lenp) { |
714 | /* If it's not ready for input, warn and set up to discard. */ | ||
465 | warn("console: no dma buffer!"); | 715 | warn("console: no dma buffer!"); |
466 | discard_iovec(iov, &num); | 716 | discard_iovec(iov, &num); |
467 | } | 717 | } |
468 | 718 | ||
719 | /* This is why we convert to iovecs: the readv() call uses them, and so | ||
720 | * it reads straight into the Guest's buffer. */ | ||
469 | len = readv(dev->fd, iov, num); | 721 | len = readv(dev->fd, iov, num); |
470 | if (len <= 0) { | 722 | if (len <= 0) { |
723 | /* This implies that the console is closed, is /dev/null, or | ||
724 | * something went terribly wrong. We still go through the rest | ||
725 | * of the logic, though, especially the exit handling below. */ | ||
471 | warnx("Failed to get console input, ignoring console."); | 726 | warnx("Failed to get console input, ignoring console."); |
472 | len = 0; | 727 | len = 0; |
473 | } | 728 | } |
474 | 729 | ||
730 | /* If we read the data into the Guest, fill in the length and send the | ||
731 | * interrupt. */ | ||
475 | if (lenp) { | 732 | if (lenp) { |
476 | *lenp = len; | 733 | *lenp = len; |
477 | trigger_irq(fd, irq); | 734 | trigger_irq(fd, irq); |
478 | } | 735 | } |
479 | 736 | ||
480 | /* Three ^C within one second? Exit. */ | 737 | /* Three ^C within one second? Exit. |
738 | * | ||
739 | * This is such a hack, but works surprisingly well. Each ^C has to be | ||
740 | * in a buffer by itself, so they can't be too fast. But we check that | ||
741 | * we get three within about a second, so they can't be too slow. */ | ||
481 | if (len == 1 && ((char *)iov[0].iov_base)[0] == 3) { | 742 | if (len == 1 && ((char *)iov[0].iov_base)[0] == 3) { |
482 | if (!abort->count++) | 743 | if (!abort->count++) |
483 | gettimeofday(&abort->start, NULL); | 744 | gettimeofday(&abort->start, NULL); |
@@ -485,43 +746,60 @@ static bool handle_console_input(int fd, struct device *dev) | |||
485 | struct timeval now; | 746 | struct timeval now; |
486 | gettimeofday(&now, NULL); | 747 | gettimeofday(&now, NULL); |
487 | if (now.tv_sec <= abort->start.tv_sec+1) { | 748 | if (now.tv_sec <= abort->start.tv_sec+1) { |
488 | /* Make sure waker is not blocked in BREAK */ | ||
489 | u32 args[] = { LHREQ_BREAK, 0 }; | 749 | u32 args[] = { LHREQ_BREAK, 0 }; |
750 | /* Close the fd so Waker will know it has to | ||
751 | * exit. */ | ||
490 | close(waker_fd); | 752 | close(waker_fd); |
753 | /* Just in case waker is blocked in BREAK, send | ||
754 | * unbreak now. */ | ||
491 | write(fd, args, sizeof(args)); | 755 | write(fd, args, sizeof(args)); |
492 | exit(2); | 756 | exit(2); |
493 | } | 757 | } |
494 | abort->count = 0; | 758 | abort->count = 0; |
495 | } | 759 | } |
496 | } else | 760 | } else |
761 | /* Any other key resets the abort counter. */ | ||
497 | abort->count = 0; | 762 | abort->count = 0; |
498 | 763 | ||
764 | /* Now, if we didn't read anything, put the input terminal back and | ||
765 | * return failure (meaning, don't call us again). */ | ||
499 | if (!len) { | 766 | if (!len) { |
500 | restore_term(); | 767 | restore_term(); |
501 | return false; | 768 | return false; |
502 | } | 769 | } |
770 | /* Everything went OK! */ | ||
503 | return true; | 771 | return true; |
504 | } | 772 | } |
505 | 773 | ||
774 | /* Handling console output is much simpler than input. */ | ||
506 | static u32 handle_console_output(int fd, const struct iovec *iov, | 775 | static u32 handle_console_output(int fd, const struct iovec *iov, |
507 | unsigned num, struct device*dev) | 776 | unsigned num, struct device*dev) |
508 | { | 777 | { |
778 | /* Whatever the Guest sends, write it to standard output. Return the | ||
779 | * number of bytes written. */ | ||
509 | return writev(STDOUT_FILENO, iov, num); | 780 | return writev(STDOUT_FILENO, iov, num); |
510 | } | 781 | } |
511 | 782 | ||
783 | /* Guest->Host network output is also pretty easy. */ | ||
512 | static u32 handle_tun_output(int fd, const struct iovec *iov, | 784 | static u32 handle_tun_output(int fd, const struct iovec *iov, |
513 | unsigned num, struct device *dev) | 785 | unsigned num, struct device *dev) |
514 | { | 786 | { |
515 | /* Now we've seen output, we should warn if we can't get buffers. */ | 787 | /* We put a flag in the "priv" pointer of the network device, and set |
788 | * it as soon as we see output. We'll see why in handle_tun_input() */ | ||
516 | *(bool *)dev->priv = true; | 789 | *(bool *)dev->priv = true; |
790 | /* Whatever packet the Guest sent us, write it out to the tun | ||
791 | * device. */ | ||
517 | return writev(dev->fd, iov, num); | 792 | return writev(dev->fd, iov, num); |
518 | } | 793 | } |
519 | 794 | ||
795 | /* This matches the peer_key() in lguest_net.c. The key for any given slot | ||
796 | * is the address of the network device's page plus 4 * the slot number. */ | ||
520 | static unsigned long peer_offset(unsigned int peernum) | 797 | static unsigned long peer_offset(unsigned int peernum) |
521 | { | 798 | { |
522 | return 4 * peernum; | 799 | return 4 * peernum; |
523 | } | 800 | } |
524 | 801 | ||
802 | /* This is where we handle a packet coming in from the tun device */ | ||
525 | static bool handle_tun_input(int fd, struct device *dev) | 803 | static bool handle_tun_input(int fd, struct device *dev) |
526 | { | 804 | { |
527 | u32 irq = 0, *lenp; | 805 | u32 irq = 0, *lenp; |
@@ -529,17 +807,28 @@ static bool handle_tun_input(int fd, struct device *dev) | |||
529 | unsigned num; | 807 | unsigned num; |
530 | struct iovec iov[LGUEST_MAX_DMA_SECTIONS]; | 808 | struct iovec iov[LGUEST_MAX_DMA_SECTIONS]; |
531 | 809 | ||
810 | /* First we get a buffer the Guest has bound to its key. */ | ||
532 | lenp = get_dma_buffer(fd, dev->mem+peer_offset(NET_PEERNUM), iov, &num, | 811 | lenp = get_dma_buffer(fd, dev->mem+peer_offset(NET_PEERNUM), iov, &num, |
533 | &irq); | 812 | &irq); |
534 | if (!lenp) { | 813 | if (!lenp) { |
814 | /* Now, it's expected that if we try to send a packet too | ||
815 | * early, the Guest won't be ready yet. This is why we set a | ||
816 | * flag when the Guest sends its first packet. If it's sent a | ||
817 | * packet we assume it should be ready to receive them. | ||
818 | * | ||
819 | * Actually, this is what the status bits in the descriptor are | ||
820 | * for: we should *use* them. FIXME! */ | ||
535 | if (*(bool *)dev->priv) | 821 | if (*(bool *)dev->priv) |
536 | warn("network: no dma buffer!"); | 822 | warn("network: no dma buffer!"); |
537 | discard_iovec(iov, &num); | 823 | discard_iovec(iov, &num); |
538 | } | 824 | } |
539 | 825 | ||
826 | /* Read the packet from the device directly into the Guest's buffer. */ | ||
540 | len = readv(dev->fd, iov, num); | 827 | len = readv(dev->fd, iov, num); |
541 | if (len <= 0) | 828 | if (len <= 0) |
542 | err(1, "reading network"); | 829 | err(1, "reading network"); |
830 | |||
831 | /* Write the used_len, and trigger the interrupt for the Guest */ | ||
543 | if (lenp) { | 832 | if (lenp) { |
544 | *lenp = len; | 833 | *lenp = len; |
545 | trigger_irq(fd, irq); | 834 | trigger_irq(fd, irq); |
@@ -547,9 +836,13 @@ static bool handle_tun_input(int fd, struct device *dev) | |||
547 | verbose("tun input packet len %i [%02x %02x] (%s)\n", len, | 836 | verbose("tun input packet len %i [%02x %02x] (%s)\n", len, |
548 | ((u8 *)iov[0].iov_base)[0], ((u8 *)iov[0].iov_base)[1], | 837 | ((u8 *)iov[0].iov_base)[0], ((u8 *)iov[0].iov_base)[1], |
549 | lenp ? "sent" : "discarded"); | 838 | lenp ? "sent" : "discarded"); |
839 | /* All good. */ | ||
550 | return true; | 840 | return true; |
551 | } | 841 | } |
552 | 842 | ||
843 | /* The last device handling routine is block output: the Guest has sent a DMA | ||
844 | * to the block device. It will have placed the command it wants in the | ||
845 | * "struct lguest_block_page". */ | ||
553 | static u32 handle_block_output(int fd, const struct iovec *iov, | 846 | static u32 handle_block_output(int fd, const struct iovec *iov, |
554 | unsigned num, struct device *dev) | 847 | unsigned num, struct device *dev) |
555 | { | 848 | { |
@@ -559,36 +852,64 @@ static u32 handle_block_output(int fd, const struct iovec *iov, | |||
559 | struct iovec reply[LGUEST_MAX_DMA_SECTIONS]; | 852 | struct iovec reply[LGUEST_MAX_DMA_SECTIONS]; |
560 | off64_t device_len, off = (off64_t)p->sector * 512; | 853 | off64_t device_len, off = (off64_t)p->sector * 512; |
561 | 854 | ||
855 | /* First we extract the device length from the dev->priv pointer. */ | ||
562 | device_len = *(off64_t *)dev->priv; | 856 | device_len = *(off64_t *)dev->priv; |
563 | 857 | ||
858 | /* We first check that the read or write is within the length of the | ||
859 | * block file. */ | ||
564 | if (off >= device_len) | 860 | if (off >= device_len) |
565 | err(1, "Bad offset %llu vs %llu", off, device_len); | 861 | err(1, "Bad offset %llu vs %llu", off, device_len); |
862 | /* Move to the right location in the block file. This shouldn't fail, | ||
863 | * but best to check. */ | ||
566 | if (lseek64(dev->fd, off, SEEK_SET) != off) | 864 | if (lseek64(dev->fd, off, SEEK_SET) != off) |
567 | err(1, "Bad seek to sector %i", p->sector); | 865 | err(1, "Bad seek to sector %i", p->sector); |
568 | 866 | ||
569 | verbose("Block: %s at offset %llu\n", p->type ? "WRITE" : "READ", off); | 867 | verbose("Block: %s at offset %llu\n", p->type ? "WRITE" : "READ", off); |
570 | 868 | ||
869 | /* They were supposed to bind a reply buffer at key equal to the start | ||
870 | * of the block device memory. We need this to tell them when the | ||
871 | * request is finished. */ | ||
571 | lenp = get_dma_buffer(fd, dev->mem, reply, &reply_num, &irq); | 872 | lenp = get_dma_buffer(fd, dev->mem, reply, &reply_num, &irq); |
572 | if (!lenp) | 873 | if (!lenp) |
573 | err(1, "Block request didn't give us a dma buffer"); | 874 | err(1, "Block request didn't give us a dma buffer"); |
574 | 875 | ||
575 | if (p->type) { | 876 | if (p->type) { |
877 | /* A write request. The DMA they sent contained the data, so | ||
878 | * write it out. */ | ||
576 | len = writev(dev->fd, iov, num); | 879 | len = writev(dev->fd, iov, num); |
880 | /* Grr... Now we know how long the "struct lguest_dma" they | ||
881 | * sent was, we make sure they didn't try to write over the end | ||
882 | * of the block file (possibly extending it). */ | ||
577 | if (off + len > device_len) { | 883 | if (off + len > device_len) { |
884 | /* Trim it back to the correct length */ | ||
578 | ftruncate(dev->fd, device_len); | 885 | ftruncate(dev->fd, device_len); |
886 | /* Die, bad Guest, die. */ | ||
579 | errx(1, "Write past end %llu+%u", off, len); | 887 | errx(1, "Write past end %llu+%u", off, len); |
580 | } | 888 | } |
889 | /* The reply length is 0: we just send back an empty DMA to | ||
890 | * interrupt them and tell them the write is finished. */ | ||
581 | *lenp = 0; | 891 | *lenp = 0; |
582 | } else { | 892 | } else { |
893 | /* A read request. They sent an empty DMA to start the | ||
894 | * request, and we put the read contents into the reply | ||
895 | * buffer. */ | ||
583 | len = readv(dev->fd, reply, reply_num); | 896 | len = readv(dev->fd, reply, reply_num); |
584 | *lenp = len; | 897 | *lenp = len; |
585 | } | 898 | } |
586 | 899 | ||
900 | /* The result is 1 (done), 2 if there was an error (short read or | ||
901 | * write). */ | ||
587 | p->result = 1 + (p->bytes != len); | 902 | p->result = 1 + (p->bytes != len); |
903 | /* Now tell them we've used their reply buffer. */ | ||
588 | trigger_irq(fd, irq); | 904 | trigger_irq(fd, irq); |
905 | |||
906 | /* We're supposed to return the number of bytes of the output buffer we | ||
907 | * used. But the block device uses the "result" field instead, so we | ||
908 | * don't bother. */ | ||
589 | return 0; | 909 | return 0; |
590 | } | 910 | } |
591 | 911 | ||
912 | /* This is the generic routine we call when the Guest sends some DMA out. */ | ||
592 | static void handle_output(int fd, unsigned long dma, unsigned long key, | 913 | static void handle_output(int fd, unsigned long dma, unsigned long key, |
593 | struct device_list *devices) | 914 | struct device_list *devices) |
594 | { | 915 | { |
@@ -597,30 +918,53 @@ static void handle_output(int fd, unsigned long dma, unsigned long key, | |||
597 | struct iovec iov[LGUEST_MAX_DMA_SECTIONS]; | 918 | struct iovec iov[LGUEST_MAX_DMA_SECTIONS]; |
598 | unsigned num = 0; | 919 | unsigned num = 0; |
599 | 920 | ||
921 | /* Convert the "struct lguest_dma" they're sending to a "struct | ||
922 | * iovec". */ | ||
600 | lenp = dma2iov(dma, iov, &num); | 923 | lenp = dma2iov(dma, iov, &num); |
924 | |||
925 | /* Check each device: if they expect output to this key, tell them to | ||
926 | * handle it. */ | ||
601 | for (i = devices->dev; i; i = i->next) { | 927 | for (i = devices->dev; i; i = i->next) { |
602 | if (i->handle_output && key == i->watch_key) { | 928 | if (i->handle_output && key == i->watch_key) { |
929 | /* We write the result straight into the used_len field | ||
930 | * for them. */ | ||
603 | *lenp = i->handle_output(fd, iov, num, i); | 931 | *lenp = i->handle_output(fd, iov, num, i); |
604 | return; | 932 | return; |
605 | } | 933 | } |
606 | } | 934 | } |
935 | |||
936 | /* This can happen: the kernel sends any SEND_DMA which doesn't match | ||
937 | * another Guest to us. It could be that another Guest just left a | ||
938 | * network, for example. But it's unusual. */ | ||
607 | warnx("Pending dma %p, key %p", (void *)dma, (void *)key); | 939 | warnx("Pending dma %p, key %p", (void *)dma, (void *)key); |
608 | } | 940 | } |
609 | 941 | ||
942 | /* This is called when the waker wakes us up: check for incoming file | ||
943 | * descriptors. */ | ||
610 | static void handle_input(int fd, struct device_list *devices) | 944 | static void handle_input(int fd, struct device_list *devices) |
611 | { | 945 | { |
946 | /* select() wants a zeroed timeval to mean "don't wait". */ | ||
612 | struct timeval poll = { .tv_sec = 0, .tv_usec = 0 }; | 947 | struct timeval poll = { .tv_sec = 0, .tv_usec = 0 }; |
613 | 948 | ||
614 | for (;;) { | 949 | for (;;) { |
615 | struct device *i; | 950 | struct device *i; |
616 | fd_set fds = devices->infds; | 951 | fd_set fds = devices->infds; |
617 | 952 | ||
953 | /* If nothing is ready, we're done. */ | ||
618 | if (select(devices->max_infd+1, &fds, NULL, NULL, &poll) == 0) | 954 | if (select(devices->max_infd+1, &fds, NULL, NULL, &poll) == 0) |
619 | break; | 955 | break; |
620 | 956 | ||
957 | /* Otherwise, call the device(s) which have readable | ||
958 | * file descriptors and a method of handling them. */ | ||
621 | for (i = devices->dev; i; i = i->next) { | 959 | for (i = devices->dev; i; i = i->next) { |
622 | if (i->handle_input && FD_ISSET(i->fd, &fds)) { | 960 | if (i->handle_input && FD_ISSET(i->fd, &fds)) { |
961 | /* If handle_input() returns false, it means we | ||
962 | * should no longer service it. | ||
963 | * handle_console_input() does this. */ | ||
623 | if (!i->handle_input(fd, i)) { | 964 | if (!i->handle_input(fd, i)) { |
965 | /* Clear it from the set of input file | ||
966 | * descriptors kept at the head of the | ||
967 | * device list. */ | ||
624 | FD_CLR(i->fd, &devices->infds); | 968 | FD_CLR(i->fd, &devices->infds); |
625 | /* Tell waker to ignore it too... */ | 969 | /* Tell waker to ignore it too... */ |
626 | write(waker_fd, &i->fd, sizeof(i->fd)); | 970 | write(waker_fd, &i->fd, sizeof(i->fd)); |
@@ -630,6 +974,15 @@ static void handle_input(int fd, struct device_list *devices) | |||
630 | } | 974 | } |
631 | } | 975 | } |
632 | 976 | ||
977 | /*L:190 | ||
978 | * Device Setup | ||
979 | * | ||
980 | * All devices need a descriptor so the Guest knows it exists, and a "struct | ||
981 | * device" so the Launcher can keep track of it. We have common helper | ||
982 | * routines to allocate them. | ||
983 | * | ||
984 | * This routine allocates a new "struct lguest_device_desc" from descriptor | ||
985 | * table in the devices array just above the Guest's normal memory. */ | ||
633 | static struct lguest_device_desc * | 986 | static struct lguest_device_desc * |
634 | new_dev_desc(struct lguest_device_desc *descs, | 987 | new_dev_desc(struct lguest_device_desc *descs, |
635 | u16 type, u16 features, u16 num_pages) | 988 | u16 type, u16 features, u16 num_pages) |
@@ -641,6 +994,8 @@ new_dev_desc(struct lguest_device_desc *descs, | |||
641 | descs[i].type = type; | 994 | descs[i].type = type; |
642 | descs[i].features = features; | 995 | descs[i].features = features; |
643 | descs[i].num_pages = num_pages; | 996 | descs[i].num_pages = num_pages; |
997 | /* If they said the device needs memory, we allocate | ||
998 | * that now, bumping up the top of Guest memory. */ | ||
644 | if (num_pages) { | 999 | if (num_pages) { |
645 | map_zeroed_pages(top, num_pages); | 1000 | map_zeroed_pages(top, num_pages); |
646 | descs[i].pfn = top/getpagesize(); | 1001 | descs[i].pfn = top/getpagesize(); |
@@ -652,6 +1007,9 @@ new_dev_desc(struct lguest_device_desc *descs, | |||
652 | errx(1, "too many devices"); | 1007 | errx(1, "too many devices"); |
653 | } | 1008 | } |
654 | 1009 | ||
1010 | /* This monster routine does all the creation and setup of a new device, | ||
1011 | * including caling new_dev_desc() to allocate the descriptor and device | ||
1012 | * memory. */ | ||
655 | static struct device *new_device(struct device_list *devices, | 1013 | static struct device *new_device(struct device_list *devices, |
656 | u16 type, u16 num_pages, u16 features, | 1014 | u16 type, u16 num_pages, u16 features, |
657 | int fd, | 1015 | int fd, |
@@ -664,12 +1022,18 @@ static struct device *new_device(struct device_list *devices, | |||
664 | { | 1022 | { |
665 | struct device *dev = malloc(sizeof(*dev)); | 1023 | struct device *dev = malloc(sizeof(*dev)); |
666 | 1024 | ||
667 | /* Append to device list. */ | 1025 | /* Append to device list. Prepending to a single-linked list is |
1026 | * easier, but the user expects the devices to be arranged on the bus | ||
1027 | * in command-line order. The first network device on the command line | ||
1028 | * is eth0, the first block device /dev/lgba, etc. */ | ||
668 | *devices->lastdev = dev; | 1029 | *devices->lastdev = dev; |
669 | dev->next = NULL; | 1030 | dev->next = NULL; |
670 | devices->lastdev = &dev->next; | 1031 | devices->lastdev = &dev->next; |
671 | 1032 | ||
1033 | /* Now we populate the fields one at a time. */ | ||
672 | dev->fd = fd; | 1034 | dev->fd = fd; |
1035 | /* If we have an input handler for this file descriptor, then we add it | ||
1036 | * to the device_list's fdset and maxfd. */ | ||
673 | if (handle_input) | 1037 | if (handle_input) |
674 | set_fd(dev->fd, devices); | 1038 | set_fd(dev->fd, devices); |
675 | dev->desc = new_dev_desc(devices->descs, type, features, num_pages); | 1039 | dev->desc = new_dev_desc(devices->descs, type, features, num_pages); |
@@ -680,27 +1044,37 @@ static struct device *new_device(struct device_list *devices, | |||
680 | return dev; | 1044 | return dev; |
681 | } | 1045 | } |
682 | 1046 | ||
1047 | /* Our first setup routine is the console. It's a fairly simple device, but | ||
1048 | * UNIX tty handling makes it uglier than it could be. */ | ||
683 | static void setup_console(struct device_list *devices) | 1049 | static void setup_console(struct device_list *devices) |
684 | { | 1050 | { |
685 | struct device *dev; | 1051 | struct device *dev; |
686 | 1052 | ||
1053 | /* If we can save the initial standard input settings... */ | ||
687 | if (tcgetattr(STDIN_FILENO, &orig_term) == 0) { | 1054 | if (tcgetattr(STDIN_FILENO, &orig_term) == 0) { |
688 | struct termios term = orig_term; | 1055 | struct termios term = orig_term; |
1056 | /* Then we turn off echo, line buffering and ^C etc. We want a | ||
1057 | * raw input stream to the Guest. */ | ||
689 | term.c_lflag &= ~(ISIG|ICANON|ECHO); | 1058 | term.c_lflag &= ~(ISIG|ICANON|ECHO); |
690 | tcsetattr(STDIN_FILENO, TCSANOW, &term); | 1059 | tcsetattr(STDIN_FILENO, TCSANOW, &term); |
1060 | /* If we exit gracefully, the original settings will be | ||
1061 | * restored so the user can see what they're typing. */ | ||
691 | atexit(restore_term); | 1062 | atexit(restore_term); |
692 | } | 1063 | } |
693 | 1064 | ||
694 | /* We don't currently require a page for the console. */ | 1065 | /* We don't currently require any memory for the console, so we ask for |
1066 | * 0 pages. */ | ||
695 | dev = new_device(devices, LGUEST_DEVICE_T_CONSOLE, 0, 0, | 1067 | dev = new_device(devices, LGUEST_DEVICE_T_CONSOLE, 0, 0, |
696 | STDIN_FILENO, handle_console_input, | 1068 | STDIN_FILENO, handle_console_input, |
697 | LGUEST_CONSOLE_DMA_KEY, handle_console_output); | 1069 | LGUEST_CONSOLE_DMA_KEY, handle_console_output); |
1070 | /* We store the console state in dev->priv, and initialize it. */ | ||
698 | dev->priv = malloc(sizeof(struct console_abort)); | 1071 | dev->priv = malloc(sizeof(struct console_abort)); |
699 | ((struct console_abort *)dev->priv)->count = 0; | 1072 | ((struct console_abort *)dev->priv)->count = 0; |
700 | verbose("device %p: console\n", | 1073 | verbose("device %p: console\n", |
701 | (void *)(dev->desc->pfn * getpagesize())); | 1074 | (void *)(dev->desc->pfn * getpagesize())); |
702 | } | 1075 | } |
703 | 1076 | ||
1077 | /* Setting up a block file is also fairly straightforward. */ | ||
704 | static void setup_block_file(const char *filename, struct device_list *devices) | 1078 | static void setup_block_file(const char *filename, struct device_list *devices) |
705 | { | 1079 | { |
706 | int fd; | 1080 | int fd; |
@@ -708,20 +1082,47 @@ static void setup_block_file(const char *filename, struct device_list *devices) | |||
708 | off64_t *device_len; | 1082 | off64_t *device_len; |
709 | struct lguest_block_page *p; | 1083 | struct lguest_block_page *p; |
710 | 1084 | ||
1085 | /* We open with O_LARGEFILE because otherwise we get stuck at 2G. We | ||
1086 | * open with O_DIRECT because otherwise our benchmarks go much too | ||
1087 | * fast. */ | ||
711 | fd = open_or_die(filename, O_RDWR|O_LARGEFILE|O_DIRECT); | 1088 | fd = open_or_die(filename, O_RDWR|O_LARGEFILE|O_DIRECT); |
1089 | |||
1090 | /* We want one page, and have no input handler (the block file never | ||
1091 | * has anything interesting to say to us). Our timing will be quite | ||
1092 | * random, so it should be a reasonable randomness source. */ | ||
712 | dev = new_device(devices, LGUEST_DEVICE_T_BLOCK, 1, | 1093 | dev = new_device(devices, LGUEST_DEVICE_T_BLOCK, 1, |
713 | LGUEST_DEVICE_F_RANDOMNESS, | 1094 | LGUEST_DEVICE_F_RANDOMNESS, |
714 | fd, NULL, 0, handle_block_output); | 1095 | fd, NULL, 0, handle_block_output); |
1096 | |||
1097 | /* We store the device size in the private area */ | ||
715 | device_len = dev->priv = malloc(sizeof(*device_len)); | 1098 | device_len = dev->priv = malloc(sizeof(*device_len)); |
1099 | /* This is the safe way of establishing the size of our device: it | ||
1100 | * might be a normal file or an actual block device like /dev/hdb. */ | ||
716 | *device_len = lseek64(fd, 0, SEEK_END); | 1101 | *device_len = lseek64(fd, 0, SEEK_END); |
717 | p = dev->mem; | ||
718 | 1102 | ||
1103 | /* The device memory is a "struct lguest_block_page". It's zeroed | ||
1104 | * already, we just need to put in the device size. Block devices | ||
1105 | * think in sectors (ie. 512 byte chunks), so we translate here. */ | ||
1106 | p = dev->mem; | ||
719 | p->num_sectors = *device_len/512; | 1107 | p->num_sectors = *device_len/512; |
720 | verbose("device %p: block %i sectors\n", | 1108 | verbose("device %p: block %i sectors\n", |
721 | (void *)(dev->desc->pfn * getpagesize()), p->num_sectors); | 1109 | (void *)(dev->desc->pfn * getpagesize()), p->num_sectors); |
722 | } | 1110 | } |
723 | 1111 | ||
724 | /* We use fnctl locks to reserve network slots (autocleanup!) */ | 1112 | /* |
1113 | * Network Devices. | ||
1114 | * | ||
1115 | * Setting up network devices is quite a pain, because we have three types. | ||
1116 | * First, we have the inter-Guest network. This is a file which is mapped into | ||
1117 | * the address space of the Guests who are on the network. Because it is a | ||
1118 | * shared mapping, the same page underlies all the devices, and they can send | ||
1119 | * DMA to each other. | ||
1120 | * | ||
1121 | * Remember from our network driver, the Guest is told what slot in the page it | ||
1122 | * is to use. We use exclusive fnctl locks to reserve a slot. If another | ||
1123 | * Guest is using a slot, the lock will fail and we try another. Because fnctl | ||
1124 | * locks are cleaned up automatically when we die, this cleverly means that our | ||
1125 | * reservation on the slot will vanish if we crash. */ | ||
725 | static unsigned int find_slot(int netfd, const char *filename) | 1126 | static unsigned int find_slot(int netfd, const char *filename) |
726 | { | 1127 | { |
727 | struct flock fl; | 1128 | struct flock fl; |
@@ -729,26 +1130,33 @@ static unsigned int find_slot(int netfd, const char *filename) | |||
729 | fl.l_type = F_WRLCK; | 1130 | fl.l_type = F_WRLCK; |
730 | fl.l_whence = SEEK_SET; | 1131 | fl.l_whence = SEEK_SET; |
731 | fl.l_len = 1; | 1132 | fl.l_len = 1; |
1133 | /* Try a 1 byte lock in each possible position number */ | ||
732 | for (fl.l_start = 0; | 1134 | for (fl.l_start = 0; |
733 | fl.l_start < getpagesize()/sizeof(struct lguest_net); | 1135 | fl.l_start < getpagesize()/sizeof(struct lguest_net); |
734 | fl.l_start++) { | 1136 | fl.l_start++) { |
1137 | /* If we succeed, return the slot number. */ | ||
735 | if (fcntl(netfd, F_SETLK, &fl) == 0) | 1138 | if (fcntl(netfd, F_SETLK, &fl) == 0) |
736 | return fl.l_start; | 1139 | return fl.l_start; |
737 | } | 1140 | } |
738 | errx(1, "No free slots in network file %s", filename); | 1141 | errx(1, "No free slots in network file %s", filename); |
739 | } | 1142 | } |
740 | 1143 | ||
1144 | /* This function sets up the network file */ | ||
741 | static void setup_net_file(const char *filename, | 1145 | static void setup_net_file(const char *filename, |
742 | struct device_list *devices) | 1146 | struct device_list *devices) |
743 | { | 1147 | { |
744 | int netfd; | 1148 | int netfd; |
745 | struct device *dev; | 1149 | struct device *dev; |
746 | 1150 | ||
1151 | /* We don't use open_or_die() here: for friendliness we create the file | ||
1152 | * if it doesn't already exist. */ | ||
747 | netfd = open(filename, O_RDWR, 0); | 1153 | netfd = open(filename, O_RDWR, 0); |
748 | if (netfd < 0) { | 1154 | if (netfd < 0) { |
749 | if (errno == ENOENT) { | 1155 | if (errno == ENOENT) { |
750 | netfd = open(filename, O_RDWR|O_CREAT, 0600); | 1156 | netfd = open(filename, O_RDWR|O_CREAT, 0600); |
751 | if (netfd >= 0) { | 1157 | if (netfd >= 0) { |
1158 | /* If we succeeded, initialize the file with a | ||
1159 | * blank page. */ | ||
752 | char page[getpagesize()]; | 1160 | char page[getpagesize()]; |
753 | memset(page, 0, sizeof(page)); | 1161 | memset(page, 0, sizeof(page)); |
754 | write(netfd, page, sizeof(page)); | 1162 | write(netfd, page, sizeof(page)); |
@@ -758,11 +1166,15 @@ static void setup_net_file(const char *filename, | |||
758 | err(1, "cannot open net file '%s'", filename); | 1166 | err(1, "cannot open net file '%s'", filename); |
759 | } | 1167 | } |
760 | 1168 | ||
1169 | /* We need 1 page, and the features indicate the slot to use and that | ||
1170 | * no checksum is needed. We never touch this device again; it's | ||
1171 | * between the Guests on the network, so we don't register input or | ||
1172 | * output handlers. */ | ||
761 | dev = new_device(devices, LGUEST_DEVICE_T_NET, 1, | 1173 | dev = new_device(devices, LGUEST_DEVICE_T_NET, 1, |
762 | find_slot(netfd, filename)|LGUEST_NET_F_NOCSUM, | 1174 | find_slot(netfd, filename)|LGUEST_NET_F_NOCSUM, |
763 | -1, NULL, 0, NULL); | 1175 | -1, NULL, 0, NULL); |
764 | 1176 | ||
765 | /* We overwrite the /dev/zero mapping with the actual file. */ | 1177 | /* Map the shared file. */ |
766 | if (mmap(dev->mem, getpagesize(), PROT_READ|PROT_WRITE, | 1178 | if (mmap(dev->mem, getpagesize(), PROT_READ|PROT_WRITE, |
767 | MAP_FIXED|MAP_SHARED, netfd, 0) != dev->mem) | 1179 | MAP_FIXED|MAP_SHARED, netfd, 0) != dev->mem) |
768 | err(1, "could not mmap '%s'", filename); | 1180 | err(1, "could not mmap '%s'", filename); |
@@ -770,6 +1182,7 @@ static void setup_net_file(const char *filename, | |||
770 | (void *)(dev->desc->pfn * getpagesize()), filename, | 1182 | (void *)(dev->desc->pfn * getpagesize()), filename, |
771 | dev->desc->features & ~LGUEST_NET_F_NOCSUM); | 1183 | dev->desc->features & ~LGUEST_NET_F_NOCSUM); |
772 | } | 1184 | } |
1185 | /*:*/ | ||
773 | 1186 | ||
774 | static u32 str2ip(const char *ipaddr) | 1187 | static u32 str2ip(const char *ipaddr) |
775 | { | 1188 | { |
@@ -779,7 +1192,11 @@ static u32 str2ip(const char *ipaddr) | |||
779 | return (byte[0] << 24) | (byte[1] << 16) | (byte[2] << 8) | byte[3]; | 1192 | return (byte[0] << 24) | (byte[1] << 16) | (byte[2] << 8) | byte[3]; |
780 | } | 1193 | } |
781 | 1194 | ||
782 | /* adapted from libbridge */ | 1195 | /* This code is "adapted" from libbridge: it attaches the Host end of the |
1196 | * network device to the bridge device specified by the command line. | ||
1197 | * | ||
1198 | * This is yet another James Morris contribution (I'm an IP-level guy, so I | ||
1199 | * dislike bridging), and I just try not to break it. */ | ||
783 | static void add_to_bridge(int fd, const char *if_name, const char *br_name) | 1200 | static void add_to_bridge(int fd, const char *if_name, const char *br_name) |
784 | { | 1201 | { |
785 | int ifidx; | 1202 | int ifidx; |
@@ -798,12 +1215,16 @@ static void add_to_bridge(int fd, const char *if_name, const char *br_name) | |||
798 | err(1, "can't add %s to bridge %s", if_name, br_name); | 1215 | err(1, "can't add %s to bridge %s", if_name, br_name); |
799 | } | 1216 | } |
800 | 1217 | ||
1218 | /* This sets up the Host end of the network device with an IP address, brings | ||
1219 | * it up so packets will flow, the copies the MAC address into the hwaddr | ||
1220 | * pointer (in practice, the Host's slot in the network device's memory). */ | ||
801 | static void configure_device(int fd, const char *devname, u32 ipaddr, | 1221 | static void configure_device(int fd, const char *devname, u32 ipaddr, |
802 | unsigned char hwaddr[6]) | 1222 | unsigned char hwaddr[6]) |
803 | { | 1223 | { |
804 | struct ifreq ifr; | 1224 | struct ifreq ifr; |
805 | struct sockaddr_in *sin = (struct sockaddr_in *)&ifr.ifr_addr; | 1225 | struct sockaddr_in *sin = (struct sockaddr_in *)&ifr.ifr_addr; |
806 | 1226 | ||
1227 | /* Don't read these incantations. Just cut & paste them like I did! */ | ||
807 | memset(&ifr, 0, sizeof(ifr)); | 1228 | memset(&ifr, 0, sizeof(ifr)); |
808 | strcpy(ifr.ifr_name, devname); | 1229 | strcpy(ifr.ifr_name, devname); |
809 | sin->sin_family = AF_INET; | 1230 | sin->sin_family = AF_INET; |
@@ -814,12 +1235,19 @@ static void configure_device(int fd, const char *devname, u32 ipaddr, | |||
814 | if (ioctl(fd, SIOCSIFFLAGS, &ifr) != 0) | 1235 | if (ioctl(fd, SIOCSIFFLAGS, &ifr) != 0) |
815 | err(1, "Bringing interface %s up", devname); | 1236 | err(1, "Bringing interface %s up", devname); |
816 | 1237 | ||
1238 | /* SIOC stands for Socket I/O Control. G means Get (vs S for Set | ||
1239 | * above). IF means Interface, and HWADDR is hardware address. | ||
1240 | * Simple! */ | ||
817 | if (ioctl(fd, SIOCGIFHWADDR, &ifr) != 0) | 1241 | if (ioctl(fd, SIOCGIFHWADDR, &ifr) != 0) |
818 | err(1, "getting hw address for %s", devname); | 1242 | err(1, "getting hw address for %s", devname); |
819 | |||
820 | memcpy(hwaddr, ifr.ifr_hwaddr.sa_data, 6); | 1243 | memcpy(hwaddr, ifr.ifr_hwaddr.sa_data, 6); |
821 | } | 1244 | } |
822 | 1245 | ||
1246 | /*L:195 The other kind of network is a Host<->Guest network. This can either | ||
1247 | * use briding or routing, but the principle is the same: it uses the "tun" | ||
1248 | * device to inject packets into the Host as if they came in from a normal | ||
1249 | * network card. We just shunt packets between the Guest and the tun | ||
1250 | * device. */ | ||
823 | static void setup_tun_net(const char *arg, struct device_list *devices) | 1251 | static void setup_tun_net(const char *arg, struct device_list *devices) |
824 | { | 1252 | { |
825 | struct device *dev; | 1253 | struct device *dev; |
@@ -828,36 +1256,56 @@ static void setup_tun_net(const char *arg, struct device_list *devices) | |||
828 | u32 ip; | 1256 | u32 ip; |
829 | const char *br_name = NULL; | 1257 | const char *br_name = NULL; |
830 | 1258 | ||
1259 | /* We open the /dev/net/tun device and tell it we want a tap device. A | ||
1260 | * tap device is like a tun device, only somehow different. To tell | ||
1261 | * the truth, I completely blundered my way through this code, but it | ||
1262 | * works now! */ | ||
831 | netfd = open_or_die("/dev/net/tun", O_RDWR); | 1263 | netfd = open_or_die("/dev/net/tun", O_RDWR); |
832 | memset(&ifr, 0, sizeof(ifr)); | 1264 | memset(&ifr, 0, sizeof(ifr)); |
833 | ifr.ifr_flags = IFF_TAP | IFF_NO_PI; | 1265 | ifr.ifr_flags = IFF_TAP | IFF_NO_PI; |
834 | strcpy(ifr.ifr_name, "tap%d"); | 1266 | strcpy(ifr.ifr_name, "tap%d"); |
835 | if (ioctl(netfd, TUNSETIFF, &ifr) != 0) | 1267 | if (ioctl(netfd, TUNSETIFF, &ifr) != 0) |
836 | err(1, "configuring /dev/net/tun"); | 1268 | err(1, "configuring /dev/net/tun"); |
1269 | /* We don't need checksums calculated for packets coming in this | ||
1270 | * device: trust us! */ | ||
837 | ioctl(netfd, TUNSETNOCSUM, 1); | 1271 | ioctl(netfd, TUNSETNOCSUM, 1); |
838 | 1272 | ||
839 | /* You will be peer 1: we should create enough jitter to randomize */ | 1273 | /* We create the net device with 1 page, using the features field of |
1274 | * the descriptor to tell the Guest it is in slot 1 (NET_PEERNUM), and | ||
1275 | * that the device has fairly random timing. We do *not* specify | ||
1276 | * LGUEST_NET_F_NOCSUM: these packets can reach the real world. | ||
1277 | * | ||
1278 | * We will put our MAC address is slot 0 for the Guest to see, so | ||
1279 | * it will send packets to us using the key "peer_offset(0)": */ | ||
840 | dev = new_device(devices, LGUEST_DEVICE_T_NET, 1, | 1280 | dev = new_device(devices, LGUEST_DEVICE_T_NET, 1, |
841 | NET_PEERNUM|LGUEST_DEVICE_F_RANDOMNESS, netfd, | 1281 | NET_PEERNUM|LGUEST_DEVICE_F_RANDOMNESS, netfd, |
842 | handle_tun_input, peer_offset(0), handle_tun_output); | 1282 | handle_tun_input, peer_offset(0), handle_tun_output); |
1283 | |||
1284 | /* We keep a flag which says whether we've seen packets come out from | ||
1285 | * this network device. */ | ||
843 | dev->priv = malloc(sizeof(bool)); | 1286 | dev->priv = malloc(sizeof(bool)); |
844 | *(bool *)dev->priv = false; | 1287 | *(bool *)dev->priv = false; |
845 | 1288 | ||
1289 | /* We need a socket to perform the magic network ioctls to bring up the | ||
1290 | * tap interface, connect to the bridge etc. Any socket will do! */ | ||
846 | ipfd = socket(PF_INET, SOCK_DGRAM, IPPROTO_IP); | 1291 | ipfd = socket(PF_INET, SOCK_DGRAM, IPPROTO_IP); |
847 | if (ipfd < 0) | 1292 | if (ipfd < 0) |
848 | err(1, "opening IP socket"); | 1293 | err(1, "opening IP socket"); |
849 | 1294 | ||
1295 | /* If the command line was --tunnet=bridge:<name> do bridging. */ | ||
850 | if (!strncmp(BRIDGE_PFX, arg, strlen(BRIDGE_PFX))) { | 1296 | if (!strncmp(BRIDGE_PFX, arg, strlen(BRIDGE_PFX))) { |
851 | ip = INADDR_ANY; | 1297 | ip = INADDR_ANY; |
852 | br_name = arg + strlen(BRIDGE_PFX); | 1298 | br_name = arg + strlen(BRIDGE_PFX); |
853 | add_to_bridge(ipfd, ifr.ifr_name, br_name); | 1299 | add_to_bridge(ipfd, ifr.ifr_name, br_name); |
854 | } else | 1300 | } else /* It is an IP address to set up the device with */ |
855 | ip = str2ip(arg); | 1301 | ip = str2ip(arg); |
856 | 1302 | ||
857 | /* We are peer 0, ie. first slot. */ | 1303 | /* We are peer 0, ie. first slot, so we hand dev->mem to this routine |
1304 | * to write the MAC address at the start of the device memory. */ | ||
858 | configure_device(ipfd, ifr.ifr_name, ip, dev->mem); | 1305 | configure_device(ipfd, ifr.ifr_name, ip, dev->mem); |
859 | 1306 | ||
860 | /* Set "promisc" bit: we want every single packet. */ | 1307 | /* Set "promisc" bit: we want every single packet if we're going to |
1308 | * bridge to other machines (and otherwise it doesn't matter). */ | ||
861 | *((u8 *)dev->mem) |= 0x1; | 1309 | *((u8 *)dev->mem) |= 0x1; |
862 | 1310 | ||
863 | close(ipfd); | 1311 | close(ipfd); |
@@ -868,7 +1316,10 @@ static void setup_tun_net(const char *arg, struct device_list *devices) | |||
868 | if (br_name) | 1316 | if (br_name) |
869 | verbose("attached to bridge: %s\n", br_name); | 1317 | verbose("attached to bridge: %s\n", br_name); |
870 | } | 1318 | } |
1319 | /* That's the end of device setup. */ | ||
871 | 1320 | ||
1321 | /*L:220 Finally we reach the core of the Launcher, which runs the Guest, serves | ||
1322 | * its input and output, and finally, lays it to rest. */ | ||
872 | static void __attribute__((noreturn)) | 1323 | static void __attribute__((noreturn)) |
873 | run_guest(int lguest_fd, struct device_list *device_list) | 1324 | run_guest(int lguest_fd, struct device_list *device_list) |
874 | { | 1325 | { |
@@ -880,20 +1331,37 @@ run_guest(int lguest_fd, struct device_list *device_list) | |||
880 | /* We read from the /dev/lguest device to run the Guest. */ | 1331 | /* We read from the /dev/lguest device to run the Guest. */ |
881 | readval = read(lguest_fd, arr, sizeof(arr)); | 1332 | readval = read(lguest_fd, arr, sizeof(arr)); |
882 | 1333 | ||
1334 | /* The read can only really return sizeof(arr) (the Guest did a | ||
1335 | * SEND_DMA to us), or an error. */ | ||
1336 | |||
1337 | /* For a successful read, arr[0] is the address of the "struct | ||
1338 | * lguest_dma", and arr[1] is the key the Guest sent to. */ | ||
883 | if (readval == sizeof(arr)) { | 1339 | if (readval == sizeof(arr)) { |
884 | handle_output(lguest_fd, arr[0], arr[1], device_list); | 1340 | handle_output(lguest_fd, arr[0], arr[1], device_list); |
885 | continue; | 1341 | continue; |
1342 | /* ENOENT means the Guest died. Reading tells us why. */ | ||
886 | } else if (errno == ENOENT) { | 1343 | } else if (errno == ENOENT) { |
887 | char reason[1024] = { 0 }; | 1344 | char reason[1024] = { 0 }; |
888 | read(lguest_fd, reason, sizeof(reason)-1); | 1345 | read(lguest_fd, reason, sizeof(reason)-1); |
889 | errx(1, "%s", reason); | 1346 | errx(1, "%s", reason); |
1347 | /* EAGAIN means the waker wanted us to look at some input. | ||
1348 | * Anything else means a bug or incompatible change. */ | ||
890 | } else if (errno != EAGAIN) | 1349 | } else if (errno != EAGAIN) |
891 | err(1, "Running guest failed"); | 1350 | err(1, "Running guest failed"); |
1351 | |||
1352 | /* Service input, then unset the BREAK which releases | ||
1353 | * the Waker. */ | ||
892 | handle_input(lguest_fd, device_list); | 1354 | handle_input(lguest_fd, device_list); |
893 | if (write(lguest_fd, args, sizeof(args)) < 0) | 1355 | if (write(lguest_fd, args, sizeof(args)) < 0) |
894 | err(1, "Resetting break"); | 1356 | err(1, "Resetting break"); |
895 | } | 1357 | } |
896 | } | 1358 | } |
1359 | /* | ||
1360 | * This is the end of the Launcher. | ||
1361 | * | ||
1362 | * But wait! We've seen I/O from the Launcher, and we've seen I/O from the | ||
1363 | * Drivers. If we were to see the Host kernel I/O code, our understanding | ||
1364 | * would be complete... :*/ | ||
897 | 1365 | ||
898 | static struct option opts[] = { | 1366 | static struct option opts[] = { |
899 | { "verbose", 0, NULL, 'v' }, | 1367 | { "verbose", 0, NULL, 'v' }, |
@@ -911,20 +1379,49 @@ static void usage(void) | |||
911 | "<mem-in-mb> vmlinux [args...]"); | 1379 | "<mem-in-mb> vmlinux [args...]"); |
912 | } | 1380 | } |
913 | 1381 | ||
1382 | /*L:100 The Launcher code itself takes us out into userspace, that scary place | ||
1383 | * where pointers run wild and free! Unfortunately, like most userspace | ||
1384 | * programs, it's quite boring (which is why everyone like to hack on the | ||
1385 | * kernel!). Perhaps if you make up an Lguest Drinking Game at this point, it | ||
1386 | * will get you through this section. Or, maybe not. | ||
1387 | * | ||
1388 | * The Launcher binary sits up high, usually starting at address 0xB8000000. | ||
1389 | * Everything below this is the "physical" memory for the Guest. For example, | ||
1390 | * if the Guest were to write a "1" at physical address 0, we would see a "1" | ||
1391 | * in the Launcher at "(int *)0". Guest physical == Launcher virtual. | ||
1392 | * | ||
1393 | * This can be tough to get your head around, but usually it just means that we | ||
1394 | * don't need to do any conversion when the Guest gives us it's "physical" | ||
1395 | * addresses. | ||
1396 | */ | ||
914 | int main(int argc, char *argv[]) | 1397 | int main(int argc, char *argv[]) |
915 | { | 1398 | { |
1399 | /* Memory, top-level pagetable, code startpoint, PAGE_OFFSET and size | ||
1400 | * of the (optional) initrd. */ | ||
916 | unsigned long mem = 0, pgdir, start, page_offset, initrd_size = 0; | 1401 | unsigned long mem = 0, pgdir, start, page_offset, initrd_size = 0; |
1402 | /* A temporary and the /dev/lguest file descriptor. */ | ||
917 | int i, c, lguest_fd; | 1403 | int i, c, lguest_fd; |
1404 | /* The list of Guest devices, based on command line arguments. */ | ||
918 | struct device_list device_list; | 1405 | struct device_list device_list; |
1406 | /* The boot information for the Guest: at guest-physical address 0. */ | ||
919 | void *boot = (void *)0; | 1407 | void *boot = (void *)0; |
1408 | /* If they specify an initrd file to load. */ | ||
920 | const char *initrd_name = NULL; | 1409 | const char *initrd_name = NULL; |
921 | 1410 | ||
1411 | /* First we initialize the device list. Since console and network | ||
1412 | * device receive input from a file descriptor, we keep an fdset | ||
1413 | * (infds) and the maximum fd number (max_infd) with the head of the | ||
1414 | * list. We also keep a pointer to the last device, for easy appending | ||
1415 | * to the list. */ | ||
922 | device_list.max_infd = -1; | 1416 | device_list.max_infd = -1; |
923 | device_list.dev = NULL; | 1417 | device_list.dev = NULL; |
924 | device_list.lastdev = &device_list.dev; | 1418 | device_list.lastdev = &device_list.dev; |
925 | FD_ZERO(&device_list.infds); | 1419 | FD_ZERO(&device_list.infds); |
926 | 1420 | ||
927 | /* We need to know how much memory so we can allocate devices. */ | 1421 | /* We need to know how much memory so we can set up the device |
1422 | * descriptor and memory pages for the devices as we parse the command | ||
1423 | * line. So we quickly look through the arguments to find the amount | ||
1424 | * of memory now. */ | ||
928 | for (i = 1; i < argc; i++) { | 1425 | for (i = 1; i < argc; i++) { |
929 | if (argv[i][0] != '-') { | 1426 | if (argv[i][0] != '-') { |
930 | mem = top = atoi(argv[i]) * 1024 * 1024; | 1427 | mem = top = atoi(argv[i]) * 1024 * 1024; |
@@ -933,6 +1430,8 @@ int main(int argc, char *argv[]) | |||
933 | break; | 1430 | break; |
934 | } | 1431 | } |
935 | } | 1432 | } |
1433 | |||
1434 | /* The options are fairly straight-forward */ | ||
936 | while ((c = getopt_long(argc, argv, "v", opts, NULL)) != EOF) { | 1435 | while ((c = getopt_long(argc, argv, "v", opts, NULL)) != EOF) { |
937 | switch (c) { | 1436 | switch (c) { |
938 | case 'v': | 1437 | case 'v': |
@@ -955,42 +1454,71 @@ int main(int argc, char *argv[]) | |||
955 | usage(); | 1454 | usage(); |
956 | } | 1455 | } |
957 | } | 1456 | } |
1457 | /* After the other arguments we expect memory and kernel image name, | ||
1458 | * followed by command line arguments for the kernel. */ | ||
958 | if (optind + 2 > argc) | 1459 | if (optind + 2 > argc) |
959 | usage(); | 1460 | usage(); |
960 | 1461 | ||
961 | /* We need a console device */ | 1462 | /* We always have a console device */ |
962 | setup_console(&device_list); | 1463 | setup_console(&device_list); |
963 | 1464 | ||
964 | /* First we map /dev/zero over all of guest-physical memory. */ | 1465 | /* We start by mapping anonymous pages over all of guest-physical |
1466 | * memory range. This fills it with 0, and ensures that the Guest | ||
1467 | * won't be killed when it tries to access it. */ | ||
965 | map_zeroed_pages(0, mem / getpagesize()); | 1468 | map_zeroed_pages(0, mem / getpagesize()); |
966 | 1469 | ||
967 | /* Now we load the kernel */ | 1470 | /* Now we load the kernel */ |
968 | start = load_kernel(open_or_die(argv[optind+1], O_RDONLY), | 1471 | start = load_kernel(open_or_die(argv[optind+1], O_RDONLY), |
969 | &page_offset); | 1472 | &page_offset); |
970 | 1473 | ||
971 | /* Map the initrd image if requested */ | 1474 | /* Map the initrd image if requested (at top of physical memory) */ |
972 | if (initrd_name) { | 1475 | if (initrd_name) { |
973 | initrd_size = load_initrd(initrd_name, mem); | 1476 | initrd_size = load_initrd(initrd_name, mem); |
1477 | /* These are the location in the Linux boot header where the | ||
1478 | * start and size of the initrd are expected to be found. */ | ||
974 | *(unsigned long *)(boot+0x218) = mem - initrd_size; | 1479 | *(unsigned long *)(boot+0x218) = mem - initrd_size; |
975 | *(unsigned long *)(boot+0x21c) = initrd_size; | 1480 | *(unsigned long *)(boot+0x21c) = initrd_size; |
1481 | /* The bootloader type 0xFF means "unknown"; that's OK. */ | ||
976 | *(unsigned char *)(boot+0x210) = 0xFF; | 1482 | *(unsigned char *)(boot+0x210) = 0xFF; |
977 | } | 1483 | } |
978 | 1484 | ||
979 | /* Set up the initial linar pagetables. */ | 1485 | /* Set up the initial linear pagetables, starting below the initrd. */ |
980 | pgdir = setup_pagetables(mem, initrd_size, page_offset); | 1486 | pgdir = setup_pagetables(mem, initrd_size, page_offset); |
981 | 1487 | ||
982 | /* E820 memory map: ours is a simple, single region. */ | 1488 | /* The Linux boot header contains an "E820" memory map: ours is a |
1489 | * simple, single region. */ | ||
983 | *(char*)(boot+E820NR) = 1; | 1490 | *(char*)(boot+E820NR) = 1; |
984 | *((struct e820entry *)(boot+E820MAP)) | 1491 | *((struct e820entry *)(boot+E820MAP)) |
985 | = ((struct e820entry) { 0, mem, E820_RAM }); | 1492 | = ((struct e820entry) { 0, mem, E820_RAM }); |
986 | /* Command line pointer and command line (at 4096) */ | 1493 | /* The boot header contains a command line pointer: we put the command |
1494 | * line after the boot header (at address 4096) */ | ||
987 | *(void **)(boot + 0x228) = boot + 4096; | 1495 | *(void **)(boot + 0x228) = boot + 4096; |
988 | concat(boot + 4096, argv+optind+2); | 1496 | concat(boot + 4096, argv+optind+2); |
989 | /* Paravirt type: 1 == lguest */ | 1497 | |
1498 | /* The guest type value of "1" tells the Guest it's under lguest. */ | ||
990 | *(int *)(boot + 0x23c) = 1; | 1499 | *(int *)(boot + 0x23c) = 1; |
991 | 1500 | ||
1501 | /* We tell the kernel to initialize the Guest: this returns the open | ||
1502 | * /dev/lguest file descriptor. */ | ||
992 | lguest_fd = tell_kernel(pgdir, start, page_offset); | 1503 | lguest_fd = tell_kernel(pgdir, start, page_offset); |
1504 | |||
1505 | /* We fork off a child process, which wakes the Launcher whenever one | ||
1506 | * of the input file descriptors needs attention. Otherwise we would | ||
1507 | * run the Guest until it tries to output something. */ | ||
993 | waker_fd = setup_waker(lguest_fd, &device_list); | 1508 | waker_fd = setup_waker(lguest_fd, &device_list); |
994 | 1509 | ||
1510 | /* Finally, run the Guest. This doesn't return. */ | ||
995 | run_guest(lguest_fd, &device_list); | 1511 | run_guest(lguest_fd, &device_list); |
996 | } | 1512 | } |
1513 | /*:*/ | ||
1514 | |||
1515 | /*M:999 | ||
1516 | * Mastery is done: you now know everything I do. | ||
1517 | * | ||
1518 | * But surely you have seen code, features and bugs in your wanderings which | ||
1519 | * you now yearn to attack? That is the real game, and I look forward to you | ||
1520 | * patching and forking lguest into the Your-Name-Here-visor. | ||
1521 | * | ||
1522 | * Farewell, and good coding! | ||
1523 | * Rusty Russell. | ||
1524 | */ | ||
diff --git a/Documentation/sched-stats.txt b/Documentation/sched-stats.txt index 6f72021aae51..442e14d35dea 100644 --- a/Documentation/sched-stats.txt +++ b/Documentation/sched-stats.txt | |||
@@ -1,10 +1,11 @@ | |||
1 | Version 10 of schedstats includes support for sched_domains, which | 1 | Version 14 of schedstats includes support for sched_domains, which hit the |
2 | hit the mainline kernel in 2.6.7. Some counters make more sense to be | 2 | mainline kernel in 2.6.20 although it is identical to the stats from version |
3 | per-runqueue; other to be per-domain. Note that domains (and their associated | 3 | 12 which was in the kernel from 2.6.13-2.6.19 (version 13 never saw a kernel |
4 | information) will only be pertinent and available on machines utilizing | 4 | release). Some counters make more sense to be per-runqueue; other to be |
5 | CONFIG_SMP. | 5 | per-domain. Note that domains (and their associated information) will only |
6 | 6 | be pertinent and available on machines utilizing CONFIG_SMP. | |
7 | In version 10 of schedstat, there is at least one level of domain | 7 | |
8 | In version 14 of schedstat, there is at least one level of domain | ||
8 | statistics for each cpu listed, and there may well be more than one | 9 | statistics for each cpu listed, and there may well be more than one |
9 | domain. Domains have no particular names in this implementation, but | 10 | domain. Domains have no particular names in this implementation, but |
10 | the highest numbered one typically arbitrates balancing across all the | 11 | the highest numbered one typically arbitrates balancing across all the |
@@ -27,7 +28,7 @@ to write their own scripts, the fields are described here. | |||
27 | 28 | ||
28 | CPU statistics | 29 | CPU statistics |
29 | -------------- | 30 | -------------- |
30 | cpu<N> 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 | 31 | cpu<N> 1 2 3 4 5 6 7 8 9 10 11 12 |
31 | 32 | ||
32 | NOTE: In the sched_yield() statistics, the active queue is considered empty | 33 | NOTE: In the sched_yield() statistics, the active queue is considered empty |
33 | if it has only one process in it, since obviously the process calling | 34 | if it has only one process in it, since obviously the process calling |
@@ -39,48 +40,20 @@ First four fields are sched_yield() statistics: | |||
39 | 3) # of times just the expired queue was empty | 40 | 3) # of times just the expired queue was empty |
40 | 4) # of times sched_yield() was called | 41 | 4) # of times sched_yield() was called |
41 | 42 | ||
42 | Next four are schedule() statistics: | 43 | Next three are schedule() statistics: |
43 | 5) # of times the active queue had at least one other process on it | 44 | 5) # of times we switched to the expired queue and reused it |
44 | 6) # of times we switched to the expired queue and reused it | 45 | 6) # of times schedule() was called |
45 | 7) # of times schedule() was called | 46 | 7) # of times schedule() left the processor idle |
46 | 8) # of times schedule() left the processor idle | ||
47 | |||
48 | Next four are active_load_balance() statistics: | ||
49 | 9) # of times active_load_balance() was called | ||
50 | 10) # of times active_load_balance() caused this cpu to gain a task | ||
51 | 11) # of times active_load_balance() caused this cpu to lose a task | ||
52 | 12) # of times active_load_balance() tried to move a task and failed | ||
53 | |||
54 | Next three are try_to_wake_up() statistics: | ||
55 | 13) # of times try_to_wake_up() was called | ||
56 | 14) # of times try_to_wake_up() successfully moved the awakening task | ||
57 | 15) # of times try_to_wake_up() attempted to move the awakening task | ||
58 | |||
59 | Next two are wake_up_new_task() statistics: | ||
60 | 16) # of times wake_up_new_task() was called | ||
61 | 17) # of times wake_up_new_task() successfully moved the new task | ||
62 | |||
63 | Next one is a sched_migrate_task() statistic: | ||
64 | 18) # of times sched_migrate_task() was called | ||
65 | 47 | ||
66 | Next one is a sched_balance_exec() statistic: | 48 | Next two are try_to_wake_up() statistics: |
67 | 19) # of times sched_balance_exec() was called | 49 | 8) # of times try_to_wake_up() was called |
50 | 9) # of times try_to_wake_up() was called to wake up the local cpu | ||
68 | 51 | ||
69 | Next three are statistics describing scheduling latency: | 52 | Next three are statistics describing scheduling latency: |
70 | 20) sum of all time spent running by tasks on this processor (in ms) | 53 | 10) sum of all time spent running by tasks on this processor (in jiffies) |
71 | 21) sum of all time spent waiting to run by tasks on this processor (in ms) | 54 | 11) sum of all time spent waiting to run by tasks on this processor (in |
72 | 22) # of tasks (not necessarily unique) given to the processor | 55 | jiffies) |
73 | 56 | 12) # of timeslices run on this cpu | |
74 | The last six are statistics dealing with pull_task(): | ||
75 | 23) # of times pull_task() moved a task to this cpu when newly idle | ||
76 | 24) # of times pull_task() stole a task from this cpu when another cpu | ||
77 | was newly idle | ||
78 | 25) # of times pull_task() moved a task to this cpu when idle | ||
79 | 26) # of times pull_task() stole a task from this cpu when another cpu | ||
80 | was idle | ||
81 | 27) # of times pull_task() moved a task to this cpu when busy | ||
82 | 28) # of times pull_task() stole a task from this cpu when another cpu | ||
83 | was busy | ||
84 | 57 | ||
85 | 58 | ||
86 | Domain statistics | 59 | Domain statistics |
@@ -89,65 +62,95 @@ One of these is produced per domain for each cpu described. (Note that if | |||
89 | CONFIG_SMP is not defined, *no* domains are utilized and these lines | 62 | CONFIG_SMP is not defined, *no* domains are utilized and these lines |
90 | will not appear in the output.) | 63 | will not appear in the output.) |
91 | 64 | ||
92 | domain<N> 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 | 65 | domain<N> <cpumask> 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 |
93 | 66 | ||
94 | The first field is a bit mask indicating what cpus this domain operates over. | 67 | The first field is a bit mask indicating what cpus this domain operates over. |
95 | 68 | ||
96 | The next fifteen are a variety of load_balance() statistics: | 69 | The next 24 are a variety of load_balance() statistics in grouped into types |
97 | 70 | of idleness (idle, busy, and newly idle): | |
98 | 1) # of times in this domain load_balance() was called when the cpu | 71 | |
99 | was idle | 72 | 1) # of times in this domain load_balance() was called when the |
100 | 2) # of times in this domain load_balance() was called when the cpu | 73 | cpu was idle |
101 | was busy | 74 | 2) # of times in this domain load_balance() checked but found |
102 | 3) # of times in this domain load_balance() was called when the cpu | 75 | the load did not require balancing when the cpu was idle |
103 | was just becoming idle | 76 | 3) # of times in this domain load_balance() tried to move one or |
104 | 4) # of times in this domain load_balance() tried to move one or more | 77 | more tasks and failed, when the cpu was idle |
105 | tasks and failed, when the cpu was idle | 78 | 4) sum of imbalances discovered (if any) with each call to |
106 | 5) # of times in this domain load_balance() tried to move one or more | 79 | load_balance() in this domain when the cpu was idle |
107 | tasks and failed, when the cpu was busy | 80 | 5) # of times in this domain pull_task() was called when the cpu |
108 | 6) # of times in this domain load_balance() tried to move one or more | 81 | was idle |
109 | tasks and failed, when the cpu was just becoming idle | 82 | 6) # of times in this domain pull_task() was called even though |
110 | 7) sum of imbalances discovered (if any) with each call to | 83 | the target task was cache-hot when idle |
111 | load_balance() in this domain when the cpu was idle | 84 | 7) # of times in this domain load_balance() was called but did |
112 | 8) sum of imbalances discovered (if any) with each call to | 85 | not find a busier queue while the cpu was idle |
113 | load_balance() in this domain when the cpu was busy | 86 | 8) # of times in this domain a busier queue was found while the |
114 | 9) sum of imbalances discovered (if any) with each call to | 87 | cpu was idle but no busier group was found |
115 | load_balance() in this domain when the cpu was just becoming idle | 88 | |
116 | 10) # of times in this domain load_balance() was called but did not find | 89 | 9) # of times in this domain load_balance() was called when the |
117 | a busier queue while the cpu was idle | 90 | cpu was busy |
118 | 11) # of times in this domain load_balance() was called but did not find | 91 | 10) # of times in this domain load_balance() checked but found the |
119 | a busier queue while the cpu was busy | 92 | load did not require balancing when busy |
120 | 12) # of times in this domain load_balance() was called but did not find | 93 | 11) # of times in this domain load_balance() tried to move one or |
121 | a busier queue while the cpu was just becoming idle | 94 | more tasks and failed, when the cpu was busy |
122 | 13) # of times in this domain a busier queue was found while the cpu was | 95 | 12) sum of imbalances discovered (if any) with each call to |
123 | idle but no busier group was found | 96 | load_balance() in this domain when the cpu was busy |
124 | 14) # of times in this domain a busier queue was found while the cpu was | 97 | 13) # of times in this domain pull_task() was called when busy |
125 | busy but no busier group was found | 98 | 14) # of times in this domain pull_task() was called even though the |
126 | 15) # of times in this domain a busier queue was found while the cpu was | 99 | target task was cache-hot when busy |
127 | just becoming idle but no busier group was found | 100 | 15) # of times in this domain load_balance() was called but did not |
128 | 101 | find a busier queue while the cpu was busy | |
129 | Next two are sched_balance_exec() statistics: | 102 | 16) # of times in this domain a busier queue was found while the cpu |
130 | 17) # of times in this domain sched_balance_exec() successfully pushed | 103 | was busy but no busier group was found |
131 | a task to a new cpu | 104 | |
132 | 18) # of times in this domain sched_balance_exec() tried but failed to | 105 | 17) # of times in this domain load_balance() was called when the |
133 | push a task to a new cpu | 106 | cpu was just becoming idle |
134 | 107 | 18) # of times in this domain load_balance() checked but found the | |
135 | Next two are try_to_wake_up() statistics: | 108 | load did not require balancing when the cpu was just becoming idle |
136 | 19) # of times in this domain try_to_wake_up() tried to move a task based | 109 | 19) # of times in this domain load_balance() tried to move one or more |
137 | on affinity and cache warmth | 110 | tasks and failed, when the cpu was just becoming idle |
138 | 20) # of times in this domain try_to_wake_up() tried to move a task based | 111 | 20) sum of imbalances discovered (if any) with each call to |
139 | on load balancing | 112 | load_balance() in this domain when the cpu was just becoming idle |
140 | 113 | 21) # of times in this domain pull_task() was called when newly idle | |
114 | 22) # of times in this domain pull_task() was called even though the | ||
115 | target task was cache-hot when just becoming idle | ||
116 | 23) # of times in this domain load_balance() was called but did not | ||
117 | find a busier queue while the cpu was just becoming idle | ||
118 | 24) # of times in this domain a busier queue was found while the cpu | ||
119 | was just becoming idle but no busier group was found | ||
120 | |||
121 | Next three are active_load_balance() statistics: | ||
122 | 25) # of times active_load_balance() was called | ||
123 | 26) # of times active_load_balance() tried to move a task and failed | ||
124 | 27) # of times active_load_balance() successfully moved a task | ||
125 | |||
126 | Next three are sched_balance_exec() statistics: | ||
127 | 28) sbe_cnt is not used | ||
128 | 29) sbe_balanced is not used | ||
129 | 30) sbe_pushed is not used | ||
130 | |||
131 | Next three are sched_balance_fork() statistics: | ||
132 | 31) sbf_cnt is not used | ||
133 | 32) sbf_balanced is not used | ||
134 | 33) sbf_pushed is not used | ||
135 | |||
136 | Next three are try_to_wake_up() statistics: | ||
137 | 34) # of times in this domain try_to_wake_up() awoke a task that | ||
138 | last ran on a different cpu in this domain | ||
139 | 35) # of times in this domain try_to_wake_up() moved a task to the | ||
140 | waking cpu because it was cache-cold on its own cpu anyway | ||
141 | 36) # of times in this domain try_to_wake_up() started passive balancing | ||
141 | 142 | ||
142 | /proc/<pid>/schedstat | 143 | /proc/<pid>/schedstat |
143 | ---------------- | 144 | ---------------- |
144 | schedstats also adds a new /proc/<pid/schedstat file to include some of | 145 | schedstats also adds a new /proc/<pid/schedstat file to include some of |
145 | the same information on a per-process level. There are three fields in | 146 | the same information on a per-process level. There are three fields in |
146 | this file correlating to fields 20, 21, and 22 in the CPU fields, but | 147 | this file correlating for that process to: |
147 | they only apply for that process. | 148 | 1) time spent on the cpu |
149 | 2) time spent waiting on a runqueue | ||
150 | 3) # of timeslices run on this cpu | ||
148 | 151 | ||
149 | A program could be easily written to make use of these extra fields to | 152 | A program could be easily written to make use of these extra fields to |
150 | report on how well a particular process or set of processes is faring | 153 | report on how well a particular process or set of processes is faring |
151 | under the scheduler's policies. A simple version of such a program is | 154 | under the scheduler's policies. A simple version of such a program is |
152 | available at | 155 | available at |
153 | http://eaglet.rain.com/rick/linux/schedstat/v10/latency.c | 156 | http://eaglet.rain.com/rick/linux/schedstat/v12/latency.c |
diff --git a/MAINTAINERS b/MAINTAINERS index 01f222e51871..babd00b0c65c 100644 --- a/MAINTAINERS +++ b/MAINTAINERS | |||
@@ -3339,6 +3339,14 @@ M: thomas@winischhofer.net | |||
3339 | W: http://www.winischhofer.at/linuxsisusbvga.shtml | 3339 | W: http://www.winischhofer.at/linuxsisusbvga.shtml |
3340 | S: Maintained | 3340 | S: Maintained |
3341 | 3341 | ||
3342 | SLAB ALLOCATOR | ||
3343 | P: Christoph Lameter | ||
3344 | M: clameter@sgi.com | ||
3345 | P: Pekka Enberg | ||
3346 | M: penberg@cs.helsinki.fi | ||
3347 | L: linux-mm@kvack.org | ||
3348 | S: Maintained | ||
3349 | |||
3342 | SMC91x ETHERNET DRIVER | 3350 | SMC91x ETHERNET DRIVER |
3343 | P: Nicolas Pitre | 3351 | P: Nicolas Pitre |
3344 | M: nico@cam.org | 3352 | M: nico@cam.org |
@@ -299,7 +299,7 @@ CHECKFLAGS := -D__linux__ -Dlinux -D__STDC__ -Dunix -D__unix__ -Wbitwise $(C | |||
299 | MODFLAGS = -DMODULE | 299 | MODFLAGS = -DMODULE |
300 | CFLAGS_MODULE = $(MODFLAGS) | 300 | CFLAGS_MODULE = $(MODFLAGS) |
301 | AFLAGS_MODULE = $(MODFLAGS) | 301 | AFLAGS_MODULE = $(MODFLAGS) |
302 | LDFLAGS_MODULE = -r | 302 | LDFLAGS_MODULE = |
303 | CFLAGS_KERNEL = | 303 | CFLAGS_KERNEL = |
304 | AFLAGS_KERNEL = | 304 | AFLAGS_KERNEL = |
305 | 305 | ||
diff --git a/arch/alpha/kernel/head.S b/arch/alpha/kernel/head.S index e27d23c74ba8..7ac1f1372c36 100644 --- a/arch/alpha/kernel/head.S +++ b/arch/alpha/kernel/head.S | |||
@@ -10,6 +10,7 @@ | |||
10 | #include <asm/system.h> | 10 | #include <asm/system.h> |
11 | #include <asm/asm-offsets.h> | 11 | #include <asm/asm-offsets.h> |
12 | 12 | ||
13 | .section .text.head, "ax" | ||
13 | .globl swapper_pg_dir | 14 | .globl swapper_pg_dir |
14 | .globl _stext | 15 | .globl _stext |
15 | swapper_pg_dir=SWAPPER_PGD | 16 | swapper_pg_dir=SWAPPER_PGD |
diff --git a/arch/alpha/kernel/pci.c b/arch/alpha/kernel/pci.c index ab642a4f08de..9dc1cee43265 100644 --- a/arch/alpha/kernel/pci.c +++ b/arch/alpha/kernel/pci.c | |||
@@ -195,7 +195,7 @@ pcibios_init(void) | |||
195 | 195 | ||
196 | subsys_initcall(pcibios_init); | 196 | subsys_initcall(pcibios_init); |
197 | 197 | ||
198 | char * __init | 198 | char * __devinit |
199 | pcibios_setup(char *str) | 199 | pcibios_setup(char *str) |
200 | { | 200 | { |
201 | return str; | 201 | return str; |
@@ -204,7 +204,7 @@ pcibios_setup(char *str) | |||
204 | #ifdef ALPHA_RESTORE_SRM_SETUP | 204 | #ifdef ALPHA_RESTORE_SRM_SETUP |
205 | static struct pdev_srm_saved_conf *srm_saved_configs; | 205 | static struct pdev_srm_saved_conf *srm_saved_configs; |
206 | 206 | ||
207 | void __init | 207 | void __devinit |
208 | pdev_save_srm_config(struct pci_dev *dev) | 208 | pdev_save_srm_config(struct pci_dev *dev) |
209 | { | 209 | { |
210 | struct pdev_srm_saved_conf *tmp; | 210 | struct pdev_srm_saved_conf *tmp; |
@@ -247,14 +247,14 @@ pci_restore_srm_config(void) | |||
247 | } | 247 | } |
248 | #endif | 248 | #endif |
249 | 249 | ||
250 | void __init | 250 | void __devinit |
251 | pcibios_fixup_resource(struct resource *res, struct resource *root) | 251 | pcibios_fixup_resource(struct resource *res, struct resource *root) |
252 | { | 252 | { |
253 | res->start += root->start; | 253 | res->start += root->start; |
254 | res->end += root->start; | 254 | res->end += root->start; |
255 | } | 255 | } |
256 | 256 | ||
257 | void __init | 257 | void __devinit |
258 | pcibios_fixup_device_resources(struct pci_dev *dev, struct pci_bus *bus) | 258 | pcibios_fixup_device_resources(struct pci_dev *dev, struct pci_bus *bus) |
259 | { | 259 | { |
260 | /* Update device resources. */ | 260 | /* Update device resources. */ |
@@ -273,7 +273,7 @@ pcibios_fixup_device_resources(struct pci_dev *dev, struct pci_bus *bus) | |||
273 | } | 273 | } |
274 | } | 274 | } |
275 | 275 | ||
276 | void __init | 276 | void __devinit |
277 | pcibios_fixup_bus(struct pci_bus *bus) | 277 | pcibios_fixup_bus(struct pci_bus *bus) |
278 | { | 278 | { |
279 | /* Propagate hose info into the subordinate devices. */ | 279 | /* Propagate hose info into the subordinate devices. */ |
diff --git a/arch/alpha/kernel/pci_iommu.c b/arch/alpha/kernel/pci_iommu.c index 6b07f89a72c7..e1c470752ebc 100644 --- a/arch/alpha/kernel/pci_iommu.c +++ b/arch/alpha/kernel/pci_iommu.c | |||
@@ -58,7 +58,7 @@ size_for_memory(unsigned long max) | |||
58 | return max; | 58 | return max; |
59 | } | 59 | } |
60 | 60 | ||
61 | struct pci_iommu_arena * | 61 | struct pci_iommu_arena * __init |
62 | iommu_arena_new_node(int nid, struct pci_controller *hose, dma_addr_t base, | 62 | iommu_arena_new_node(int nid, struct pci_controller *hose, dma_addr_t base, |
63 | unsigned long window_size, unsigned long align) | 63 | unsigned long window_size, unsigned long align) |
64 | { | 64 | { |
@@ -117,7 +117,7 @@ iommu_arena_new_node(int nid, struct pci_controller *hose, dma_addr_t base, | |||
117 | return arena; | 117 | return arena; |
118 | } | 118 | } |
119 | 119 | ||
120 | struct pci_iommu_arena * | 120 | struct pci_iommu_arena * __init |
121 | iommu_arena_new(struct pci_controller *hose, dma_addr_t base, | 121 | iommu_arena_new(struct pci_controller *hose, dma_addr_t base, |
122 | unsigned long window_size, unsigned long align) | 122 | unsigned long window_size, unsigned long align) |
123 | { | 123 | { |
diff --git a/arch/alpha/kernel/smp.c b/arch/alpha/kernel/smp.c index b28731437c31..0804b6abe203 100644 --- a/arch/alpha/kernel/smp.c +++ b/arch/alpha/kernel/smp.c | |||
@@ -358,7 +358,7 @@ secondary_cpu_start(int cpuid, struct task_struct *idle) | |||
358 | /* | 358 | /* |
359 | * Bring one cpu online. | 359 | * Bring one cpu online. |
360 | */ | 360 | */ |
361 | static int __devinit | 361 | static int __cpuinit |
362 | smp_boot_one_cpu(int cpuid) | 362 | smp_boot_one_cpu(int cpuid) |
363 | { | 363 | { |
364 | struct task_struct *idle; | 364 | struct task_struct *idle; |
@@ -487,7 +487,7 @@ smp_prepare_boot_cpu(void) | |||
487 | { | 487 | { |
488 | } | 488 | } |
489 | 489 | ||
490 | int __devinit | 490 | int __cpuinit |
491 | __cpu_up(unsigned int cpu) | 491 | __cpu_up(unsigned int cpu) |
492 | { | 492 | { |
493 | smp_boot_one_cpu(cpu); | 493 | smp_boot_one_cpu(cpu); |
@@ -541,7 +541,7 @@ smp_percpu_timer_interrupt(struct pt_regs *regs) | |||
541 | set_irq_regs(old_regs); | 541 | set_irq_regs(old_regs); |
542 | } | 542 | } |
543 | 543 | ||
544 | int __init | 544 | int |
545 | setup_profiling_timer(unsigned int multiplier) | 545 | setup_profiling_timer(unsigned int multiplier) |
546 | { | 546 | { |
547 | return -EINVAL; | 547 | return -EINVAL; |
diff --git a/arch/alpha/kernel/vmlinux.lds.S b/arch/alpha/kernel/vmlinux.lds.S index fe13daa5cb2c..7af07d3ad5f0 100644 --- a/arch/alpha/kernel/vmlinux.lds.S +++ b/arch/alpha/kernel/vmlinux.lds.S | |||
@@ -15,6 +15,7 @@ SECTIONS | |||
15 | 15 | ||
16 | _text = .; /* Text and read-only data */ | 16 | _text = .; /* Text and read-only data */ |
17 | .text : { | 17 | .text : { |
18 | *(.text.head) | ||
18 | TEXT_TEXT | 19 | TEXT_TEXT |
19 | SCHED_TEXT | 20 | SCHED_TEXT |
20 | LOCK_TEXT | 21 | LOCK_TEXT |
diff --git a/arch/arm/plat-omap/mailbox.c b/arch/arm/plat-omap/mailbox.c index de7e6ef48bd0..0360b1f14d11 100644 --- a/arch/arm/plat-omap/mailbox.c +++ b/arch/arm/plat-omap/mailbox.c | |||
@@ -161,11 +161,11 @@ static void mbox_rx_work(struct work_struct *work) | |||
161 | /* | 161 | /* |
162 | * Mailbox interrupt handler | 162 | * Mailbox interrupt handler |
163 | */ | 163 | */ |
164 | static void mbox_txq_fn(request_queue_t * q) | 164 | static void mbox_txq_fn(struct request_queue * q) |
165 | { | 165 | { |
166 | } | 166 | } |
167 | 167 | ||
168 | static void mbox_rxq_fn(request_queue_t * q) | 168 | static void mbox_rxq_fn(struct request_queue * q) |
169 | { | 169 | { |
170 | } | 170 | } |
171 | 171 | ||
@@ -180,7 +180,7 @@ static void __mbox_rx_interrupt(struct omap_mbox *mbox) | |||
180 | { | 180 | { |
181 | struct request *rq; | 181 | struct request *rq; |
182 | mbox_msg_t msg; | 182 | mbox_msg_t msg; |
183 | request_queue_t *q = mbox->rxq->queue; | 183 | struct request_queue *q = mbox->rxq->queue; |
184 | 184 | ||
185 | disable_mbox_irq(mbox, IRQ_RX); | 185 | disable_mbox_irq(mbox, IRQ_RX); |
186 | 186 | ||
@@ -297,7 +297,7 @@ static struct omap_mbox_queue *mbox_queue_alloc(struct omap_mbox *mbox, | |||
297 | request_fn_proc * proc, | 297 | request_fn_proc * proc, |
298 | void (*work) (struct work_struct *)) | 298 | void (*work) (struct work_struct *)) |
299 | { | 299 | { |
300 | request_queue_t *q; | 300 | struct request_queue *q; |
301 | struct omap_mbox_queue *mq; | 301 | struct omap_mbox_queue *mq; |
302 | 302 | ||
303 | mq = kzalloc(sizeof(struct omap_mbox_queue), GFP_KERNEL); | 303 | mq = kzalloc(sizeof(struct omap_mbox_queue), GFP_KERNEL); |
diff --git a/arch/i386/boot/apm.c b/arch/i386/boot/apm.c index a34087c370c0..eab50c55a3a5 100644 --- a/arch/i386/boot/apm.c +++ b/arch/i386/boot/apm.c | |||
@@ -40,14 +40,15 @@ int query_apm_bios(void) | |||
40 | if (bx != 0x504d) /* "PM" signature */ | 40 | if (bx != 0x504d) /* "PM" signature */ |
41 | return -1; | 41 | return -1; |
42 | 42 | ||
43 | if (cx & 0x02) /* 32 bits supported? */ | 43 | if (!(cx & 0x02)) /* 32 bits supported? */ |
44 | return -1; | 44 | return -1; |
45 | 45 | ||
46 | /* Disconnect first, just in case */ | 46 | /* Disconnect first, just in case */ |
47 | ax = 0x5304; | 47 | ax = 0x5304; |
48 | bx = 0; | ||
48 | asm volatile("pushl %%ebp ; int $0x15 ; popl %%ebp" | 49 | asm volatile("pushl %%ebp ; int $0x15 ; popl %%ebp" |
49 | : "+a" (ax) | 50 | : "+a" (ax), "+b" (bx) |
50 | : : "ebx", "ecx", "edx", "esi", "edi"); | 51 | : : "ecx", "edx", "esi", "edi"); |
51 | 52 | ||
52 | /* Paranoia */ | 53 | /* Paranoia */ |
53 | ebx = esi = 0; | 54 | ebx = esi = 0; |
diff --git a/arch/i386/boot/main.c b/arch/i386/boot/main.c index 7f01f96c4fb8..0eeef3989a17 100644 --- a/arch/i386/boot/main.c +++ b/arch/i386/boot/main.c | |||
@@ -73,15 +73,15 @@ static void keyboard_set_repeat(void) | |||
73 | } | 73 | } |
74 | 74 | ||
75 | /* | 75 | /* |
76 | * Get Intel SpeedStep IST information. | 76 | * Get Intel SpeedStep (IST) information. |
77 | */ | 77 | */ |
78 | static void query_speedstep_ist(void) | 78 | static void query_ist(void) |
79 | { | 79 | { |
80 | asm("int $0x15" | 80 | asm("int $0x15" |
81 | : "=a" (boot_params.speedstep_info[0]), | 81 | : "=a" (boot_params.ist_info.signature), |
82 | "=b" (boot_params.speedstep_info[1]), | 82 | "=b" (boot_params.ist_info.command), |
83 | "=c" (boot_params.speedstep_info[2]), | 83 | "=c" (boot_params.ist_info.event), |
84 | "=d" (boot_params.speedstep_info[3]) | 84 | "=d" (boot_params.ist_info.perf_level) |
85 | : "a" (0x0000e980), /* IST Support */ | 85 | : "a" (0x0000e980), /* IST Support */ |
86 | "d" (0x47534943)); /* Request value */ | 86 | "d" (0x47534943)); /* Request value */ |
87 | } | 87 | } |
@@ -144,8 +144,8 @@ void main(void) | |||
144 | query_voyager(); | 144 | query_voyager(); |
145 | #endif | 145 | #endif |
146 | 146 | ||
147 | /* Query SpeedStep IST information */ | 147 | /* Query Intel SpeedStep (IST) information */ |
148 | query_speedstep_ist(); | 148 | query_ist(); |
149 | 149 | ||
150 | /* Query APM information */ | 150 | /* Query APM information */ |
151 | #if defined(CONFIG_APM) || defined(CONFIG_APM_MODULE) | 151 | #if defined(CONFIG_APM) || defined(CONFIG_APM_MODULE) |
diff --git a/arch/i386/kernel/acpi/Makefile b/arch/i386/kernel/acpi/Makefile index 7f7be01f44e6..223f58fc9f46 100644 --- a/arch/i386/kernel/acpi/Makefile +++ b/arch/i386/kernel/acpi/Makefile | |||
@@ -2,7 +2,7 @@ obj-$(CONFIG_ACPI) += boot.o | |||
2 | ifneq ($(CONFIG_PCI),) | 2 | ifneq ($(CONFIG_PCI),) |
3 | obj-$(CONFIG_X86_IO_APIC) += earlyquirk.o | 3 | obj-$(CONFIG_X86_IO_APIC) += earlyquirk.o |
4 | endif | 4 | endif |
5 | obj-$(CONFIG_ACPI_SLEEP) += sleep.o wakeup.o | 5 | obj-$(CONFIG_ACPI) += sleep.o wakeup.o |
6 | 6 | ||
7 | ifneq ($(CONFIG_ACPI_PROCESSOR),) | 7 | ifneq ($(CONFIG_ACPI_PROCESSOR),) |
8 | obj-y += cstate.o processor.o | 8 | obj-y += cstate.o processor.o |
diff --git a/arch/i386/kernel/alternative.c b/arch/i386/kernel/alternative.c index c3750c2c4113..c85598acb8fd 100644 --- a/arch/i386/kernel/alternative.c +++ b/arch/i386/kernel/alternative.c | |||
@@ -430,22 +430,12 @@ void __init alternative_instructions(void) | |||
430 | * And on the local CPU you need to be protected again NMI or MCE handlers | 430 | * And on the local CPU you need to be protected again NMI or MCE handlers |
431 | * seeing an inconsistent instruction while you patch. | 431 | * seeing an inconsistent instruction while you patch. |
432 | */ | 432 | */ |
433 | void __kprobes text_poke(void *oaddr, unsigned char *opcode, int len) | 433 | void __kprobes text_poke(void *addr, unsigned char *opcode, int len) |
434 | { | 434 | { |
435 | u8 *addr = oaddr; | ||
436 | if (!pte_write(*lookup_address((unsigned long)addr))) { | ||
437 | struct page *p[2] = { virt_to_page(addr), virt_to_page(addr+PAGE_SIZE) }; | ||
438 | addr = vmap(p, 2, VM_MAP, PAGE_KERNEL); | ||
439 | if (!addr) | ||
440 | return; | ||
441 | addr += ((unsigned long)oaddr) % PAGE_SIZE; | ||
442 | } | ||
443 | memcpy(addr, opcode, len); | 435 | memcpy(addr, opcode, len); |
444 | sync_core(); | 436 | sync_core(); |
445 | /* Not strictly needed, but can speed CPU recovery up. Ignore cross cacheline | 437 | /* Not strictly needed, but can speed CPU recovery up. Ignore cross cacheline |
446 | case. */ | 438 | case. */ |
447 | if (cpu_has_clflush) | 439 | if (cpu_has_clflush) |
448 | asm("clflush (%0) " :: "r" (oaddr) : "memory"); | 440 | asm("clflush (%0) " :: "r" (addr) : "memory"); |
449 | if (addr != oaddr) | ||
450 | vunmap(addr); | ||
451 | } | 441 | } |
diff --git a/arch/i386/kernel/cpu/cpufreq/Kconfig b/arch/i386/kernel/cpu/cpufreq/Kconfig index 094118ba00da..d8c6f132dc7a 100644 --- a/arch/i386/kernel/cpu/cpufreq/Kconfig +++ b/arch/i386/kernel/cpu/cpufreq/Kconfig | |||
@@ -92,7 +92,7 @@ config X86_POWERNOW_K8 | |||
92 | config X86_POWERNOW_K8_ACPI | 92 | config X86_POWERNOW_K8_ACPI |
93 | bool "ACPI Support" | 93 | bool "ACPI Support" |
94 | select ACPI_PROCESSOR | 94 | select ACPI_PROCESSOR |
95 | depends on X86_POWERNOW_K8 | 95 | depends on ACPI && X86_POWERNOW_K8 |
96 | default y | 96 | default y |
97 | help | 97 | help |
98 | This provides access to the K8s Processor Performance States via ACPI. | 98 | This provides access to the K8s Processor Performance States via ACPI. |
diff --git a/arch/i386/kernel/setup.c b/arch/i386/kernel/setup.c index d474cd639bcb..7fe5da3c932e 100644 --- a/arch/i386/kernel/setup.c +++ b/arch/i386/kernel/setup.c | |||
@@ -422,7 +422,7 @@ void __init setup_bootmem_allocator(void) | |||
422 | */ | 422 | */ |
423 | reserve_bootmem(PAGE_SIZE, PAGE_SIZE); | 423 | reserve_bootmem(PAGE_SIZE, PAGE_SIZE); |
424 | #endif | 424 | #endif |
425 | #ifdef CONFIG_ACPI_SLEEP | 425 | #ifdef CONFIG_ACPI |
426 | /* | 426 | /* |
427 | * Reserve low memory region for sleep support. | 427 | * Reserve low memory region for sleep support. |
428 | */ | 428 | */ |
diff --git a/arch/i386/mm/init.c b/arch/i386/mm/init.c index c3b9905af2d5..4c4809f13cb1 100644 --- a/arch/i386/mm/init.c +++ b/arch/i386/mm/init.c | |||
@@ -432,7 +432,7 @@ static void __init pagetable_init (void) | |||
432 | paravirt_pagetable_setup_done(pgd_base); | 432 | paravirt_pagetable_setup_done(pgd_base); |
433 | } | 433 | } |
434 | 434 | ||
435 | #if defined(CONFIG_SOFTWARE_SUSPEND) || defined(CONFIG_ACPI_SLEEP) | 435 | #if defined(CONFIG_SOFTWARE_SUSPEND) || defined(CONFIG_ACPI) |
436 | /* | 436 | /* |
437 | * Swap suspend & friends need this for resume because things like the intel-agp | 437 | * Swap suspend & friends need this for resume because things like the intel-agp |
438 | * driver might have split up a kernel 4MB mapping. | 438 | * driver might have split up a kernel 4MB mapping. |
@@ -800,9 +800,17 @@ void mark_rodata_ro(void) | |||
800 | unsigned long start = PFN_ALIGN(_text); | 800 | unsigned long start = PFN_ALIGN(_text); |
801 | unsigned long size = PFN_ALIGN(_etext) - start; | 801 | unsigned long size = PFN_ALIGN(_etext) - start; |
802 | 802 | ||
803 | change_page_attr(virt_to_page(start), | 803 | #ifndef CONFIG_KPROBES |
804 | size >> PAGE_SHIFT, PAGE_KERNEL_RX); | 804 | #ifdef CONFIG_HOTPLUG_CPU |
805 | printk("Write protecting the kernel text: %luk\n", size >> 10); | 805 | /* It must still be possible to apply SMP alternatives. */ |
806 | if (num_possible_cpus() <= 1) | ||
807 | #endif | ||
808 | { | ||
809 | change_page_attr(virt_to_page(start), | ||
810 | size >> PAGE_SHIFT, PAGE_KERNEL_RX); | ||
811 | printk("Write protecting the kernel text: %luk\n", size >> 10); | ||
812 | } | ||
813 | #endif | ||
806 | start += size; | 814 | start += size; |
807 | size = (unsigned long)__end_rodata - start; | 815 | size = (unsigned long)__end_rodata - start; |
808 | change_page_attr(virt_to_page(start), | 816 | change_page_attr(virt_to_page(start), |
diff --git a/arch/ia64/ia32/ia32_support.c b/arch/ia64/ia32/ia32_support.c index e13a1a1db4b5..d1d50cd1c38a 100644 --- a/arch/ia64/ia32/ia32_support.c +++ b/arch/ia64/ia32/ia32_support.c | |||
@@ -249,11 +249,11 @@ ia32_init (void) | |||
249 | 249 | ||
250 | #if PAGE_SHIFT > IA32_PAGE_SHIFT | 250 | #if PAGE_SHIFT > IA32_PAGE_SHIFT |
251 | { | 251 | { |
252 | extern struct kmem_cache *partial_page_cachep; | 252 | extern struct kmem_cache *ia64_partial_page_cachep; |
253 | 253 | ||
254 | partial_page_cachep = kmem_cache_create("partial_page_cache", | 254 | ia64_partial_page_cachep = kmem_cache_create("ia64_partial_page_cache", |
255 | sizeof(struct partial_page), | 255 | sizeof(struct ia64_partial_page), |
256 | 0, SLAB_PANIC, NULL); | 256 | 0, SLAB_PANIC, NULL); |
257 | } | 257 | } |
258 | #endif | 258 | #endif |
259 | return 0; | 259 | return 0; |
diff --git a/arch/ia64/ia32/ia32priv.h b/arch/ia64/ia32/ia32priv.h index cfa0bc0026b5..466bbcb138b2 100644 --- a/arch/ia64/ia32/ia32priv.h +++ b/arch/ia64/ia32/ia32priv.h | |||
@@ -25,8 +25,8 @@ | |||
25 | * partially mapped pages provide precise accounting of which 4k sub pages | 25 | * partially mapped pages provide precise accounting of which 4k sub pages |
26 | * are mapped and which ones are not, thereby improving IA-32 compatibility. | 26 | * are mapped and which ones are not, thereby improving IA-32 compatibility. |
27 | */ | 27 | */ |
28 | struct partial_page { | 28 | struct ia64_partial_page { |
29 | struct partial_page *next; /* linked list, sorted by address */ | 29 | struct ia64_partial_page *next; /* linked list, sorted by address */ |
30 | struct rb_node pp_rb; | 30 | struct rb_node pp_rb; |
31 | /* 64K is the largest "normal" page supported by ia64 ABI. So 4K*64 | 31 | /* 64K is the largest "normal" page supported by ia64 ABI. So 4K*64 |
32 | * should suffice.*/ | 32 | * should suffice.*/ |
@@ -34,17 +34,17 @@ struct partial_page { | |||
34 | unsigned int base; | 34 | unsigned int base; |
35 | }; | 35 | }; |
36 | 36 | ||
37 | struct partial_page_list { | 37 | struct ia64_partial_page_list { |
38 | struct partial_page *pp_head; /* list head, points to the lowest | 38 | struct ia64_partial_page *pp_head; /* list head, points to the lowest |
39 | * addressed partial page */ | 39 | * addressed partial page */ |
40 | struct rb_root ppl_rb; | 40 | struct rb_root ppl_rb; |
41 | struct partial_page *pp_hint; /* pp_hint->next is the last | 41 | struct ia64_partial_page *pp_hint; /* pp_hint->next is the last |
42 | * accessed partial page */ | 42 | * accessed partial page */ |
43 | atomic_t pp_count; /* reference count */ | 43 | atomic_t pp_count; /* reference count */ |
44 | }; | 44 | }; |
45 | 45 | ||
46 | #if PAGE_SHIFT > IA32_PAGE_SHIFT | 46 | #if PAGE_SHIFT > IA32_PAGE_SHIFT |
47 | struct partial_page_list* ia32_init_pp_list (void); | 47 | struct ia64_partial_page_list* ia32_init_pp_list (void); |
48 | #else | 48 | #else |
49 | # define ia32_init_pp_list() 0 | 49 | # define ia32_init_pp_list() 0 |
50 | #endif | 50 | #endif |
diff --git a/arch/ia64/ia32/sys_ia32.c b/arch/ia64/ia32/sys_ia32.c index 0afb4fe7c35b..af10462d44d4 100644 --- a/arch/ia64/ia32/sys_ia32.c +++ b/arch/ia64/ia32/sys_ia32.c | |||
@@ -253,17 +253,17 @@ mmap_subpage (struct file *file, unsigned long start, unsigned long end, int pro | |||
253 | return ret; | 253 | return ret; |
254 | } | 254 | } |
255 | 255 | ||
256 | /* SLAB cache for partial_page structures */ | 256 | /* SLAB cache for ia64_partial_page structures */ |
257 | struct kmem_cache *partial_page_cachep; | 257 | struct kmem_cache *ia64_partial_page_cachep; |
258 | 258 | ||
259 | /* | 259 | /* |
260 | * init partial_page_list. | 260 | * init ia64_partial_page_list. |
261 | * return 0 means kmalloc fail. | 261 | * return 0 means kmalloc fail. |
262 | */ | 262 | */ |
263 | struct partial_page_list* | 263 | struct ia64_partial_page_list* |
264 | ia32_init_pp_list(void) | 264 | ia32_init_pp_list(void) |
265 | { | 265 | { |
266 | struct partial_page_list *p; | 266 | struct ia64_partial_page_list *p; |
267 | 267 | ||
268 | if ((p = kmalloc(sizeof(*p), GFP_KERNEL)) == NULL) | 268 | if ((p = kmalloc(sizeof(*p), GFP_KERNEL)) == NULL) |
269 | return p; | 269 | return p; |
@@ -280,12 +280,12 @@ ia32_init_pp_list(void) | |||
280 | * Else, return 0 and provide @pprev, @rb_link, @rb_parent to | 280 | * Else, return 0 and provide @pprev, @rb_link, @rb_parent to |
281 | * be used by later __ia32_insert_pp(). | 281 | * be used by later __ia32_insert_pp(). |
282 | */ | 282 | */ |
283 | static struct partial_page * | 283 | static struct ia64_partial_page * |
284 | __ia32_find_pp(struct partial_page_list *ppl, unsigned int start, | 284 | __ia32_find_pp(struct ia64_partial_page_list *ppl, unsigned int start, |
285 | struct partial_page **pprev, struct rb_node ***rb_link, | 285 | struct ia64_partial_page **pprev, struct rb_node ***rb_link, |
286 | struct rb_node **rb_parent) | 286 | struct rb_node **rb_parent) |
287 | { | 287 | { |
288 | struct partial_page *pp; | 288 | struct ia64_partial_page *pp; |
289 | struct rb_node **__rb_link, *__rb_parent, *rb_prev; | 289 | struct rb_node **__rb_link, *__rb_parent, *rb_prev; |
290 | 290 | ||
291 | pp = ppl->pp_hint; | 291 | pp = ppl->pp_hint; |
@@ -297,7 +297,7 @@ __ia32_find_pp(struct partial_page_list *ppl, unsigned int start, | |||
297 | 297 | ||
298 | while (*__rb_link) { | 298 | while (*__rb_link) { |
299 | __rb_parent = *__rb_link; | 299 | __rb_parent = *__rb_link; |
300 | pp = rb_entry(__rb_parent, struct partial_page, pp_rb); | 300 | pp = rb_entry(__rb_parent, struct ia64_partial_page, pp_rb); |
301 | 301 | ||
302 | if (pp->base == start) { | 302 | if (pp->base == start) { |
303 | ppl->pp_hint = pp; | 303 | ppl->pp_hint = pp; |
@@ -314,7 +314,7 @@ __ia32_find_pp(struct partial_page_list *ppl, unsigned int start, | |||
314 | *rb_parent = __rb_parent; | 314 | *rb_parent = __rb_parent; |
315 | *pprev = NULL; | 315 | *pprev = NULL; |
316 | if (rb_prev) | 316 | if (rb_prev) |
317 | *pprev = rb_entry(rb_prev, struct partial_page, pp_rb); | 317 | *pprev = rb_entry(rb_prev, struct ia64_partial_page, pp_rb); |
318 | return NULL; | 318 | return NULL; |
319 | } | 319 | } |
320 | 320 | ||
@@ -322,9 +322,9 @@ __ia32_find_pp(struct partial_page_list *ppl, unsigned int start, | |||
322 | * insert @pp into @ppl. | 322 | * insert @pp into @ppl. |
323 | */ | 323 | */ |
324 | static void | 324 | static void |
325 | __ia32_insert_pp(struct partial_page_list *ppl, struct partial_page *pp, | 325 | __ia32_insert_pp(struct ia64_partial_page_list *ppl, |
326 | struct partial_page *prev, struct rb_node **rb_link, | 326 | struct ia64_partial_page *pp, struct ia64_partial_page *prev, |
327 | struct rb_node *rb_parent) | 327 | struct rb_node **rb_link, struct rb_node *rb_parent) |
328 | { | 328 | { |
329 | /* link list */ | 329 | /* link list */ |
330 | if (prev) { | 330 | if (prev) { |
@@ -334,7 +334,7 @@ __ia32_insert_pp(struct partial_page_list *ppl, struct partial_page *pp, | |||
334 | ppl->pp_head = pp; | 334 | ppl->pp_head = pp; |
335 | if (rb_parent) | 335 | if (rb_parent) |
336 | pp->next = rb_entry(rb_parent, | 336 | pp->next = rb_entry(rb_parent, |
337 | struct partial_page, pp_rb); | 337 | struct ia64_partial_page, pp_rb); |
338 | else | 338 | else |
339 | pp->next = NULL; | 339 | pp->next = NULL; |
340 | } | 340 | } |
@@ -350,8 +350,8 @@ __ia32_insert_pp(struct partial_page_list *ppl, struct partial_page *pp, | |||
350 | * delete @pp from partial page list @ppl. | 350 | * delete @pp from partial page list @ppl. |
351 | */ | 351 | */ |
352 | static void | 352 | static void |
353 | __ia32_delete_pp(struct partial_page_list *ppl, struct partial_page *pp, | 353 | __ia32_delete_pp(struct ia64_partial_page_list *ppl, |
354 | struct partial_page *prev) | 354 | struct ia64_partial_page *pp, struct ia64_partial_page *prev) |
355 | { | 355 | { |
356 | if (prev) { | 356 | if (prev) { |
357 | prev->next = pp->next; | 357 | prev->next = pp->next; |
@@ -363,15 +363,15 @@ __ia32_delete_pp(struct partial_page_list *ppl, struct partial_page *pp, | |||
363 | ppl->pp_hint = pp->next; | 363 | ppl->pp_hint = pp->next; |
364 | } | 364 | } |
365 | rb_erase(&pp->pp_rb, &ppl->ppl_rb); | 365 | rb_erase(&pp->pp_rb, &ppl->ppl_rb); |
366 | kmem_cache_free(partial_page_cachep, pp); | 366 | kmem_cache_free(ia64_partial_page_cachep, pp); |
367 | } | 367 | } |
368 | 368 | ||
369 | static struct partial_page * | 369 | static struct ia64_partial_page * |
370 | __pp_prev(struct partial_page *pp) | 370 | __pp_prev(struct ia64_partial_page *pp) |
371 | { | 371 | { |
372 | struct rb_node *prev = rb_prev(&pp->pp_rb); | 372 | struct rb_node *prev = rb_prev(&pp->pp_rb); |
373 | if (prev) | 373 | if (prev) |
374 | return rb_entry(prev, struct partial_page, pp_rb); | 374 | return rb_entry(prev, struct ia64_partial_page, pp_rb); |
375 | else | 375 | else |
376 | return NULL; | 376 | return NULL; |
377 | } | 377 | } |
@@ -383,7 +383,7 @@ __pp_prev(struct partial_page *pp) | |||
383 | static void | 383 | static void |
384 | __ia32_delete_pp_range(unsigned int start, unsigned int end) | 384 | __ia32_delete_pp_range(unsigned int start, unsigned int end) |
385 | { | 385 | { |
386 | struct partial_page *pp, *prev; | 386 | struct ia64_partial_page *pp, *prev; |
387 | struct rb_node **rb_link, *rb_parent; | 387 | struct rb_node **rb_link, *rb_parent; |
388 | 388 | ||
389 | if (start >= end) | 389 | if (start >= end) |
@@ -401,7 +401,7 @@ __ia32_delete_pp_range(unsigned int start, unsigned int end) | |||
401 | } | 401 | } |
402 | 402 | ||
403 | while (pp && pp->base < end) { | 403 | while (pp && pp->base < end) { |
404 | struct partial_page *tmp = pp->next; | 404 | struct ia64_partial_page *tmp = pp->next; |
405 | __ia32_delete_pp(current->thread.ppl, pp, prev); | 405 | __ia32_delete_pp(current->thread.ppl, pp, prev); |
406 | pp = tmp; | 406 | pp = tmp; |
407 | } | 407 | } |
@@ -414,7 +414,7 @@ __ia32_delete_pp_range(unsigned int start, unsigned int end) | |||
414 | static int | 414 | static int |
415 | __ia32_set_pp(unsigned int start, unsigned int end, int flags) | 415 | __ia32_set_pp(unsigned int start, unsigned int end, int flags) |
416 | { | 416 | { |
417 | struct partial_page *pp, *prev; | 417 | struct ia64_partial_page *pp, *prev; |
418 | struct rb_node ** rb_link, *rb_parent; | 418 | struct rb_node ** rb_link, *rb_parent; |
419 | unsigned int pstart, start_bit, end_bit, i; | 419 | unsigned int pstart, start_bit, end_bit, i; |
420 | 420 | ||
@@ -450,8 +450,8 @@ __ia32_set_pp(unsigned int start, unsigned int end, int flags) | |||
450 | return 0; | 450 | return 0; |
451 | } | 451 | } |
452 | 452 | ||
453 | /* new a partial_page */ | 453 | /* new a ia64_partial_page */ |
454 | pp = kmem_cache_alloc(partial_page_cachep, GFP_KERNEL); | 454 | pp = kmem_cache_alloc(ia64_partial_page_cachep, GFP_KERNEL); |
455 | if (!pp) | 455 | if (!pp) |
456 | return -ENOMEM; | 456 | return -ENOMEM; |
457 | pp->base = pstart; | 457 | pp->base = pstart; |
@@ -504,7 +504,7 @@ ia32_set_pp(unsigned int start, unsigned int end, int flags) | |||
504 | static int | 504 | static int |
505 | __ia32_unset_pp(unsigned int start, unsigned int end) | 505 | __ia32_unset_pp(unsigned int start, unsigned int end) |
506 | { | 506 | { |
507 | struct partial_page *pp, *prev; | 507 | struct ia64_partial_page *pp, *prev; |
508 | struct rb_node ** rb_link, *rb_parent; | 508 | struct rb_node ** rb_link, *rb_parent; |
509 | unsigned int pstart, start_bit, end_bit, i; | 509 | unsigned int pstart, start_bit, end_bit, i; |
510 | struct vm_area_struct *vma; | 510 | struct vm_area_struct *vma; |
@@ -532,8 +532,8 @@ __ia32_unset_pp(unsigned int start, unsigned int end) | |||
532 | return -ENOMEM; | 532 | return -ENOMEM; |
533 | } | 533 | } |
534 | 534 | ||
535 | /* new a partial_page */ | 535 | /* new a ia64_partial_page */ |
536 | pp = kmem_cache_alloc(partial_page_cachep, GFP_KERNEL); | 536 | pp = kmem_cache_alloc(ia64_partial_page_cachep, GFP_KERNEL); |
537 | if (!pp) | 537 | if (!pp) |
538 | return -ENOMEM; | 538 | return -ENOMEM; |
539 | pp->base = pstart; | 539 | pp->base = pstart; |
@@ -605,7 +605,7 @@ ia32_unset_pp(unsigned int *startp, unsigned int *endp) | |||
605 | static int | 605 | static int |
606 | __ia32_compare_pp(unsigned int start, unsigned int end) | 606 | __ia32_compare_pp(unsigned int start, unsigned int end) |
607 | { | 607 | { |
608 | struct partial_page *pp, *prev; | 608 | struct ia64_partial_page *pp, *prev; |
609 | struct rb_node ** rb_link, *rb_parent; | 609 | struct rb_node ** rb_link, *rb_parent; |
610 | unsigned int pstart, start_bit, end_bit, size; | 610 | unsigned int pstart, start_bit, end_bit, size; |
611 | unsigned int first_bit, next_zero_bit; /* the first range in bitmap */ | 611 | unsigned int first_bit, next_zero_bit; /* the first range in bitmap */ |
@@ -682,13 +682,13 @@ ia32_compare_pp(unsigned int *startp, unsigned int *endp) | |||
682 | } | 682 | } |
683 | 683 | ||
684 | static void | 684 | static void |
685 | __ia32_drop_pp_list(struct partial_page_list *ppl) | 685 | __ia32_drop_pp_list(struct ia64_partial_page_list *ppl) |
686 | { | 686 | { |
687 | struct partial_page *pp = ppl->pp_head; | 687 | struct ia64_partial_page *pp = ppl->pp_head; |
688 | 688 | ||
689 | while (pp) { | 689 | while (pp) { |
690 | struct partial_page *next = pp->next; | 690 | struct ia64_partial_page *next = pp->next; |
691 | kmem_cache_free(partial_page_cachep, pp); | 691 | kmem_cache_free(ia64_partial_page_cachep, pp); |
692 | pp = next; | 692 | pp = next; |
693 | } | 693 | } |
694 | 694 | ||
@@ -696,9 +696,9 @@ __ia32_drop_pp_list(struct partial_page_list *ppl) | |||
696 | } | 696 | } |
697 | 697 | ||
698 | void | 698 | void |
699 | ia32_drop_partial_page_list(struct task_struct *task) | 699 | ia32_drop_ia64_partial_page_list(struct task_struct *task) |
700 | { | 700 | { |
701 | struct partial_page_list* ppl = task->thread.ppl; | 701 | struct ia64_partial_page_list* ppl = task->thread.ppl; |
702 | 702 | ||
703 | if (ppl && atomic_dec_and_test(&ppl->pp_count)) | 703 | if (ppl && atomic_dec_and_test(&ppl->pp_count)) |
704 | __ia32_drop_pp_list(ppl); | 704 | __ia32_drop_pp_list(ppl); |
@@ -708,9 +708,9 @@ ia32_drop_partial_page_list(struct task_struct *task) | |||
708 | * Copy current->thread.ppl to ppl (already initialized). | 708 | * Copy current->thread.ppl to ppl (already initialized). |
709 | */ | 709 | */ |
710 | static int | 710 | static int |
711 | __ia32_copy_pp_list(struct partial_page_list *ppl) | 711 | __ia32_copy_pp_list(struct ia64_partial_page_list *ppl) |
712 | { | 712 | { |
713 | struct partial_page *pp, *tmp, *prev; | 713 | struct ia64_partial_page *pp, *tmp, *prev; |
714 | struct rb_node **rb_link, *rb_parent; | 714 | struct rb_node **rb_link, *rb_parent; |
715 | 715 | ||
716 | ppl->pp_head = NULL; | 716 | ppl->pp_head = NULL; |
@@ -721,7 +721,7 @@ __ia32_copy_pp_list(struct partial_page_list *ppl) | |||
721 | prev = NULL; | 721 | prev = NULL; |
722 | 722 | ||
723 | for (pp = current->thread.ppl->pp_head; pp; pp = pp->next) { | 723 | for (pp = current->thread.ppl->pp_head; pp; pp = pp->next) { |
724 | tmp = kmem_cache_alloc(partial_page_cachep, GFP_KERNEL); | 724 | tmp = kmem_cache_alloc(ia64_partial_page_cachep, GFP_KERNEL); |
725 | if (!tmp) | 725 | if (!tmp) |
726 | return -ENOMEM; | 726 | return -ENOMEM; |
727 | *tmp = *pp; | 727 | *tmp = *pp; |
@@ -734,7 +734,8 @@ __ia32_copy_pp_list(struct partial_page_list *ppl) | |||
734 | } | 734 | } |
735 | 735 | ||
736 | int | 736 | int |
737 | ia32_copy_partial_page_list(struct task_struct *p, unsigned long clone_flags) | 737 | ia32_copy_ia64_partial_page_list(struct task_struct *p, |
738 | unsigned long clone_flags) | ||
738 | { | 739 | { |
739 | int retval = 0; | 740 | int retval = 0; |
740 | 741 | ||
diff --git a/arch/ia64/kernel/acpi.c b/arch/ia64/kernel/acpi.c index 103dd8edda71..c6ede8780ded 100644 --- a/arch/ia64/kernel/acpi.c +++ b/arch/ia64/kernel/acpi.c | |||
@@ -67,6 +67,8 @@ EXPORT_SYMBOL(pm_power_off); | |||
67 | unsigned int acpi_cpei_override; | 67 | unsigned int acpi_cpei_override; |
68 | unsigned int acpi_cpei_phys_cpuid; | 68 | unsigned int acpi_cpei_phys_cpuid; |
69 | 69 | ||
70 | unsigned long acpi_wakeup_address = 0; | ||
71 | |||
70 | const char __init * | 72 | const char __init * |
71 | acpi_get_sysname(void) | 73 | acpi_get_sysname(void) |
72 | { | 74 | { |
@@ -986,4 +988,21 @@ int acpi_unregister_ioapic(acpi_handle handle, u32 gsi_base) | |||
986 | 988 | ||
987 | EXPORT_SYMBOL(acpi_unregister_ioapic); | 989 | EXPORT_SYMBOL(acpi_unregister_ioapic); |
988 | 990 | ||
991 | /* | ||
992 | * acpi_save_state_mem() - save kernel state | ||
993 | * | ||
994 | * TBD when when IA64 starts to support suspend... | ||
995 | */ | ||
996 | int acpi_save_state_mem(void) { return 0; } | ||
997 | |||
998 | /* | ||
999 | * acpi_restore_state() | ||
1000 | */ | ||
1001 | void acpi_restore_state_mem(void) {} | ||
1002 | |||
1003 | /* | ||
1004 | * do_suspend_lowlevel() | ||
1005 | */ | ||
1006 | void do_suspend_lowlevel(void) {} | ||
1007 | |||
989 | #endif /* CONFIG_ACPI */ | 1008 | #endif /* CONFIG_ACPI */ |
diff --git a/arch/ia64/kernel/cyclone.c b/arch/ia64/kernel/cyclone.c index 2fd96d9062a1..790ef0d87e12 100644 --- a/arch/ia64/kernel/cyclone.c +++ b/arch/ia64/kernel/cyclone.c | |||
@@ -38,11 +38,11 @@ static struct clocksource clocksource_cyclone = { | |||
38 | 38 | ||
39 | int __init init_cyclone_clock(void) | 39 | int __init init_cyclone_clock(void) |
40 | { | 40 | { |
41 | u64* reg; | 41 | u64 __iomem *reg; |
42 | u64 base; /* saved cyclone base address */ | 42 | u64 base; /* saved cyclone base address */ |
43 | u64 offset; /* offset from pageaddr to cyclone_timer register */ | 43 | u64 offset; /* offset from pageaddr to cyclone_timer register */ |
44 | int i; | 44 | int i; |
45 | u32* volatile cyclone_timer; /* Cyclone MPMC0 register */ | 45 | u32 __iomem *cyclone_timer; /* Cyclone MPMC0 register */ |
46 | 46 | ||
47 | if (!use_cyclone) | 47 | if (!use_cyclone) |
48 | return 0; | 48 | return 0; |
@@ -51,7 +51,7 @@ int __init init_cyclone_clock(void) | |||
51 | 51 | ||
52 | /* find base address */ | 52 | /* find base address */ |
53 | offset = (CYCLONE_CBAR_ADDR); | 53 | offset = (CYCLONE_CBAR_ADDR); |
54 | reg = (u64*)ioremap_nocache(offset, sizeof(u64)); | 54 | reg = ioremap_nocache(offset, sizeof(u64)); |
55 | if(!reg){ | 55 | if(!reg){ |
56 | printk(KERN_ERR "Summit chipset: Could not find valid CBAR" | 56 | printk(KERN_ERR "Summit chipset: Could not find valid CBAR" |
57 | " register.\n"); | 57 | " register.\n"); |
@@ -69,7 +69,7 @@ int __init init_cyclone_clock(void) | |||
69 | 69 | ||
70 | /* setup PMCC */ | 70 | /* setup PMCC */ |
71 | offset = (base + CYCLONE_PMCC_OFFSET); | 71 | offset = (base + CYCLONE_PMCC_OFFSET); |
72 | reg = (u64*)ioremap_nocache(offset, sizeof(u64)); | 72 | reg = ioremap_nocache(offset, sizeof(u64)); |
73 | if(!reg){ | 73 | if(!reg){ |
74 | printk(KERN_ERR "Summit chipset: Could not find valid PMCC" | 74 | printk(KERN_ERR "Summit chipset: Could not find valid PMCC" |
75 | " register.\n"); | 75 | " register.\n"); |
@@ -81,7 +81,7 @@ int __init init_cyclone_clock(void) | |||
81 | 81 | ||
82 | /* setup MPCS */ | 82 | /* setup MPCS */ |
83 | offset = (base + CYCLONE_MPCS_OFFSET); | 83 | offset = (base + CYCLONE_MPCS_OFFSET); |
84 | reg = (u64*)ioremap_nocache(offset, sizeof(u64)); | 84 | reg = ioremap_nocache(offset, sizeof(u64)); |
85 | if(!reg){ | 85 | if(!reg){ |
86 | printk(KERN_ERR "Summit chipset: Could not find valid MPCS" | 86 | printk(KERN_ERR "Summit chipset: Could not find valid MPCS" |
87 | " register.\n"); | 87 | " register.\n"); |
@@ -93,7 +93,7 @@ int __init init_cyclone_clock(void) | |||
93 | 93 | ||
94 | /* map in cyclone_timer */ | 94 | /* map in cyclone_timer */ |
95 | offset = (base + CYCLONE_MPMC_OFFSET); | 95 | offset = (base + CYCLONE_MPMC_OFFSET); |
96 | cyclone_timer = (u32*)ioremap_nocache(offset, sizeof(u32)); | 96 | cyclone_timer = ioremap_nocache(offset, sizeof(u32)); |
97 | if(!cyclone_timer){ | 97 | if(!cyclone_timer){ |
98 | printk(KERN_ERR "Summit chipset: Could not find valid MPMC" | 98 | printk(KERN_ERR "Summit chipset: Could not find valid MPMC" |
99 | " register.\n"); | 99 | " register.\n"); |
@@ -110,7 +110,7 @@ int __init init_cyclone_clock(void) | |||
110 | printk(KERN_ERR "Summit chipset: Counter not counting!" | 110 | printk(KERN_ERR "Summit chipset: Counter not counting!" |
111 | " DISABLED\n"); | 111 | " DISABLED\n"); |
112 | iounmap(cyclone_timer); | 112 | iounmap(cyclone_timer); |
113 | cyclone_timer = 0; | 113 | cyclone_timer = NULL; |
114 | use_cyclone = 0; | 114 | use_cyclone = 0; |
115 | return -ENODEV; | 115 | return -ENODEV; |
116 | } | 116 | } |
diff --git a/arch/ia64/kernel/head.S b/arch/ia64/kernel/head.S index 44d540efa6d1..4e5e27540e27 100644 --- a/arch/ia64/kernel/head.S +++ b/arch/ia64/kernel/head.S | |||
@@ -178,7 +178,7 @@ swapper_pg_dir: | |||
178 | halt_msg: | 178 | halt_msg: |
179 | stringz "Halting kernel\n" | 179 | stringz "Halting kernel\n" |
180 | 180 | ||
181 | .text | 181 | .section .text.head,"ax" |
182 | 182 | ||
183 | .global start_ap | 183 | .global start_ap |
184 | 184 | ||
@@ -392,6 +392,8 @@ self: hint @pause | |||
392 | br.sptk.many self // endless loop | 392 | br.sptk.many self // endless loop |
393 | END(_start) | 393 | END(_start) |
394 | 394 | ||
395 | .text | ||
396 | |||
395 | GLOBAL_ENTRY(ia64_save_debug_regs) | 397 | GLOBAL_ENTRY(ia64_save_debug_regs) |
396 | alloc r16=ar.pfs,1,0,0,0 | 398 | alloc r16=ar.pfs,1,0,0,0 |
397 | mov r20=ar.lc // preserve ar.lc | 399 | mov r20=ar.lc // preserve ar.lc |
diff --git a/arch/ia64/kernel/irq_ia64.c b/arch/ia64/kernel/irq_ia64.c index 91797c111162..fcb77338cc09 100644 --- a/arch/ia64/kernel/irq_ia64.c +++ b/arch/ia64/kernel/irq_ia64.c | |||
@@ -85,8 +85,8 @@ DEFINE_PER_CPU(int[IA64_NUM_VECTORS], vector_irq) = { | |||
85 | [0 ... IA64_NUM_VECTORS - 1] = IA64_SPURIOUS_INT_VECTOR | 85 | [0 ... IA64_NUM_VECTORS - 1] = IA64_SPURIOUS_INT_VECTOR |
86 | }; | 86 | }; |
87 | 87 | ||
88 | static cpumask_t vector_table[IA64_MAX_DEVICE_VECTORS] = { | 88 | static cpumask_t vector_table[IA64_NUM_VECTORS] = { |
89 | [0 ... IA64_MAX_DEVICE_VECTORS - 1] = CPU_MASK_NONE | 89 | [0 ... IA64_NUM_VECTORS - 1] = CPU_MASK_NONE |
90 | }; | 90 | }; |
91 | 91 | ||
92 | static int irq_status[NR_IRQS] = { | 92 | static int irq_status[NR_IRQS] = { |
@@ -123,17 +123,18 @@ static inline int find_unassigned_irq(void) | |||
123 | static inline int find_unassigned_vector(cpumask_t domain) | 123 | static inline int find_unassigned_vector(cpumask_t domain) |
124 | { | 124 | { |
125 | cpumask_t mask; | 125 | cpumask_t mask; |
126 | int pos; | 126 | int pos, vector; |
127 | 127 | ||
128 | cpus_and(mask, domain, cpu_online_map); | 128 | cpus_and(mask, domain, cpu_online_map); |
129 | if (cpus_empty(mask)) | 129 | if (cpus_empty(mask)) |
130 | return -EINVAL; | 130 | return -EINVAL; |
131 | 131 | ||
132 | for (pos = 0; pos < IA64_NUM_DEVICE_VECTORS; pos++) { | 132 | for (pos = 0; pos < IA64_NUM_DEVICE_VECTORS; pos++) { |
133 | cpus_and(mask, domain, vector_table[pos]); | 133 | vector = IA64_FIRST_DEVICE_VECTOR + pos; |
134 | cpus_and(mask, domain, vector_table[vector]); | ||
134 | if (!cpus_empty(mask)) | 135 | if (!cpus_empty(mask)) |
135 | continue; | 136 | continue; |
136 | return IA64_FIRST_DEVICE_VECTOR + pos; | 137 | return vector; |
137 | } | 138 | } |
138 | return -ENOSPC; | 139 | return -ENOSPC; |
139 | } | 140 | } |
@@ -141,7 +142,7 @@ static inline int find_unassigned_vector(cpumask_t domain) | |||
141 | static int __bind_irq_vector(int irq, int vector, cpumask_t domain) | 142 | static int __bind_irq_vector(int irq, int vector, cpumask_t domain) |
142 | { | 143 | { |
143 | cpumask_t mask; | 144 | cpumask_t mask; |
144 | int cpu, pos; | 145 | int cpu; |
145 | struct irq_cfg *cfg = &irq_cfg[irq]; | 146 | struct irq_cfg *cfg = &irq_cfg[irq]; |
146 | 147 | ||
147 | cpus_and(mask, domain, cpu_online_map); | 148 | cpus_and(mask, domain, cpu_online_map); |
@@ -156,8 +157,7 @@ static int __bind_irq_vector(int irq, int vector, cpumask_t domain) | |||
156 | cfg->vector = vector; | 157 | cfg->vector = vector; |
157 | cfg->domain = domain; | 158 | cfg->domain = domain; |
158 | irq_status[irq] = IRQ_USED; | 159 | irq_status[irq] = IRQ_USED; |
159 | pos = vector - IA64_FIRST_DEVICE_VECTOR; | 160 | cpus_or(vector_table[vector], vector_table[vector], domain); |
160 | cpus_or(vector_table[pos], vector_table[pos], domain); | ||
161 | return 0; | 161 | return 0; |
162 | } | 162 | } |
163 | 163 | ||
@@ -174,7 +174,7 @@ int bind_irq_vector(int irq, int vector, cpumask_t domain) | |||
174 | 174 | ||
175 | static void __clear_irq_vector(int irq) | 175 | static void __clear_irq_vector(int irq) |
176 | { | 176 | { |
177 | int vector, cpu, pos; | 177 | int vector, cpu; |
178 | cpumask_t mask; | 178 | cpumask_t mask; |
179 | cpumask_t domain; | 179 | cpumask_t domain; |
180 | struct irq_cfg *cfg = &irq_cfg[irq]; | 180 | struct irq_cfg *cfg = &irq_cfg[irq]; |
@@ -189,8 +189,7 @@ static void __clear_irq_vector(int irq) | |||
189 | cfg->vector = IRQ_VECTOR_UNASSIGNED; | 189 | cfg->vector = IRQ_VECTOR_UNASSIGNED; |
190 | cfg->domain = CPU_MASK_NONE; | 190 | cfg->domain = CPU_MASK_NONE; |
191 | irq_status[irq] = IRQ_UNUSED; | 191 | irq_status[irq] = IRQ_UNUSED; |
192 | pos = vector - IA64_FIRST_DEVICE_VECTOR; | 192 | cpus_andnot(vector_table[vector], vector_table[vector], domain); |
193 | cpus_andnot(vector_table[pos], vector_table[pos], domain); | ||
194 | } | 193 | } |
195 | 194 | ||
196 | static void clear_irq_vector(int irq) | 195 | static void clear_irq_vector(int irq) |
@@ -212,9 +211,6 @@ assign_irq_vector (int irq) | |||
212 | vector = -ENOSPC; | 211 | vector = -ENOSPC; |
213 | 212 | ||
214 | spin_lock_irqsave(&vector_lock, flags); | 213 | spin_lock_irqsave(&vector_lock, flags); |
215 | if (irq < 0) { | ||
216 | goto out; | ||
217 | } | ||
218 | for_each_online_cpu(cpu) { | 214 | for_each_online_cpu(cpu) { |
219 | domain = vector_allocation_domain(cpu); | 215 | domain = vector_allocation_domain(cpu); |
220 | vector = find_unassigned_vector(domain); | 216 | vector = find_unassigned_vector(domain); |
@@ -223,6 +219,8 @@ assign_irq_vector (int irq) | |||
223 | } | 219 | } |
224 | if (vector < 0) | 220 | if (vector < 0) |
225 | goto out; | 221 | goto out; |
222 | if (irq == AUTO_ASSIGN) | ||
223 | irq = vector; | ||
226 | BUG_ON(__bind_irq_vector(irq, vector, domain)); | 224 | BUG_ON(__bind_irq_vector(irq, vector, domain)); |
227 | out: | 225 | out: |
228 | spin_unlock_irqrestore(&vector_lock, flags); | 226 | spin_unlock_irqrestore(&vector_lock, flags); |
diff --git a/arch/ia64/kernel/machvec.c b/arch/ia64/kernel/machvec.c index 13df337508e7..7ccb228ceedc 100644 --- a/arch/ia64/kernel/machvec.c +++ b/arch/ia64/kernel/machvec.c | |||
@@ -13,14 +13,6 @@ | |||
13 | struct ia64_machine_vector ia64_mv; | 13 | struct ia64_machine_vector ia64_mv; |
14 | EXPORT_SYMBOL(ia64_mv); | 14 | EXPORT_SYMBOL(ia64_mv); |
15 | 15 | ||
16 | static __initdata const char *mvec_name; | ||
17 | static __init int setup_mvec(char *s) | ||
18 | { | ||
19 | mvec_name = s; | ||
20 | return 0; | ||
21 | } | ||
22 | early_param("machvec", setup_mvec); | ||
23 | |||
24 | static struct ia64_machine_vector * __init | 16 | static struct ia64_machine_vector * __init |
25 | lookup_machvec (const char *name) | 17 | lookup_machvec (const char *name) |
26 | { | 18 | { |
@@ -41,7 +33,7 @@ machvec_init (const char *name) | |||
41 | struct ia64_machine_vector *mv; | 33 | struct ia64_machine_vector *mv; |
42 | 34 | ||
43 | if (!name) | 35 | if (!name) |
44 | name = mvec_name ? mvec_name : acpi_get_sysname(); | 36 | name = acpi_get_sysname(); |
45 | mv = lookup_machvec(name); | 37 | mv = lookup_machvec(name); |
46 | if (!mv) | 38 | if (!mv) |
47 | panic("generic kernel failed to find machine vector for" | 39 | panic("generic kernel failed to find machine vector for" |
@@ -51,6 +43,23 @@ machvec_init (const char *name) | |||
51 | printk(KERN_INFO "booting generic kernel on platform %s\n", name); | 43 | printk(KERN_INFO "booting generic kernel on platform %s\n", name); |
52 | } | 44 | } |
53 | 45 | ||
46 | void __init | ||
47 | machvec_init_from_cmdline(const char *cmdline) | ||
48 | { | ||
49 | char str[64]; | ||
50 | const char *start; | ||
51 | char *end; | ||
52 | |||
53 | if (! (start = strstr(cmdline, "machvec=")) ) | ||
54 | return machvec_init(NULL); | ||
55 | |||
56 | strlcpy(str, start + strlen("machvec="), sizeof(str)); | ||
57 | if ( (end = strchr(str, ' ')) ) | ||
58 | *end = '\0'; | ||
59 | |||
60 | return machvec_init(str); | ||
61 | } | ||
62 | |||
54 | #endif /* CONFIG_IA64_GENERIC */ | 63 | #endif /* CONFIG_IA64_GENERIC */ |
55 | 64 | ||
56 | void | 65 | void |
diff --git a/arch/ia64/kernel/process.c b/arch/ia64/kernel/process.c index fa40cba43350..4158906c45aa 100644 --- a/arch/ia64/kernel/process.c +++ b/arch/ia64/kernel/process.c | |||
@@ -499,7 +499,8 @@ copy_thread (int nr, unsigned long clone_flags, | |||
499 | 499 | ||
500 | /* Copy partially mapped page list */ | 500 | /* Copy partially mapped page list */ |
501 | if (!retval) | 501 | if (!retval) |
502 | retval = ia32_copy_partial_page_list(p, clone_flags); | 502 | retval = ia32_copy_ia64_partial_page_list(p, |
503 | clone_flags); | ||
503 | } | 504 | } |
504 | #endif | 505 | #endif |
505 | 506 | ||
@@ -728,7 +729,7 @@ flush_thread (void) | |||
728 | ia64_drop_fpu(current); | 729 | ia64_drop_fpu(current); |
729 | #ifdef CONFIG_IA32_SUPPORT | 730 | #ifdef CONFIG_IA32_SUPPORT |
730 | if (IS_IA32_PROCESS(task_pt_regs(current))) { | 731 | if (IS_IA32_PROCESS(task_pt_regs(current))) { |
731 | ia32_drop_partial_page_list(current); | 732 | ia32_drop_ia64_partial_page_list(current); |
732 | current->thread.task_size = IA32_PAGE_OFFSET; | 733 | current->thread.task_size = IA32_PAGE_OFFSET; |
733 | set_fs(USER_DS); | 734 | set_fs(USER_DS); |
734 | } | 735 | } |
@@ -754,7 +755,7 @@ exit_thread (void) | |||
754 | pfm_release_debug_registers(current); | 755 | pfm_release_debug_registers(current); |
755 | #endif | 756 | #endif |
756 | if (IS_IA32_PROCESS(task_pt_regs(current))) | 757 | if (IS_IA32_PROCESS(task_pt_regs(current))) |
757 | ia32_drop_partial_page_list(current); | 758 | ia32_drop_ia64_partial_page_list(current); |
758 | } | 759 | } |
759 | 760 | ||
760 | unsigned long | 761 | unsigned long |
diff --git a/arch/ia64/kernel/setup.c b/arch/ia64/kernel/setup.c index cf06fe799041..7cecd2964200 100644 --- a/arch/ia64/kernel/setup.c +++ b/arch/ia64/kernel/setup.c | |||
@@ -491,12 +491,17 @@ setup_arch (char **cmdline_p) | |||
491 | efi_init(); | 491 | efi_init(); |
492 | io_port_init(); | 492 | io_port_init(); |
493 | 493 | ||
494 | parse_early_param(); | ||
495 | |||
496 | #ifdef CONFIG_IA64_GENERIC | 494 | #ifdef CONFIG_IA64_GENERIC |
497 | machvec_init(NULL); | 495 | /* machvec needs to be parsed from the command line |
496 | * before parse_early_param() is called to ensure | ||
497 | * that ia64_mv is initialised before any command line | ||
498 | * settings may cause console setup to occur | ||
499 | */ | ||
500 | machvec_init_from_cmdline(*cmdline_p); | ||
498 | #endif | 501 | #endif |
499 | 502 | ||
503 | parse_early_param(); | ||
504 | |||
500 | if (early_console_setup(*cmdline_p) == 0) | 505 | if (early_console_setup(*cmdline_p) == 0) |
501 | mark_bsp_online(); | 506 | mark_bsp_online(); |
502 | 507 | ||
diff --git a/arch/ia64/kernel/smp.c b/arch/ia64/kernel/smp.c index 9f72838db26e..0982882bfb80 100644 --- a/arch/ia64/kernel/smp.c +++ b/arch/ia64/kernel/smp.c | |||
@@ -468,7 +468,7 @@ smp_send_stop (void) | |||
468 | send_IPI_allbutself(IPI_CPU_STOP); | 468 | send_IPI_allbutself(IPI_CPU_STOP); |
469 | } | 469 | } |
470 | 470 | ||
471 | int __init | 471 | int |
472 | setup_profiling_timer (unsigned int multiplier) | 472 | setup_profiling_timer (unsigned int multiplier) |
473 | { | 473 | { |
474 | return -EINVAL; | 474 | return -EINVAL; |
diff --git a/arch/ia64/kernel/smpboot.c b/arch/ia64/kernel/smpboot.c index 9f5c90b594b9..62209dcf06d3 100644 --- a/arch/ia64/kernel/smpboot.c +++ b/arch/ia64/kernel/smpboot.c | |||
@@ -487,7 +487,7 @@ struct create_idle { | |||
487 | int cpu; | 487 | int cpu; |
488 | }; | 488 | }; |
489 | 489 | ||
490 | void | 490 | void __cpuinit |
491 | do_fork_idle(struct work_struct *work) | 491 | do_fork_idle(struct work_struct *work) |
492 | { | 492 | { |
493 | struct create_idle *c_idle = | 493 | struct create_idle *c_idle = |
@@ -497,7 +497,7 @@ do_fork_idle(struct work_struct *work) | |||
497 | complete(&c_idle->done); | 497 | complete(&c_idle->done); |
498 | } | 498 | } |
499 | 499 | ||
500 | static int __devinit | 500 | static int __cpuinit |
501 | do_boot_cpu (int sapicid, int cpu) | 501 | do_boot_cpu (int sapicid, int cpu) |
502 | { | 502 | { |
503 | int timeout; | 503 | int timeout; |
@@ -808,7 +808,7 @@ set_cpu_sibling_map(int cpu) | |||
808 | } | 808 | } |
809 | } | 809 | } |
810 | 810 | ||
811 | int __devinit | 811 | int __cpuinit |
812 | __cpu_up (unsigned int cpu) | 812 | __cpu_up (unsigned int cpu) |
813 | { | 813 | { |
814 | int ret; | 814 | int ret; |
diff --git a/arch/ia64/kernel/time.c b/arch/ia64/kernel/time.c index 627785c48ea9..6c0e9e2e1b82 100644 --- a/arch/ia64/kernel/time.c +++ b/arch/ia64/kernel/time.c | |||
@@ -52,7 +52,7 @@ static struct clocksource clocksource_itc = { | |||
52 | .name = "itc", | 52 | .name = "itc", |
53 | .rating = 350, | 53 | .rating = 350, |
54 | .read = itc_get_cycles, | 54 | .read = itc_get_cycles, |
55 | .mask = 0xffffffffffffffff, | 55 | .mask = CLOCKSOURCE_MASK(64), |
56 | .mult = 0, /*to be caluclated*/ | 56 | .mult = 0, /*to be caluclated*/ |
57 | .shift = 16, | 57 | .shift = 16, |
58 | .flags = CLOCK_SOURCE_IS_CONTINUOUS, | 58 | .flags = CLOCK_SOURCE_IS_CONTINUOUS, |
@@ -255,7 +255,7 @@ ia64_init_itm (void) | |||
255 | } | 255 | } |
256 | } | 256 | } |
257 | 257 | ||
258 | static cycle_t itc_get_cycles() | 258 | static cycle_t itc_get_cycles(void) |
259 | { | 259 | { |
260 | u64 lcycle, now, ret; | 260 | u64 lcycle, now, ret; |
261 | 261 | ||
diff --git a/arch/ia64/kernel/vmlinux.lds.S b/arch/ia64/kernel/vmlinux.lds.S index 860f251d2fc2..83e80677de70 100644 --- a/arch/ia64/kernel/vmlinux.lds.S +++ b/arch/ia64/kernel/vmlinux.lds.S | |||
@@ -50,6 +50,8 @@ SECTIONS | |||
50 | KPROBES_TEXT | 50 | KPROBES_TEXT |
51 | *(.gnu.linkonce.t*) | 51 | *(.gnu.linkonce.t*) |
52 | } | 52 | } |
53 | .text.head : AT(ADDR(.text.head) - LOAD_OFFSET) | ||
54 | { *(.text.head) } | ||
53 | .text2 : AT(ADDR(.text2) - LOAD_OFFSET) | 55 | .text2 : AT(ADDR(.text2) - LOAD_OFFSET) |
54 | { *(.text2) } | 56 | { *(.text2) } |
55 | #ifdef CONFIG_SMP | 57 | #ifdef CONFIG_SMP |
diff --git a/arch/ia64/pci/pci.c b/arch/ia64/pci/pci.c index 07d0e92742c8..488e48a5deea 100644 --- a/arch/ia64/pci/pci.c +++ b/arch/ia64/pci/pci.c | |||
@@ -581,7 +581,7 @@ pcibios_align_resource (void *data, struct resource *res, | |||
581 | /* | 581 | /* |
582 | * PCI BIOS setup, always defaults to SAL interface | 582 | * PCI BIOS setup, always defaults to SAL interface |
583 | */ | 583 | */ |
584 | char * __init | 584 | char * __devinit |
585 | pcibios_setup (char *str) | 585 | pcibios_setup (char *str) |
586 | { | 586 | { |
587 | return str; | 587 | return str; |
diff --git a/arch/m32r/kernel/setup_mappi.c b/arch/m32r/kernel/setup_mappi.c index 6b2d77da0683..fe73c9ec611f 100644 --- a/arch/m32r/kernel/setup_mappi.c +++ b/arch/m32r/kernel/setup_mappi.c | |||
@@ -45,7 +45,8 @@ static void mask_and_ack_mappi(unsigned int irq) | |||
45 | 45 | ||
46 | static void end_mappi_irq(unsigned int irq) | 46 | static void end_mappi_irq(unsigned int irq) |
47 | { | 47 | { |
48 | enable_mappi_irq(irq); | 48 | if (!(irq_desc[irq].status & (IRQ_DISABLED | IRQ_INPROGRESS))) |
49 | enable_mappi_irq(irq); | ||
49 | } | 50 | } |
50 | 51 | ||
51 | static unsigned int startup_mappi_irq(unsigned int irq) | 52 | static unsigned int startup_mappi_irq(unsigned int irq) |
@@ -88,7 +89,7 @@ void __init init_IRQ(void) | |||
88 | irq_desc[M32R_IRQ_INT0].chip = &mappi_irq_type; | 89 | irq_desc[M32R_IRQ_INT0].chip = &mappi_irq_type; |
89 | irq_desc[M32R_IRQ_INT0].action = NULL; | 90 | irq_desc[M32R_IRQ_INT0].action = NULL; |
90 | irq_desc[M32R_IRQ_INT0].depth = 1; | 91 | irq_desc[M32R_IRQ_INT0].depth = 1; |
91 | icu_data[M32R_IRQ_INT0].icucr = M32R_ICUCR_IEN|M32R_ICUCR_ISMOD10; | 92 | icu_data[M32R_IRQ_INT0].icucr = M32R_ICUCR_IEN|M32R_ICUCR_ISMOD11; |
92 | disable_mappi_irq(M32R_IRQ_INT0); | 93 | disable_mappi_irq(M32R_IRQ_INT0); |
93 | #endif /* CONFIG_M32R_NE2000 */ | 94 | #endif /* CONFIG_M32R_NE2000 */ |
94 | 95 | ||
diff --git a/arch/m68knommu/Kconfig b/arch/m68knommu/Kconfig index 1175ceff8b2a..185906b54cb0 100644 --- a/arch/m68knommu/Kconfig +++ b/arch/m68knommu/Kconfig | |||
@@ -216,6 +216,18 @@ config XCOPILOT_BUGS | |||
216 | help | 216 | help |
217 | Support the bugs of Xcopilot. | 217 | Support the bugs of Xcopilot. |
218 | 218 | ||
219 | config UC5272 | ||
220 | bool 'Arcturus Networks uC5272 dimm board support' | ||
221 | depends on M5272 | ||
222 | help | ||
223 | Support for the Arcturus Networks uC5272 dimm board. | ||
224 | |||
225 | config UC5282 | ||
226 | bool "Arcturus Networks uC5282 board support" | ||
227 | depends on M528x | ||
228 | help | ||
229 | Support for the Arcturus Networks uC5282 dimm board. | ||
230 | |||
219 | config UCSIMM | 231 | config UCSIMM |
220 | bool "uCsimm module support" | 232 | bool "uCsimm module support" |
221 | depends on M68EZ328 | 233 | depends on M68EZ328 |
@@ -342,6 +354,18 @@ config SOM5282EM | |||
342 | depends on M528x | 354 | depends on M528x |
343 | help | 355 | help |
344 | Support for the EMAC.Inc SOM5282EM module. | 356 | Support for the EMAC.Inc SOM5282EM module. |
357 | |||
358 | config WILDFIRE | ||
359 | bool "Intec Automation Inc. WildFire board support" | ||
360 | depends on M528x | ||
361 | help | ||
362 | Support for the Intec Automation Inc. WildFire. | ||
363 | |||
364 | config WILDFIREMOD | ||
365 | bool "Intec Automation Inc. WildFire module support" | ||
366 | depends on M528x | ||
367 | help | ||
368 | Support for the Intec Automation Inc. WildFire module. | ||
345 | 369 | ||
346 | config ARN5307 | 370 | config ARN5307 |
347 | bool "Arnewsh 5307 board support" | 371 | bool "Arnewsh 5307 board support" |
diff --git a/arch/m68knommu/Makefile b/arch/m68knommu/Makefile index 8951793fd8d4..1305cc980023 100644 --- a/arch/m68knommu/Makefile +++ b/arch/m68knommu/Makefile | |||
@@ -26,6 +26,8 @@ platform-$(CONFIG_M5407) := 5407 | |||
26 | PLATFORM := $(platform-y) | 26 | PLATFORM := $(platform-y) |
27 | 27 | ||
28 | board-$(CONFIG_PILOT) := pilot | 28 | board-$(CONFIG_PILOT) := pilot |
29 | board-$(CONFIG_UC5272) := UC5272 | ||
30 | board-$(CONFIG_UC5282) := UC5282 | ||
29 | board-$(CONFIG_UCSIMM) := ucsimm | 31 | board-$(CONFIG_UCSIMM) := ucsimm |
30 | board-$(CONFIG_UCDIMM) := ucdimm | 32 | board-$(CONFIG_UCDIMM) := ucdimm |
31 | board-$(CONFIG_UCQUICC) := uCquicc | 33 | board-$(CONFIG_UCQUICC) := uCquicc |
diff --git a/arch/m68knommu/kernel/dma.c b/arch/m68knommu/kernel/dma.c index 0a25874a2aae..e10eafc52789 100644 --- a/arch/m68knommu/kernel/dma.c +++ b/arch/m68knommu/kernel/dma.c | |||
@@ -8,6 +8,7 @@ | |||
8 | #include <linux/types.h> | 8 | #include <linux/types.h> |
9 | #include <linux/mm.h> | 9 | #include <linux/mm.h> |
10 | #include <linux/string.h> | 10 | #include <linux/string.h> |
11 | #include <linux/device.h> | ||
11 | #include <asm/io.h> | 12 | #include <asm/io.h> |
12 | 13 | ||
13 | void *dma_alloc_coherent(struct device *dev, size_t size, | 14 | void *dma_alloc_coherent(struct device *dev, size_t size, |
diff --git a/arch/m68knommu/kernel/setup.c b/arch/m68knommu/kernel/setup.c index 2203f694f26b..3f86ade3a22a 100644 --- a/arch/m68knommu/kernel/setup.c +++ b/arch/m68knommu/kernel/setup.c | |||
@@ -42,8 +42,6 @@ EXPORT_SYMBOL(memory_end); | |||
42 | 42 | ||
43 | char __initdata command_line[COMMAND_LINE_SIZE]; | 43 | char __initdata command_line[COMMAND_LINE_SIZE]; |
44 | 44 | ||
45 | void (*mach_trap_init)(void); | ||
46 | |||
47 | /* machine dependent timer functions */ | 45 | /* machine dependent timer functions */ |
48 | void (*mach_sched_init)(irq_handler_t handler); | 46 | void (*mach_sched_init)(irq_handler_t handler); |
49 | void (*mach_tick)(void); | 47 | void (*mach_tick)(void); |
@@ -132,6 +130,11 @@ void setup_arch(char **cmdline_p) | |||
132 | 130 | ||
133 | config_BSP(&command_line[0], sizeof(command_line)); | 131 | config_BSP(&command_line[0], sizeof(command_line)); |
134 | 132 | ||
133 | #if defined(CONFIG_BOOTPARAM) | ||
134 | strncpy(&command_line[0], CONFIG_BOOTPARAM_STRING, sizeof(command_line)); | ||
135 | command_line[sizeof(command_line) - 1] = 0; | ||
136 | #endif | ||
137 | |||
135 | printk(KERN_INFO "\x0F\r\n\nuClinux/" CPU "\n"); | 138 | printk(KERN_INFO "\x0F\r\n\nuClinux/" CPU "\n"); |
136 | 139 | ||
137 | #ifdef CONFIG_UCDIMM | 140 | #ifdef CONFIG_UCDIMM |
diff --git a/arch/m68knommu/platform/5206/config.c b/arch/m68knommu/platform/5206/config.c index 3343830aad10..d0f2dc5cb5a1 100644 --- a/arch/m68knommu/platform/5206/config.c +++ b/arch/m68knommu/platform/5206/config.c | |||
@@ -28,7 +28,6 @@ | |||
28 | void coldfire_tick(void); | 28 | void coldfire_tick(void); |
29 | void coldfire_timer_init(irq_handler_t handler); | 29 | void coldfire_timer_init(irq_handler_t handler); |
30 | unsigned long coldfire_timer_offset(void); | 30 | unsigned long coldfire_timer_offset(void); |
31 | void coldfire_trap_init(void); | ||
32 | void coldfire_reset(void); | 31 | void coldfire_reset(void); |
33 | 32 | ||
34 | /***************************************************************************/ | 33 | /***************************************************************************/ |
@@ -98,18 +97,9 @@ int mcf_timerirqpending(int timer) | |||
98 | void config_BSP(char *commandp, int size) | 97 | void config_BSP(char *commandp, int size) |
99 | { | 98 | { |
100 | mcf_setimr(MCFSIM_IMR_MASKALL); | 99 | mcf_setimr(MCFSIM_IMR_MASKALL); |
101 | |||
102 | #if defined(CONFIG_BOOTPARAM) | ||
103 | strncpy(commandp, CONFIG_BOOTPARAM_STRING, size); | ||
104 | commandp[size-1] = 0; | ||
105 | #else | ||
106 | memset(commandp, 0, size); | ||
107 | #endif | ||
108 | |||
109 | mach_sched_init = coldfire_timer_init; | 100 | mach_sched_init = coldfire_timer_init; |
110 | mach_tick = coldfire_tick; | 101 | mach_tick = coldfire_tick; |
111 | mach_gettimeoffset = coldfire_timer_offset; | 102 | mach_gettimeoffset = coldfire_timer_offset; |
112 | mach_trap_init = coldfire_trap_init; | ||
113 | mach_reset = coldfire_reset; | 103 | mach_reset = coldfire_reset; |
114 | } | 104 | } |
115 | 105 | ||
diff --git a/arch/m68knommu/platform/5206e/config.c b/arch/m68knommu/platform/5206e/config.c index 0f67320b4031..4ab614f1ecda 100644 --- a/arch/m68knommu/platform/5206e/config.c +++ b/arch/m68knommu/platform/5206e/config.c | |||
@@ -27,7 +27,6 @@ | |||
27 | void coldfire_tick(void); | 27 | void coldfire_tick(void); |
28 | void coldfire_timer_init(irq_handler_t handler); | 28 | void coldfire_timer_init(irq_handler_t handler); |
29 | unsigned long coldfire_timer_offset(void); | 29 | unsigned long coldfire_timer_offset(void); |
30 | void coldfire_trap_init(void); | ||
31 | void coldfire_reset(void); | 30 | void coldfire_reset(void); |
32 | 31 | ||
33 | /***************************************************************************/ | 32 | /***************************************************************************/ |
@@ -98,21 +97,15 @@ void config_BSP(char *commandp, int size) | |||
98 | { | 97 | { |
99 | mcf_setimr(MCFSIM_IMR_MASKALL); | 98 | mcf_setimr(MCFSIM_IMR_MASKALL); |
100 | 99 | ||
101 | #if defined(CONFIG_BOOTPARAM) | 100 | #if defined(CONFIG_NETtel) |
102 | strncpy(commandp, CONFIG_BOOTPARAM_STRING, size); | ||
103 | commandp[size-1] = 0; | ||
104 | #elif defined(CONFIG_NETtel) | ||
105 | /* Copy command line from FLASH to local buffer... */ | 101 | /* Copy command line from FLASH to local buffer... */ |
106 | memcpy(commandp, (char *) 0xf0004000, size); | 102 | memcpy(commandp, (char *) 0xf0004000, size); |
107 | commandp[size-1] = 0; | 103 | commandp[size-1] = 0; |
108 | #else | ||
109 | memset(commandp, 0, size); | ||
110 | #endif /* CONFIG_NETtel */ | 104 | #endif /* CONFIG_NETtel */ |
111 | 105 | ||
112 | mach_sched_init = coldfire_timer_init; | 106 | mach_sched_init = coldfire_timer_init; |
113 | mach_tick = coldfire_tick; | 107 | mach_tick = coldfire_tick; |
114 | mach_gettimeoffset = coldfire_timer_offset; | 108 | mach_gettimeoffset = coldfire_timer_offset; |
115 | mach_trap_init = coldfire_trap_init; | ||
116 | mach_reset = coldfire_reset; | 109 | mach_reset = coldfire_reset; |
117 | } | 110 | } |
118 | 111 | ||
diff --git a/arch/m68knommu/platform/520x/config.c b/arch/m68knommu/platform/520x/config.c index 58b2878deb61..a2c95bebd004 100644 --- a/arch/m68knommu/platform/520x/config.c +++ b/arch/m68knommu/platform/520x/config.c | |||
@@ -30,7 +30,6 @@ unsigned int dma_device_address[MAX_M68K_DMA_CHANNELS]; | |||
30 | void coldfire_pit_tick(void); | 30 | void coldfire_pit_tick(void); |
31 | void coldfire_pit_init(irq_handler_t handler); | 31 | void coldfire_pit_init(irq_handler_t handler); |
32 | unsigned long coldfire_pit_offset(void); | 32 | unsigned long coldfire_pit_offset(void); |
33 | void coldfire_trap_init(void); | ||
34 | void coldfire_reset(void); | 33 | void coldfire_reset(void); |
35 | 34 | ||
36 | /***************************************************************************/ | 35 | /***************************************************************************/ |
@@ -48,17 +47,9 @@ void mcf_autovector(unsigned int vec) | |||
48 | 47 | ||
49 | void config_BSP(char *commandp, int size) | 48 | void config_BSP(char *commandp, int size) |
50 | { | 49 | { |
51 | #ifdef CONFIG_BOOTPARAM | ||
52 | strncpy(commandp, CONFIG_BOOTPARAM_STRING, size); | ||
53 | commandp[size-1] = 0; | ||
54 | #else | ||
55 | memset(commandp, 0, size); | ||
56 | #endif | ||
57 | |||
58 | mach_sched_init = coldfire_pit_init; | 50 | mach_sched_init = coldfire_pit_init; |
59 | mach_tick = coldfire_pit_tick; | 51 | mach_tick = coldfire_pit_tick; |
60 | mach_gettimeoffset = coldfire_pit_offset; | 52 | mach_gettimeoffset = coldfire_pit_offset; |
61 | mach_trap_init = coldfire_trap_init; | ||
62 | mach_reset = coldfire_reset; | 53 | mach_reset = coldfire_reset; |
63 | } | 54 | } |
64 | 55 | ||
diff --git a/arch/m68knommu/platform/523x/config.c b/arch/m68knommu/platform/523x/config.c index 9b054e6caee2..0a3af05a434b 100644 --- a/arch/m68knommu/platform/523x/config.c +++ b/arch/m68knommu/platform/523x/config.c | |||
@@ -29,7 +29,6 @@ | |||
29 | void coldfire_pit_tick(void); | 29 | void coldfire_pit_tick(void); |
30 | void coldfire_pit_init(irq_handler_t handler); | 30 | void coldfire_pit_init(irq_handler_t handler); |
31 | unsigned long coldfire_pit_offset(void); | 31 | unsigned long coldfire_pit_offset(void); |
32 | void coldfire_trap_init(void); | ||
33 | void coldfire_reset(void); | 32 | void coldfire_reset(void); |
34 | 33 | ||
35 | /***************************************************************************/ | 34 | /***************************************************************************/ |
@@ -63,18 +62,9 @@ void mcf_autovector(unsigned int vec) | |||
63 | void config_BSP(char *commandp, int size) | 62 | void config_BSP(char *commandp, int size) |
64 | { | 63 | { |
65 | mcf_disableall(); | 64 | mcf_disableall(); |
66 | |||
67 | #ifdef CONFIG_BOOTPARAM | ||
68 | strncpy(commandp, CONFIG_BOOTPARAM_STRING, size); | ||
69 | commandp[size-1] = 0; | ||
70 | #else | ||
71 | memset(commandp, 0, size); | ||
72 | #endif | ||
73 | |||
74 | mach_sched_init = coldfire_pit_init; | 65 | mach_sched_init = coldfire_pit_init; |
75 | mach_tick = coldfire_pit_tick; | 66 | mach_tick = coldfire_pit_tick; |
76 | mach_gettimeoffset = coldfire_pit_offset; | 67 | mach_gettimeoffset = coldfire_pit_offset; |
77 | mach_trap_init = coldfire_trap_init; | ||
78 | mach_reset = coldfire_reset; | 68 | mach_reset = coldfire_reset; |
79 | } | 69 | } |
80 | 70 | ||
diff --git a/arch/m68knommu/platform/5249/config.c b/arch/m68knommu/platform/5249/config.c index d6706079d64a..dc2c362590c2 100644 --- a/arch/m68knommu/platform/5249/config.c +++ b/arch/m68knommu/platform/5249/config.c | |||
@@ -27,7 +27,6 @@ | |||
27 | void coldfire_tick(void); | 27 | void coldfire_tick(void); |
28 | void coldfire_timer_init(irq_handler_t handler); | 28 | void coldfire_timer_init(irq_handler_t handler); |
29 | unsigned long coldfire_timer_offset(void); | 29 | unsigned long coldfire_timer_offset(void); |
30 | void coldfire_trap_init(void); | ||
31 | void coldfire_reset(void); | 30 | void coldfire_reset(void); |
32 | 31 | ||
33 | /***************************************************************************/ | 32 | /***************************************************************************/ |
@@ -96,18 +95,9 @@ int mcf_timerirqpending(int timer) | |||
96 | void config_BSP(char *commandp, int size) | 95 | void config_BSP(char *commandp, int size) |
97 | { | 96 | { |
98 | mcf_setimr(MCFSIM_IMR_MASKALL); | 97 | mcf_setimr(MCFSIM_IMR_MASKALL); |
99 | |||
100 | #if defined(CONFIG_BOOTPARAM) | ||
101 | strncpy(commandp, CONFIG_BOOTPARAM_STRING, size); | ||
102 | commandp[size-1] = 0; | ||
103 | #else | ||
104 | memset(commandp, 0, size); | ||
105 | #endif | ||
106 | |||
107 | mach_sched_init = coldfire_timer_init; | 98 | mach_sched_init = coldfire_timer_init; |
108 | mach_tick = coldfire_tick; | 99 | mach_tick = coldfire_tick; |
109 | mach_gettimeoffset = coldfire_timer_offset; | 100 | mach_gettimeoffset = coldfire_timer_offset; |
110 | mach_trap_init = coldfire_trap_init; | ||
111 | mach_reset = coldfire_reset; | 101 | mach_reset = coldfire_reset; |
112 | } | 102 | } |
113 | 103 | ||
diff --git a/arch/m68knommu/platform/5272/config.c b/arch/m68knommu/platform/5272/config.c index 6b437cc97776..1365a8300d5d 100644 --- a/arch/m68knommu/platform/5272/config.c +++ b/arch/m68knommu/platform/5272/config.c | |||
@@ -28,7 +28,6 @@ | |||
28 | void coldfire_tick(void); | 28 | void coldfire_tick(void); |
29 | void coldfire_timer_init(irq_handler_t handler); | 29 | void coldfire_timer_init(irq_handler_t handler); |
30 | unsigned long coldfire_timer_offset(void); | 30 | unsigned long coldfire_timer_offset(void); |
31 | void coldfire_trap_init(void); | ||
32 | void coldfire_reset(void); | 31 | void coldfire_reset(void); |
33 | 32 | ||
34 | extern unsigned int mcf_timervector; | 33 | extern unsigned int mcf_timervector; |
@@ -113,10 +112,7 @@ void config_BSP(char *commandp, int size) | |||
113 | 112 | ||
114 | mcf_disableall(); | 113 | mcf_disableall(); |
115 | 114 | ||
116 | #if defined(CONFIG_BOOTPARAM) | 115 | #if defined(CONFIG_NETtel) || defined(CONFIG_SCALES) |
117 | strncpy(commandp, CONFIG_BOOTPARAM_STRING, size); | ||
118 | commandp[size-1] = 0; | ||
119 | #elif defined(CONFIG_NETtel) || defined(CONFIG_SCALES) | ||
120 | /* Copy command line from FLASH to local buffer... */ | 116 | /* Copy command line from FLASH to local buffer... */ |
121 | memcpy(commandp, (char *) 0xf0004000, size); | 117 | memcpy(commandp, (char *) 0xf0004000, size); |
122 | commandp[size-1] = 0; | 118 | commandp[size-1] = 0; |
@@ -128,8 +124,6 @@ void config_BSP(char *commandp, int size) | |||
128 | /* Copy command line from FLASH to local buffer... */ | 124 | /* Copy command line from FLASH to local buffer... */ |
129 | memcpy(commandp, (char *) 0xf0010000, size); | 125 | memcpy(commandp, (char *) 0xf0010000, size); |
130 | commandp[size-1] = 0; | 126 | commandp[size-1] = 0; |
131 | #else | ||
132 | memset(commandp, 0, size); | ||
133 | #endif | 127 | #endif |
134 | 128 | ||
135 | mcf_timervector = 69; | 129 | mcf_timervector = 69; |
@@ -137,7 +131,6 @@ void config_BSP(char *commandp, int size) | |||
137 | mach_sched_init = coldfire_timer_init; | 131 | mach_sched_init = coldfire_timer_init; |
138 | mach_tick = coldfire_tick; | 132 | mach_tick = coldfire_tick; |
139 | mach_gettimeoffset = coldfire_timer_offset; | 133 | mach_gettimeoffset = coldfire_timer_offset; |
140 | mach_trap_init = coldfire_trap_init; | ||
141 | mach_reset = coldfire_reset; | 134 | mach_reset = coldfire_reset; |
142 | } | 135 | } |
143 | 136 | ||
diff --git a/arch/m68knommu/platform/527x/config.c b/arch/m68knommu/platform/527x/config.c index 28e7d964eef1..1b820441419a 100644 --- a/arch/m68knommu/platform/527x/config.c +++ b/arch/m68knommu/platform/527x/config.c | |||
@@ -29,7 +29,6 @@ | |||
29 | void coldfire_pit_tick(void); | 29 | void coldfire_pit_tick(void); |
30 | void coldfire_pit_init(irq_handler_t handler); | 30 | void coldfire_pit_init(irq_handler_t handler); |
31 | unsigned long coldfire_pit_offset(void); | 31 | unsigned long coldfire_pit_offset(void); |
32 | void coldfire_trap_init(void); | ||
33 | void coldfire_reset(void); | 32 | void coldfire_reset(void); |
34 | 33 | ||
35 | /***************************************************************************/ | 34 | /***************************************************************************/ |
@@ -63,18 +62,9 @@ void mcf_autovector(unsigned int vec) | |||
63 | void config_BSP(char *commandp, int size) | 62 | void config_BSP(char *commandp, int size) |
64 | { | 63 | { |
65 | mcf_disableall(); | 64 | mcf_disableall(); |
66 | |||
67 | #ifdef CONFIG_BOOTPARAM | ||
68 | strncpy(commandp, CONFIG_BOOTPARAM_STRING, size); | ||
69 | commandp[size-1] = 0; | ||
70 | #else | ||
71 | memset(commandp, 0, size); | ||
72 | #endif | ||
73 | |||
74 | mach_sched_init = coldfire_pit_init; | 65 | mach_sched_init = coldfire_pit_init; |
75 | mach_tick = coldfire_pit_tick; | 66 | mach_tick = coldfire_pit_tick; |
76 | mach_gettimeoffset = coldfire_pit_offset; | 67 | mach_gettimeoffset = coldfire_pit_offset; |
77 | mach_trap_init = coldfire_trap_init; | ||
78 | mach_reset = coldfire_reset; | 68 | mach_reset = coldfire_reset; |
79 | } | 69 | } |
80 | 70 | ||
diff --git a/arch/m68knommu/platform/528x/config.c b/arch/m68knommu/platform/528x/config.c index 805b4f74ff19..a089e9513699 100644 --- a/arch/m68knommu/platform/528x/config.c +++ b/arch/m68knommu/platform/528x/config.c | |||
@@ -29,7 +29,6 @@ | |||
29 | void coldfire_pit_tick(void); | 29 | void coldfire_pit_tick(void); |
30 | void coldfire_pit_init(irq_handler_t handler); | 30 | void coldfire_pit_init(irq_handler_t handler); |
31 | unsigned long coldfire_pit_offset(void); | 31 | unsigned long coldfire_pit_offset(void); |
32 | void coldfire_trap_init(void); | ||
33 | void coldfire_reset(void); | 32 | void coldfire_reset(void); |
34 | 33 | ||
35 | /***************************************************************************/ | 34 | /***************************************************************************/ |
@@ -63,18 +62,9 @@ void mcf_autovector(unsigned int vec) | |||
63 | void config_BSP(char *commandp, int size) | 62 | void config_BSP(char *commandp, int size) |
64 | { | 63 | { |
65 | mcf_disableall(); | 64 | mcf_disableall(); |
66 | |||
67 | #ifdef CONFIG_BOOTPARAM | ||
68 | strncpy(commandp, CONFIG_BOOTPARAM_STRING, size); | ||
69 | commandp[size-1] = 0; | ||
70 | #else | ||
71 | memset(commandp, 0, size); | ||
72 | #endif | ||
73 | |||
74 | mach_sched_init = coldfire_pit_init; | 65 | mach_sched_init = coldfire_pit_init; |
75 | mach_tick = coldfire_pit_tick; | 66 | mach_tick = coldfire_pit_tick; |
76 | mach_gettimeoffset = coldfire_pit_offset; | 67 | mach_gettimeoffset = coldfire_pit_offset; |
77 | mach_trap_init = coldfire_trap_init; | ||
78 | mach_reset = coldfire_reset; | 68 | mach_reset = coldfire_reset; |
79 | } | 69 | } |
80 | 70 | ||
diff --git a/arch/m68knommu/platform/5307/config.c b/arch/m68knommu/platform/5307/config.c index e04b84deb57d..e3461619fd65 100644 --- a/arch/m68knommu/platform/5307/config.c +++ b/arch/m68knommu/platform/5307/config.c | |||
@@ -29,7 +29,6 @@ | |||
29 | void coldfire_tick(void); | 29 | void coldfire_tick(void); |
30 | void coldfire_timer_init(irq_handler_t handler); | 30 | void coldfire_timer_init(irq_handler_t handler); |
31 | unsigned long coldfire_timer_offset(void); | 31 | unsigned long coldfire_timer_offset(void); |
32 | void coldfire_trap_init(void); | ||
33 | void coldfire_reset(void); | 32 | void coldfire_reset(void); |
34 | 33 | ||
35 | extern unsigned int mcf_timervector; | 34 | extern unsigned int mcf_timervector; |
@@ -111,10 +110,7 @@ void config_BSP(char *commandp, int size) | |||
111 | { | 110 | { |
112 | mcf_setimr(MCFSIM_IMR_MASKALL); | 111 | mcf_setimr(MCFSIM_IMR_MASKALL); |
113 | 112 | ||
114 | #if defined(CONFIG_BOOTPARAM) | 113 | #if defined(CONFIG_NETtel) || defined(CONFIG_eLIA) || \ |
115 | strncpy(commandp, CONFIG_BOOTPARAM_STRING, size); | ||
116 | commandp[size-1] = 0; | ||
117 | #elif defined(CONFIG_NETtel) || defined(CONFIG_eLIA) || \ | ||
118 | defined(CONFIG_DISKtel) || defined(CONFIG_SECUREEDGEMP3) || \ | 114 | defined(CONFIG_DISKtel) || defined(CONFIG_SECUREEDGEMP3) || \ |
119 | defined(CONFIG_CLEOPATRA) | 115 | defined(CONFIG_CLEOPATRA) |
120 | /* Copy command line from FLASH to local buffer... */ | 116 | /* Copy command line from FLASH to local buffer... */ |
@@ -124,14 +120,11 @@ void config_BSP(char *commandp, int size) | |||
124 | mcf_timervector = 30; | 120 | mcf_timervector = 30; |
125 | mcf_profilevector = 31; | 121 | mcf_profilevector = 31; |
126 | mcf_timerlevel = 6; | 122 | mcf_timerlevel = 6; |
127 | #else | ||
128 | memset(commandp, 0, size); | ||
129 | #endif | 123 | #endif |
130 | 124 | ||
131 | mach_sched_init = coldfire_timer_init; | 125 | mach_sched_init = coldfire_timer_init; |
132 | mach_tick = coldfire_tick; | 126 | mach_tick = coldfire_tick; |
133 | mach_gettimeoffset = coldfire_timer_offset; | 127 | mach_gettimeoffset = coldfire_timer_offset; |
134 | mach_trap_init = coldfire_trap_init; | ||
135 | mach_reset = coldfire_reset; | 128 | mach_reset = coldfire_reset; |
136 | 129 | ||
137 | #ifdef MCF_BDM_DISABLE | 130 | #ifdef MCF_BDM_DISABLE |
diff --git a/arch/m68knommu/platform/5307/entry.S b/arch/m68knommu/platform/5307/entry.S index c358aebe0af3..a8cd867805ca 100644 --- a/arch/m68knommu/platform/5307/entry.S +++ b/arch/m68knommu/platform/5307/entry.S | |||
@@ -213,16 +213,12 @@ ENTRY(ret_from_interrupt) | |||
213 | * Beware - when entering resume, prev (the current task) is | 213 | * Beware - when entering resume, prev (the current task) is |
214 | * in a0, next (the new task) is in a1,so don't change these | 214 | * in a0, next (the new task) is in a1,so don't change these |
215 | * registers until their contents are no longer needed. | 215 | * registers until their contents are no longer needed. |
216 | * This is always called in supervisor mode, so don't bother to save | ||
217 | * and restore sr; user's process sr is actually in the stack. | ||
216 | */ | 218 | */ |
217 | ENTRY(resume) | 219 | ENTRY(resume) |
218 | movel %a0, %d1 /* get prev thread in d1 */ | 220 | movel %a0, %d1 /* get prev thread in d1 */ |
219 | 221 | ||
220 | movew %sr,%d0 /* save thread status reg */ | ||
221 | movew %d0,%a0@(TASK_THREAD+THREAD_SR) | ||
222 | |||
223 | oril #0x700,%d0 /* disable interrupts */ | ||
224 | move %d0,%sr | ||
225 | |||
226 | movel sw_usp,%d0 /* save usp */ | 222 | movel sw_usp,%d0 /* save usp */ |
227 | movel %d0,%a0@(TASK_THREAD+THREAD_USP) | 223 | movel %d0,%a0@(TASK_THREAD+THREAD_USP) |
228 | 224 | ||
@@ -233,7 +229,4 @@ ENTRY(resume) | |||
233 | 229 | ||
234 | movel %a1@(TASK_THREAD+THREAD_USP),%a0 /* restore thread user stack */ | 230 | movel %a1@(TASK_THREAD+THREAD_USP),%a0 /* restore thread user stack */ |
235 | movel %a0, sw_usp | 231 | movel %a0, sw_usp |
236 | |||
237 | movew %a1@(TASK_THREAD+THREAD_SR),%d0 /* restore thread status reg */ | ||
238 | movew %d0, %sr | ||
239 | rts | 232 | rts |
diff --git a/arch/m68knommu/platform/5307/pit.c b/arch/m68knommu/platform/5307/pit.c index aa15beeb36ca..e53c446d10e4 100644 --- a/arch/m68knommu/platform/5307/pit.c +++ b/arch/m68knommu/platform/5307/pit.c | |||
@@ -5,9 +5,8 @@ | |||
5 | * hardware timer only exists in the Freescale ColdFire | 5 | * hardware timer only exists in the Freescale ColdFire |
6 | * 5270/5271, 5282 and other CPUs. | 6 | * 5270/5271, 5282 and other CPUs. |
7 | * | 7 | * |
8 | * Copyright (C) 1999-2006, Greg Ungerer (gerg@snapgear.com) | 8 | * Copyright (C) 1999-2007, Greg Ungerer (gerg@snapgear.com) |
9 | * Copyright (C) 2001-2004, SnapGear Inc. (www.snapgear.com) | 9 | * Copyright (C) 2001-2004, SnapGear Inc. (www.snapgear.com) |
10 | * | ||
11 | */ | 10 | */ |
12 | 11 | ||
13 | /***************************************************************************/ | 12 | /***************************************************************************/ |
@@ -17,8 +16,8 @@ | |||
17 | #include <linux/param.h> | 16 | #include <linux/param.h> |
18 | #include <linux/init.h> | 17 | #include <linux/init.h> |
19 | #include <linux/interrupt.h> | 18 | #include <linux/interrupt.h> |
19 | #include <linux/irq.h> | ||
20 | #include <asm/io.h> | 20 | #include <asm/io.h> |
21 | #include <asm/irq.h> | ||
22 | #include <asm/coldfire.h> | 21 | #include <asm/coldfire.h> |
23 | #include <asm/mcfpit.h> | 22 | #include <asm/mcfpit.h> |
24 | #include <asm/mcfsim.h> | 23 | #include <asm/mcfsim.h> |
@@ -43,13 +42,18 @@ void coldfire_pit_tick(void) | |||
43 | 42 | ||
44 | /***************************************************************************/ | 43 | /***************************************************************************/ |
45 | 44 | ||
45 | static struct irqaction coldfire_pit_irq = { | ||
46 | .name = "timer", | ||
47 | .flags = IRQF_DISABLED | IRQF_TIMER, | ||
48 | }; | ||
49 | |||
46 | void coldfire_pit_init(irq_handler_t handler) | 50 | void coldfire_pit_init(irq_handler_t handler) |
47 | { | 51 | { |
48 | volatile unsigned char *icrp; | 52 | volatile unsigned char *icrp; |
49 | volatile unsigned long *imrp; | 53 | volatile unsigned long *imrp; |
50 | 54 | ||
51 | request_irq(MCFINT_VECBASE + MCFINT_PIT1, handler, IRQF_DISABLED, | 55 | coldfire_pit_irq.handler = handler; |
52 | "ColdFire Timer", NULL); | 56 | setup_irq(MCFINT_VECBASE + MCFINT_PIT1, &coldfire_pit_irq); |
53 | 57 | ||
54 | icrp = (volatile unsigned char *) (MCF_IPSBAR + MCFICM_INTC0 + | 58 | icrp = (volatile unsigned char *) (MCF_IPSBAR + MCFICM_INTC0 + |
55 | MCFINTC_ICR0 + MCFINT_PIT1); | 59 | MCFINTC_ICR0 + MCFINT_PIT1); |
diff --git a/arch/m68knommu/platform/5307/timers.c b/arch/m68knommu/platform/5307/timers.c index fb66eadd5896..64bd0ff9029e 100644 --- a/arch/m68knommu/platform/5307/timers.c +++ b/arch/m68knommu/platform/5307/timers.c | |||
@@ -3,7 +3,7 @@ | |||
3 | /* | 3 | /* |
4 | * timers.c -- generic ColdFire hardware timer support. | 4 | * timers.c -- generic ColdFire hardware timer support. |
5 | * | 5 | * |
6 | * Copyright (C) 1999-2006, Greg Ungerer (gerg@snapgear.com) | 6 | * Copyright (C) 1999-2007, Greg Ungerer (gerg@snapgear.com) |
7 | */ | 7 | */ |
8 | 8 | ||
9 | /***************************************************************************/ | 9 | /***************************************************************************/ |
@@ -13,8 +13,8 @@ | |||
13 | #include <linux/param.h> | 13 | #include <linux/param.h> |
14 | #include <linux/interrupt.h> | 14 | #include <linux/interrupt.h> |
15 | #include <linux/init.h> | 15 | #include <linux/init.h> |
16 | #include <linux/irq.h> | ||
16 | #include <asm/io.h> | 17 | #include <asm/io.h> |
17 | #include <asm/irq.h> | ||
18 | #include <asm/traps.h> | 18 | #include <asm/traps.h> |
19 | #include <asm/machdep.h> | 19 | #include <asm/machdep.h> |
20 | #include <asm/coldfire.h> | 20 | #include <asm/coldfire.h> |
@@ -62,17 +62,24 @@ void coldfire_tick(void) | |||
62 | 62 | ||
63 | /***************************************************************************/ | 63 | /***************************************************************************/ |
64 | 64 | ||
65 | static struct irqaction coldfire_timer_irq = { | ||
66 | .name = "timer", | ||
67 | .flags = IRQF_DISABLED | IRQF_TIMER, | ||
68 | }; | ||
69 | |||
65 | static int ticks_per_intr; | 70 | static int ticks_per_intr; |
66 | 71 | ||
67 | void coldfire_timer_init(irq_handler_t handler) | 72 | void coldfire_timer_init(irq_handler_t handler) |
68 | { | 73 | { |
74 | coldfire_timer_irq.handler = handler; | ||
75 | setup_irq(mcf_timervector, &coldfire_timer_irq); | ||
76 | |||
69 | __raw_writew(MCFTIMER_TMR_DISABLE, TA(MCFTIMER_TMR)); | 77 | __raw_writew(MCFTIMER_TMR_DISABLE, TA(MCFTIMER_TMR)); |
70 | ticks_per_intr = (MCF_BUSCLK / 16) / HZ; | 78 | ticks_per_intr = (MCF_BUSCLK / 16) / HZ; |
71 | __raw_writetrr(ticks_per_intr - 1, TA(MCFTIMER_TRR)); | 79 | __raw_writetrr(ticks_per_intr - 1, TA(MCFTIMER_TRR)); |
72 | __raw_writew(MCFTIMER_TMR_ENORI | MCFTIMER_TMR_CLK16 | | 80 | __raw_writew(MCFTIMER_TMR_ENORI | MCFTIMER_TMR_CLK16 | |
73 | MCFTIMER_TMR_RESTART | MCFTIMER_TMR_ENABLE, TA(MCFTIMER_TMR)); | 81 | MCFTIMER_TMR_RESTART | MCFTIMER_TMR_ENABLE, TA(MCFTIMER_TMR)); |
74 | 82 | ||
75 | request_irq(mcf_timervector, handler, IRQF_DISABLED, "timer", NULL); | ||
76 | mcf_settimericr(1, mcf_timerlevel); | 83 | mcf_settimericr(1, mcf_timerlevel); |
77 | 84 | ||
78 | #ifdef CONFIG_HIGHPROFILE | 85 | #ifdef CONFIG_HIGHPROFILE |
diff --git a/arch/m68knommu/platform/532x/config.c b/arch/m68knommu/platform/532x/config.c index 664c3a12b0c1..b32c6425f821 100644 --- a/arch/m68knommu/platform/532x/config.c +++ b/arch/m68knommu/platform/532x/config.c | |||
@@ -37,7 +37,6 @@ | |||
37 | void coldfire_tick(void); | 37 | void coldfire_tick(void); |
38 | void coldfire_timer_init(irq_handler_t handler); | 38 | void coldfire_timer_init(irq_handler_t handler); |
39 | unsigned long coldfire_timer_offset(void); | 39 | unsigned long coldfire_timer_offset(void); |
40 | void coldfire_trap_init(void); | ||
41 | void coldfire_reset(void); | 40 | void coldfire_reset(void); |
42 | 41 | ||
43 | extern unsigned int mcf_timervector; | 42 | extern unsigned int mcf_timervector; |
@@ -92,10 +91,7 @@ void config_BSP(char *commandp, int size) | |||
92 | { | 91 | { |
93 | mcf_setimr(MCFSIM_IMR_MASKALL); | 92 | mcf_setimr(MCFSIM_IMR_MASKALL); |
94 | 93 | ||
95 | #if defined(CONFIG_BOOTPARAM) | 94 | #if !defined(CONFIG_BOOTPARAM) |
96 | strncpy(commandp, CONFIG_BOOTPARAM_STRING, size); | ||
97 | commandp[size-1] = 0; | ||
98 | #else | ||
99 | /* Copy command line from FLASH to local buffer... */ | 95 | /* Copy command line from FLASH to local buffer... */ |
100 | memcpy(commandp, (char *) 0x4000, 4); | 96 | memcpy(commandp, (char *) 0x4000, 4); |
101 | if(strncmp(commandp, "kcl ", 4) == 0){ | 97 | if(strncmp(commandp, "kcl ", 4) == 0){ |
@@ -111,7 +107,6 @@ void config_BSP(char *commandp, int size) | |||
111 | mach_sched_init = coldfire_timer_init; | 107 | mach_sched_init = coldfire_timer_init; |
112 | mach_tick = coldfire_tick; | 108 | mach_tick = coldfire_tick; |
113 | mach_gettimeoffset = coldfire_timer_offset; | 109 | mach_gettimeoffset = coldfire_timer_offset; |
114 | mach_trap_init = coldfire_trap_init; | ||
115 | mach_reset = coldfire_reset; | 110 | mach_reset = coldfire_reset; |
116 | 111 | ||
117 | #ifdef MCF_BDM_DISABLE | 112 | #ifdef MCF_BDM_DISABLE |
diff --git a/arch/m68knommu/platform/5407/config.c b/arch/m68knommu/platform/5407/config.c index 036f62876241..e692536817d8 100644 --- a/arch/m68knommu/platform/5407/config.c +++ b/arch/m68knommu/platform/5407/config.c | |||
@@ -28,7 +28,6 @@ | |||
28 | void coldfire_tick(void); | 28 | void coldfire_tick(void); |
29 | void coldfire_timer_init(irq_handler_t handler); | 29 | void coldfire_timer_init(irq_handler_t handler); |
30 | unsigned long coldfire_timer_offset(void); | 30 | unsigned long coldfire_timer_offset(void); |
31 | void coldfire_trap_init(void); | ||
32 | void coldfire_reset(void); | 31 | void coldfire_reset(void); |
33 | 32 | ||
34 | extern unsigned int mcf_timervector; | 33 | extern unsigned int mcf_timervector; |
@@ -102,13 +101,6 @@ void config_BSP(char *commandp, int size) | |||
102 | { | 101 | { |
103 | mcf_setimr(MCFSIM_IMR_MASKALL); | 102 | mcf_setimr(MCFSIM_IMR_MASKALL); |
104 | 103 | ||
105 | #if defined(CONFIG_BOOTPARAM) | ||
106 | strncpy(commandp, CONFIG_BOOTPARAM_STRING, size); | ||
107 | commandp[size-1] = 0; | ||
108 | #else | ||
109 | memset(commandp, 0, size); | ||
110 | #endif | ||
111 | |||
112 | #if defined(CONFIG_CLEOPATRA) | 104 | #if defined(CONFIG_CLEOPATRA) |
113 | /* Different timer setup - to prevent device clash */ | 105 | /* Different timer setup - to prevent device clash */ |
114 | mcf_timervector = 30; | 106 | mcf_timervector = 30; |
@@ -119,7 +111,6 @@ void config_BSP(char *commandp, int size) | |||
119 | mach_sched_init = coldfire_timer_init; | 111 | mach_sched_init = coldfire_timer_init; |
120 | mach_tick = coldfire_tick; | 112 | mach_tick = coldfire_tick; |
121 | mach_gettimeoffset = coldfire_timer_offset; | 113 | mach_gettimeoffset = coldfire_timer_offset; |
122 | mach_trap_init = coldfire_trap_init; | ||
123 | mach_reset = coldfire_reset; | 114 | mach_reset = coldfire_reset; |
124 | } | 115 | } |
125 | 116 | ||
diff --git a/arch/m68knommu/platform/68328/timers.c b/arch/m68knommu/platform/68328/timers.c index ef067f4c3cd4..0396476f955d 100644 --- a/arch/m68knommu/platform/68328/timers.c +++ b/arch/m68knommu/platform/68328/timers.c | |||
@@ -18,10 +18,10 @@ | |||
18 | #include <linux/kernel.h> | 18 | #include <linux/kernel.h> |
19 | #include <linux/mm.h> | 19 | #include <linux/mm.h> |
20 | #include <linux/interrupt.h> | 20 | #include <linux/interrupt.h> |
21 | #include <linux/irq.h> | ||
21 | #include <asm/setup.h> | 22 | #include <asm/setup.h> |
22 | #include <asm/system.h> | 23 | #include <asm/system.h> |
23 | #include <asm/pgtable.h> | 24 | #include <asm/pgtable.h> |
24 | #include <asm/irq.h> | ||
25 | #include <asm/machdep.h> | 25 | #include <asm/machdep.h> |
26 | #include <asm/MC68VZ328.h> | 26 | #include <asm/MC68VZ328.h> |
27 | 27 | ||
@@ -53,14 +53,19 @@ | |||
53 | 53 | ||
54 | /***************************************************************************/ | 54 | /***************************************************************************/ |
55 | 55 | ||
56 | static struct irqaction m68328_timer_irq = { | ||
57 | .name = "timer", | ||
58 | .flags = IRQF_DISABLED | IRQF_TIMER, | ||
59 | }; | ||
60 | |||
56 | void m68328_timer_init(irq_handler_t timer_routine) | 61 | void m68328_timer_init(irq_handler_t timer_routine) |
57 | { | 62 | { |
58 | /* disable timer 1 */ | 63 | /* disable timer 1 */ |
59 | TCTL = 0; | 64 | TCTL = 0; |
60 | 65 | ||
61 | /* set ISR */ | 66 | /* set ISR */ |
62 | if (request_irq(TMR_IRQ_NUM, timer_routine, IRQ_FLG_LOCK, "timer", NULL)) | 67 | m68328_timer_irq.handler = timer_routine; |
63 | panic("Unable to attach timer interrupt\n"); | 68 | setup_irq(TMR_IRQ_NUM, &m68328_timer_irq); |
64 | 69 | ||
65 | /* Restart mode, Enable int, Set clock source */ | 70 | /* Restart mode, Enable int, Set clock source */ |
66 | TCTL = TCTL_OM | TCTL_IRQEN | CLOCK_SOURCE; | 71 | TCTL = TCTL_OM | TCTL_IRQEN | CLOCK_SOURCE; |
diff --git a/arch/m68knommu/platform/68360/config.c b/arch/m68knommu/platform/68360/config.c index 4ff13bd51ffd..155b72fe2607 100644 --- a/arch/m68knommu/platform/68360/config.c +++ b/arch/m68knommu/platform/68360/config.c | |||
@@ -17,11 +17,11 @@ | |||
17 | #include <linux/tty.h> | 17 | #include <linux/tty.h> |
18 | #include <linux/console.h> | 18 | #include <linux/console.h> |
19 | #include <linux/interrupt.h> | 19 | #include <linux/interrupt.h> |
20 | #include <linux/irq.h> | ||
20 | 21 | ||
21 | #include <asm/setup.h> | 22 | #include <asm/setup.h> |
22 | #include <asm/system.h> | 23 | #include <asm/system.h> |
23 | #include <asm/pgtable.h> | 24 | #include <asm/pgtable.h> |
24 | #include <asm/irq.h> | ||
25 | #include <asm/machdep.h> | 25 | #include <asm/machdep.h> |
26 | #include <asm/m68360.h> | 26 | #include <asm/m68360.h> |
27 | 27 | ||
@@ -51,11 +51,15 @@ extern unsigned long int system_clock; //In kernel setup.c | |||
51 | 51 | ||
52 | extern void config_M68360_irq(void); | 52 | extern void config_M68360_irq(void); |
53 | 53 | ||
54 | static struct irqaction m68360_timer_irq = { | ||
55 | .name = "timer", | ||
56 | .flags = IRQF_DISABLED | IRQF_TIMER, | ||
57 | }; | ||
58 | |||
54 | void BSP_sched_init(irq_handler_t timer_routine) | 59 | void BSP_sched_init(irq_handler_t timer_routine) |
55 | { | 60 | { |
56 | unsigned char prescaler; | 61 | unsigned char prescaler; |
57 | unsigned short tgcr_save; | 62 | unsigned short tgcr_save; |
58 | int return_value; | ||
59 | 63 | ||
60 | #if 0 | 64 | #if 0 |
61 | /* Restart mode, Enable int, 32KHz, Enable timer */ | 65 | /* Restart mode, Enable int, 32KHz, Enable timer */ |
@@ -86,10 +90,8 @@ void BSP_sched_init(irq_handler_t timer_routine) | |||
86 | pquicc->timer_ter1 = 0x0003; /* clear timer events */ | 90 | pquicc->timer_ter1 = 0x0003; /* clear timer events */ |
87 | 91 | ||
88 | /* enable timer 1 interrupt in CIMR */ | 92 | /* enable timer 1 interrupt in CIMR */ |
89 | // request_irq(IRQ_MACHSPEC | CPMVEC_TIMER1, timer_routine, IRQ_FLG_LOCK, "timer", NULL); | 93 | m68360_timer_irq.handler = timer_routine; |
90 | //return_value = request_irq( CPMVEC_TIMER1, timer_routine, IRQ_FLG_LOCK, "timer", NULL); | 94 | setup_irq(CPMVEC_TIMER1, &m68360_timer_irq); |
91 | return_value = request_irq(CPMVEC_TIMER1 , timer_routine, IRQ_FLG_LOCK, | ||
92 | "Timer", NULL); | ||
93 | 95 | ||
94 | /* Start timer 1: */ | 96 | /* Start timer 1: */ |
95 | tgcr_save = (pquicc->timer_tgcr & 0xfff0) | 0x0001; | 97 | tgcr_save = (pquicc->timer_tgcr & 0xfff0) | 0x0001; |
diff --git a/arch/m68knommu/platform/68VZ328/config.c b/arch/m68knommu/platform/68VZ328/config.c index 8abe0f6e7235..79dced929c97 100644 --- a/arch/m68knommu/platform/68VZ328/config.c +++ b/arch/m68knommu/platform/68VZ328/config.c | |||
@@ -191,13 +191,6 @@ void config_BSP(char *command, int size) | |||
191 | { | 191 | { |
192 | printk(KERN_INFO "68VZ328 DragonBallVZ support (c) 2001 Lineo, Inc.\n"); | 192 | printk(KERN_INFO "68VZ328 DragonBallVZ support (c) 2001 Lineo, Inc.\n"); |
193 | 193 | ||
194 | #if defined(CONFIG_BOOTPARAM) | ||
195 | strncpy(command, CONFIG_BOOTPARAM_STRING, size); | ||
196 | command[size-1] = 0; | ||
197 | #else | ||
198 | memset(command, 0, size); | ||
199 | #endif | ||
200 | |||
201 | init_hardware(command, size); | 194 | init_hardware(command, size); |
202 | 195 | ||
203 | mach_sched_init = (void *) m68328_timer_init; | 196 | mach_sched_init = (void *) m68328_timer_init; |
diff --git a/arch/mips/Makefile b/arch/mips/Makefile index 20d19c9b7761..a9a987a06daf 100644 --- a/arch/mips/Makefile +++ b/arch/mips/Makefile | |||
@@ -328,7 +328,7 @@ load-$(CONFIG_MIPS_SEAD) += 0xffffffff80100000 | |||
328 | # MIPS SIM | 328 | # MIPS SIM |
329 | # | 329 | # |
330 | core-$(CONFIG_MIPS_SIM) += arch/mips/mipssim/ | 330 | core-$(CONFIG_MIPS_SIM) += arch/mips/mipssim/ |
331 | cflags-$(CONFIG_MIPS_SIM) += -Iinclude/asm-mips/mach-sim | 331 | cflags-$(CONFIG_MIPS_SIM) += -Iinclude/asm-mips/mach-mipssim |
332 | load-$(CONFIG_MIPS_SIM) += 0x80100000 | 332 | load-$(CONFIG_MIPS_SIM) += 0x80100000 |
333 | 333 | ||
334 | # | 334 | # |
diff --git a/arch/mips/arc/console.c b/arch/mips/arc/console.c deleted file mode 100644 index 0fe6032999cb..000000000000 --- a/arch/mips/arc/console.c +++ /dev/null | |||
@@ -1,31 +0,0 @@ | |||
1 | /* | ||
2 | * This file is subject to the terms and conditions of the GNU General Public | ||
3 | * License. See the file "COPYING" in the main directory of this archive | ||
4 | * for more details. | ||
5 | * | ||
6 | * Copyright (C) 1996 David S. Miller (dm@sgi.com) | ||
7 | * Compability with board caches, Ulf Carlsson | ||
8 | */ | ||
9 | #include <linux/kernel.h> | ||
10 | #include <asm/sgialib.h> | ||
11 | #include <asm/bcache.h> | ||
12 | |||
13 | /* | ||
14 | * IP22 boardcache is not compatible with board caches. Thus we disable it | ||
15 | * during romvec action. Since r4xx0.c is always compiled and linked with your | ||
16 | * kernel, this shouldn't cause any harm regardless what MIPS processor you | ||
17 | * have. | ||
18 | * | ||
19 | * The ARC write and read functions seem to interfere with the serial lines | ||
20 | * in some way. You should be careful with them. | ||
21 | */ | ||
22 | |||
23 | void prom_putchar(char c) | ||
24 | { | ||
25 | ULONG cnt; | ||
26 | CHAR it = c; | ||
27 | |||
28 | bc_disable(); | ||
29 | ArcWrite(1, &it, 1, &cnt); | ||
30 | bc_enable(); | ||
31 | } | ||
diff --git a/arch/mips/jazz/io.c b/arch/mips/jazz/io.c deleted file mode 100644 index e86904454c89..000000000000 --- a/arch/mips/jazz/io.c +++ /dev/null | |||
@@ -1,135 +0,0 @@ | |||
1 | /* | ||
2 | * This file is subject to the terms and conditions of the GNU General Public | ||
3 | * License. See the file "COPYING" in the main directory of this archive | ||
4 | * for more details. | ||
5 | * | ||
6 | * Low level I/O functions for Jazz family machines. | ||
7 | * | ||
8 | * Copyright (C) 1997 by Ralf Baechle. | ||
9 | */ | ||
10 | #include <linux/string.h> | ||
11 | #include <linux/spinlock.h> | ||
12 | #include <asm/addrspace.h> | ||
13 | #include <asm/system.h> | ||
14 | #include <asm/jazz.h> | ||
15 | |||
16 | /* | ||
17 | * Map an 16mb segment of the EISA address space to 0xe3000000; | ||
18 | */ | ||
19 | static inline void map_eisa_address(unsigned long address) | ||
20 | { | ||
21 | /* XXX */ | ||
22 | /* We've got an wired entry in the TLB. We just need to modify it. | ||
23 | fast and clean. But since we want to get rid of wired entries | ||
24 | things are a little bit more complicated ... */ | ||
25 | } | ||
26 | |||
27 | static unsigned char jazz_readb(unsigned long addr) | ||
28 | { | ||
29 | unsigned char res; | ||
30 | |||
31 | map_eisa_address(addr); | ||
32 | addr &= 0xffffff; | ||
33 | res = *(volatile unsigned char *) (JAZZ_EISA_BASE + addr); | ||
34 | |||
35 | return res; | ||
36 | } | ||
37 | |||
38 | static unsigned short jazz_readw(unsigned long addr) | ||
39 | { | ||
40 | unsigned short res; | ||
41 | |||
42 | map_eisa_address(addr); | ||
43 | addr &= 0xffffff; | ||
44 | res = *(volatile unsigned char *) (JAZZ_EISA_BASE + addr); | ||
45 | |||
46 | return res; | ||
47 | } | ||
48 | |||
49 | static unsigned int jazz_readl(unsigned long addr) | ||
50 | { | ||
51 | unsigned int res; | ||
52 | |||
53 | map_eisa_address(addr); | ||
54 | addr &= 0xffffff; | ||
55 | res = *(volatile unsigned char *) (JAZZ_EISA_BASE + addr); | ||
56 | |||
57 | return res; | ||
58 | } | ||
59 | |||
60 | static void jazz_writeb(unsigned char val, unsigned long addr) | ||
61 | { | ||
62 | map_eisa_address(addr); | ||
63 | addr &= 0xffffff; | ||
64 | *(volatile unsigned char *) (JAZZ_EISA_BASE + addr) = val; | ||
65 | } | ||
66 | |||
67 | static void jazz_writew(unsigned short val, unsigned long addr) | ||
68 | { | ||
69 | map_eisa_address(addr); | ||
70 | addr &= 0xffffff; | ||
71 | *(volatile unsigned char *) (JAZZ_EISA_BASE + addr) = val; | ||
72 | } | ||
73 | |||
74 | static void jazz_writel(unsigned int val, unsigned long addr) | ||
75 | { | ||
76 | map_eisa_address(addr); | ||
77 | addr &= 0xffffff; | ||
78 | *(volatile unsigned char *) (JAZZ_EISA_BASE + addr) = val; | ||
79 | } | ||
80 | |||
81 | static void jazz_memset_io(unsigned long addr, int val, unsigned long len) | ||
82 | { | ||
83 | unsigned long waddr; | ||
84 | |||
85 | waddr = JAZZ_EISA_BASE | (addr & 0xffffff); | ||
86 | while(len) { | ||
87 | unsigned long fraglen; | ||
88 | |||
89 | fraglen = (~addr + 1) & 0xffffff; | ||
90 | fraglen = (fraglen < len) ? fraglen : len; | ||
91 | map_eisa_address(addr); | ||
92 | memset((char *)waddr, val, fraglen); | ||
93 | addr += fraglen; | ||
94 | waddr = waddr + fraglen - 0x1000000; | ||
95 | len -= fraglen; | ||
96 | } | ||
97 | } | ||
98 | |||
99 | static void jazz_memcpy_fromio(unsigned long to, unsigned long from, unsigned long len) | ||
100 | { | ||
101 | unsigned long waddr; | ||
102 | |||
103 | waddr = JAZZ_EISA_BASE | (from & 0xffffff); | ||
104 | while(len) { | ||
105 | unsigned long fraglen; | ||
106 | |||
107 | fraglen = (~from + 1) & 0xffffff; | ||
108 | fraglen = (fraglen < len) ? fraglen : len; | ||
109 | map_eisa_address(from); | ||
110 | memcpy((void *)to, (void *)waddr, fraglen); | ||
111 | to += fraglen; | ||
112 | from += fraglen; | ||
113 | waddr = waddr + fraglen - 0x1000000; | ||
114 | len -= fraglen; | ||
115 | } | ||
116 | } | ||
117 | |||
118 | static void jazz_memcpy_toio(unsigned long to, unsigned long from, unsigned long len) | ||
119 | { | ||
120 | unsigned long waddr; | ||
121 | |||
122 | waddr = JAZZ_EISA_BASE | (to & 0xffffff); | ||
123 | while(len) { | ||
124 | unsigned long fraglen; | ||
125 | |||
126 | fraglen = (~to + 1) & 0xffffff; | ||
127 | fraglen = (fraglen < len) ? fraglen : len; | ||
128 | map_eisa_address(to); | ||
129 | memcpy((char *)to + JAZZ_EISA_BASE, (void *)from, fraglen); | ||
130 | to += fraglen; | ||
131 | from += fraglen; | ||
132 | waddr = waddr + fraglen - 0x1000000; | ||
133 | len -= fraglen; | ||
134 | } | ||
135 | } | ||
diff --git a/arch/mips/jazz/reset.c b/arch/mips/jazz/reset.c index 2a9754750bc8..d8ade85060b3 100644 --- a/arch/mips/jazz/reset.c +++ b/arch/mips/jazz/reset.c | |||
@@ -6,10 +6,6 @@ | |||
6 | */ | 6 | */ |
7 | #include <linux/jiffies.h> | 7 | #include <linux/jiffies.h> |
8 | #include <asm/jazz.h> | 8 | #include <asm/jazz.h> |
9 | #include <asm/io.h> | ||
10 | #include <asm/system.h> | ||
11 | #include <asm/reboot.h> | ||
12 | #include <asm/delay.h> | ||
13 | 9 | ||
14 | #define KBD_STAT_IBF 0x02 /* Keyboard input buffer full */ | 10 | #define KBD_STAT_IBF 0x02 /* Keyboard input buffer full */ |
15 | 11 | ||
@@ -58,12 +54,3 @@ void jazz_machine_restart(char *command) | |||
58 | jazz_write_output (0x00); | 54 | jazz_write_output (0x00); |
59 | } | 55 | } |
60 | } | 56 | } |
61 | |||
62 | void jazz_machine_halt(void) | ||
63 | { | ||
64 | } | ||
65 | |||
66 | void jazz_machine_power_off(void) | ||
67 | { | ||
68 | /* Jazz machines don't have a software power switch */ | ||
69 | } | ||
diff --git a/arch/mips/jazz/setup.c b/arch/mips/jazz/setup.c index 81ec559a1c26..798279e06691 100644 --- a/arch/mips/jazz/setup.c +++ b/arch/mips/jazz/setup.c | |||
@@ -34,8 +34,6 @@ | |||
34 | extern asmlinkage void jazz_handle_int(void); | 34 | extern asmlinkage void jazz_handle_int(void); |
35 | 35 | ||
36 | extern void jazz_machine_restart(char *command); | 36 | extern void jazz_machine_restart(char *command); |
37 | extern void jazz_machine_halt(void); | ||
38 | extern void jazz_machine_power_off(void); | ||
39 | 37 | ||
40 | void __init plat_timer_setup(struct irqaction *irq) | 38 | void __init plat_timer_setup(struct irqaction *irq) |
41 | { | 39 | { |
@@ -95,8 +93,6 @@ void __init plat_mem_setup(void) | |||
95 | /* The RTC is outside the port address space */ | 93 | /* The RTC is outside the port address space */ |
96 | 94 | ||
97 | _machine_restart = jazz_machine_restart; | 95 | _machine_restart = jazz_machine_restart; |
98 | _machine_halt = jazz_machine_halt; | ||
99 | pm_power_off = jazz_machine_power_off; | ||
100 | 96 | ||
101 | screen_info = (struct screen_info) { | 97 | screen_info = (struct screen_info) { |
102 | 0, 0, /* orig-x, orig-y */ | 98 | 0, 0, /* orig-x, orig-y */ |
diff --git a/arch/mips/jmr3927/rbhma3100/setup.c b/arch/mips/jmr3927/rbhma3100/setup.c index d1ef2895d564..8303001516d2 100644 --- a/arch/mips/jmr3927/rbhma3100/setup.c +++ b/arch/mips/jmr3927/rbhma3100/setup.c | |||
@@ -434,7 +434,7 @@ EXPORT_SYMBOL(__swizzle_addr_b); | |||
434 | 434 | ||
435 | static int __init jmr3927_rtc_init(void) | 435 | static int __init jmr3927_rtc_init(void) |
436 | { | 436 | { |
437 | struct resource res = { | 437 | static struct resource __initdata res = { |
438 | .start = JMR3927_IOC_NVRAMB_ADDR - IO_BASE, | 438 | .start = JMR3927_IOC_NVRAMB_ADDR - IO_BASE, |
439 | .end = JMR3927_IOC_NVRAMB_ADDR - IO_BASE + 0x800 - 1, | 439 | .end = JMR3927_IOC_NVRAMB_ADDR - IO_BASE + 0x800 - 1, |
440 | .flags = IORESOURCE_MEM, | 440 | .flags = IORESOURCE_MEM, |
diff --git a/arch/mips/kernel/gdb-stub.c b/arch/mips/kernel/gdb-stub.c index 7bc882049269..cb5623aad552 100644 --- a/arch/mips/kernel/gdb-stub.c +++ b/arch/mips/kernel/gdb-stub.c | |||
@@ -1099,12 +1099,12 @@ void adel(void) | |||
1099 | * malloc is needed by gdb client in "call func()", even a private one | 1099 | * malloc is needed by gdb client in "call func()", even a private one |
1100 | * will make gdb happy | 1100 | * will make gdb happy |
1101 | */ | 1101 | */ |
1102 | static void * __attribute_used__ malloc(size_t size) | 1102 | static void __used *malloc(size_t size) |
1103 | { | 1103 | { |
1104 | return kmalloc(size, GFP_ATOMIC); | 1104 | return kmalloc(size, GFP_ATOMIC); |
1105 | } | 1105 | } |
1106 | 1106 | ||
1107 | static void __attribute_used__ free (void *where) | 1107 | static void __used free(void *where) |
1108 | { | 1108 | { |
1109 | kfree(where); | 1109 | kfree(where); |
1110 | } | 1110 | } |
diff --git a/arch/mips/kernel/head.S b/arch/mips/kernel/head.S index f78538eceef7..c15bbc436bbd 100644 --- a/arch/mips/kernel/head.S +++ b/arch/mips/kernel/head.S | |||
@@ -141,7 +141,7 @@ | |||
141 | EXPORT(stext) # used for profiling | 141 | EXPORT(stext) # used for profiling |
142 | EXPORT(_stext) | 142 | EXPORT(_stext) |
143 | 143 | ||
144 | #ifdef CONFIG_BOOT_RAW | 144 | #ifndef CONFIG_BOOT_RAW |
145 | /* | 145 | /* |
146 | * Give us a fighting chance of running if execution beings at the | 146 | * Give us a fighting chance of running if execution beings at the |
147 | * kernel load address. This is needed because this platform does | 147 | * kernel load address. This is needed because this platform does |
diff --git a/arch/mips/kernel/linux32.c b/arch/mips/kernel/linux32.c index 06e04da211d5..c37568d6fb55 100644 --- a/arch/mips/kernel/linux32.c +++ b/arch/mips/kernel/linux32.c | |||
@@ -567,7 +567,7 @@ asmlinkage long sys32_fadvise64_64(int fd, int __pad, | |||
567 | } | 567 | } |
568 | 568 | ||
569 | save_static_function(sys32_clone); | 569 | save_static_function(sys32_clone); |
570 | __attribute_used__ noinline static int | 570 | static int noinline __used |
571 | _sys32_clone(nabi_no_regargs struct pt_regs regs) | 571 | _sys32_clone(nabi_no_regargs struct pt_regs regs) |
572 | { | 572 | { |
573 | unsigned long clone_flags; | 573 | unsigned long clone_flags; |
diff --git a/arch/mips/kernel/rtlx.c b/arch/mips/kernel/rtlx.c index bfc8ca168f83..8cf24d716d41 100644 --- a/arch/mips/kernel/rtlx.c +++ b/arch/mips/kernel/rtlx.c | |||
@@ -85,7 +85,7 @@ static irqreturn_t rtlx_interrupt(int irq, void *dev_id) | |||
85 | return IRQ_HANDLED; | 85 | return IRQ_HANDLED; |
86 | } | 86 | } |
87 | 87 | ||
88 | static __attribute_used__ void dump_rtlx(void) | 88 | static void __used dump_rtlx(void) |
89 | { | 89 | { |
90 | int i; | 90 | int i; |
91 | 91 | ||
diff --git a/arch/mips/kernel/syscall.c b/arch/mips/kernel/syscall.c index b947c61c0cc8..b53f7edbc159 100644 --- a/arch/mips/kernel/syscall.c +++ b/arch/mips/kernel/syscall.c | |||
@@ -167,14 +167,14 @@ sys_mmap2(unsigned long addr, unsigned long len, unsigned long prot, | |||
167 | } | 167 | } |
168 | 168 | ||
169 | save_static_function(sys_fork); | 169 | save_static_function(sys_fork); |
170 | __attribute_used__ noinline static int | 170 | static int __used noinline |
171 | _sys_fork(nabi_no_regargs struct pt_regs regs) | 171 | _sys_fork(nabi_no_regargs struct pt_regs regs) |
172 | { | 172 | { |
173 | return do_fork(SIGCHLD, regs.regs[29], ®s, 0, NULL, NULL); | 173 | return do_fork(SIGCHLD, regs.regs[29], ®s, 0, NULL, NULL); |
174 | } | 174 | } |
175 | 175 | ||
176 | save_static_function(sys_clone); | 176 | save_static_function(sys_clone); |
177 | __attribute_used__ noinline static int | 177 | static int __used noinline |
178 | _sys_clone(nabi_no_regargs struct pt_regs regs) | 178 | _sys_clone(nabi_no_regargs struct pt_regs regs) |
179 | { | 179 | { |
180 | unsigned long clone_flags; | 180 | unsigned long clone_flags; |
diff --git a/arch/mips/kernel/vpe.c b/arch/mips/kernel/vpe.c index 9e66354dee8b..a2bee10f04cf 100644 --- a/arch/mips/kernel/vpe.c +++ b/arch/mips/kernel/vpe.c | |||
@@ -154,7 +154,6 @@ struct { | |||
154 | }; | 154 | }; |
155 | 155 | ||
156 | static void release_progmem(void *ptr); | 156 | static void release_progmem(void *ptr); |
157 | /* static __attribute_used__ void dump_vpe(struct vpe * v); */ | ||
158 | extern void save_gp_address(unsigned int secbase, unsigned int rel); | 157 | extern void save_gp_address(unsigned int secbase, unsigned int rel); |
159 | 158 | ||
160 | /* get the vpe associated with this minor */ | 159 | /* get the vpe associated with this minor */ |
@@ -1024,7 +1023,7 @@ static int vpe_elfload(struct vpe * v) | |||
1024 | return 0; | 1023 | return 0; |
1025 | } | 1024 | } |
1026 | 1025 | ||
1027 | __attribute_used__ void dump_vpe(struct vpe * v) | 1026 | void __used dump_vpe(struct vpe * v) |
1028 | { | 1027 | { |
1029 | struct tc *t; | 1028 | struct tc *t; |
1030 | 1029 | ||
diff --git a/arch/mips/mm/c-sb1.c b/arch/mips/mm/c-sb1.c index 6f9bd7fbd481..85ce2842d0da 100644 --- a/arch/mips/mm/c-sb1.c +++ b/arch/mips/mm/c-sb1.c | |||
@@ -272,7 +272,7 @@ void sb1_flush_cache_data_page(unsigned long) | |||
272 | /* | 272 | /* |
273 | * Invalidate all caches on this CPU | 273 | * Invalidate all caches on this CPU |
274 | */ | 274 | */ |
275 | static void __attribute_used__ local_sb1___flush_cache_all(void) | 275 | static void __used local_sb1___flush_cache_all(void) |
276 | { | 276 | { |
277 | __sb1_writeback_inv_dcache_all(); | 277 | __sb1_writeback_inv_dcache_all(); |
278 | __sb1_flush_icache_all(); | 278 | __sb1_flush_icache_all(); |
diff --git a/arch/mips/mm/init.c b/arch/mips/mm/init.c index 4c80528deadd..b8cb0dde3af0 100644 --- a/arch/mips/mm/init.c +++ b/arch/mips/mm/init.c | |||
@@ -484,7 +484,7 @@ void free_initrd_mem(unsigned long start, unsigned long end) | |||
484 | } | 484 | } |
485 | #endif | 485 | #endif |
486 | 486 | ||
487 | void free_initmem(void) | 487 | void __init_refok free_initmem(void) |
488 | { | 488 | { |
489 | prom_free_prom_memory(); | 489 | prom_free_prom_memory(); |
490 | free_init_pages("unused kernel memory", | 490 | free_init_pages("unused kernel memory", |
diff --git a/arch/mips/sni/sniprom.c b/arch/mips/sni/sniprom.c index 00a03a6e8f58..db544a6e23f3 100644 --- a/arch/mips/sni/sniprom.c +++ b/arch/mips/sni/sniprom.c | |||
@@ -19,6 +19,7 @@ | |||
19 | #include <asm/addrspace.h> | 19 | #include <asm/addrspace.h> |
20 | #include <asm/sni.h> | 20 | #include <asm/sni.h> |
21 | #include <asm/mipsprom.h> | 21 | #include <asm/mipsprom.h> |
22 | #include <asm/mipsregs.h> | ||
22 | #include <asm/bootinfo.h> | 23 | #include <asm/bootinfo.h> |
23 | 24 | ||
24 | /* special SNI prom calls */ | 25 | /* special SNI prom calls */ |
@@ -71,7 +72,7 @@ const char *get_system_type(void) | |||
71 | #define SNI_IDPROM_SIZE 0x1000 | 72 | #define SNI_IDPROM_SIZE 0x1000 |
72 | 73 | ||
73 | #ifdef DEBUG | 74 | #ifdef DEBUG |
74 | static void sni_idprom_dump(void) | 75 | static void __init sni_idprom_dump(void) |
75 | { | 76 | { |
76 | int i; | 77 | int i; |
77 | 78 | ||
@@ -88,7 +89,7 @@ static void sni_idprom_dump(void) | |||
88 | } | 89 | } |
89 | #endif | 90 | #endif |
90 | 91 | ||
91 | static void sni_mem_init(void ) | 92 | static void __init sni_mem_init(void ) |
92 | { | 93 | { |
93 | int i, memsize; | 94 | int i, memsize; |
94 | struct membank { | 95 | struct membank { |
diff --git a/arch/mips/tx4927/toshiba_rbtx4927/toshiba_rbtx4927_setup.c b/arch/mips/tx4927/toshiba_rbtx4927/toshiba_rbtx4927_setup.c index 40c7c3eeafaf..ab72292a172e 100644 --- a/arch/mips/tx4927/toshiba_rbtx4927/toshiba_rbtx4927_setup.c +++ b/arch/mips/tx4927/toshiba_rbtx4927/toshiba_rbtx4927_setup.c | |||
@@ -1020,7 +1020,7 @@ void __init toshiba_rbtx4927_timer_setup(struct irqaction *irq) | |||
1020 | 1020 | ||
1021 | static int __init toshiba_rbtx4927_rtc_init(void) | 1021 | static int __init toshiba_rbtx4927_rtc_init(void) |
1022 | { | 1022 | { |
1023 | struct resource res = { | 1023 | static struct resource __initdata res = { |
1024 | .start = 0x1c010000, | 1024 | .start = 0x1c010000, |
1025 | .end = 0x1c010000 + 0x800 - 1, | 1025 | .end = 0x1c010000 + 0x800 - 1, |
1026 | .flags = IORESOURCE_MEM, | 1026 | .flags = IORESOURCE_MEM, |
diff --git a/arch/powerpc/Kconfig b/arch/powerpc/Kconfig index 853c282da22e..00099efe0e9f 100644 --- a/arch/powerpc/Kconfig +++ b/arch/powerpc/Kconfig | |||
@@ -411,11 +411,6 @@ config PPC_INDIRECT_PCI | |||
411 | default y if 40x || 44x | 411 | default y if 40x || 44x |
412 | default n | 412 | default n |
413 | 413 | ||
414 | config PPC_INDIRECT_PCI_BE | ||
415 | bool | ||
416 | depends PPC_INDIRECT_PCI | ||
417 | default n | ||
418 | |||
419 | config EISA | 414 | config EISA |
420 | bool | 415 | bool |
421 | 416 | ||
@@ -425,6 +420,10 @@ config SBUS | |||
425 | config FSL_SOC | 420 | config FSL_SOC |
426 | bool | 421 | bool |
427 | 422 | ||
423 | config FSL_PCI | ||
424 | bool | ||
425 | select PPC_INDIRECT_PCI | ||
426 | |||
428 | # Yes MCA RS/6000s exist but Linux-PPC does not currently support any | 427 | # Yes MCA RS/6000s exist but Linux-PPC does not currently support any |
429 | config MCA | 428 | config MCA |
430 | bool | 429 | bool |
diff --git a/arch/powerpc/Kconfig.debug b/arch/powerpc/Kconfig.debug index 346cd3befe1e..5c71624ee38d 100644 --- a/arch/powerpc/Kconfig.debug +++ b/arch/powerpc/Kconfig.debug | |||
@@ -134,7 +134,7 @@ config BDI_SWITCH | |||
134 | 134 | ||
135 | config BOOTX_TEXT | 135 | config BOOTX_TEXT |
136 | bool "Support for early boot text console (BootX or OpenFirmware only)" | 136 | bool "Support for early boot text console (BootX or OpenFirmware only)" |
137 | depends PPC_OF | 137 | depends PPC_OF && PPC_MULTIPLATFORM |
138 | help | 138 | help |
139 | Say Y here to see progress messages from the boot firmware in text | 139 | Say Y here to see progress messages from the boot firmware in text |
140 | mode. Requires either BootX or Open Firmware. | 140 | mode. Requires either BootX or Open Firmware. |
diff --git a/arch/powerpc/boot/dts/kuroboxHD.dts b/arch/powerpc/boot/dts/kuroboxHD.dts index a983680c3263..122537419d9f 100644 --- a/arch/powerpc/boot/dts/kuroboxHD.dts +++ b/arch/powerpc/boot/dts/kuroboxHD.dts | |||
@@ -33,12 +33,10 @@ build with: "dtc -f -I dts -O dtb -o kuroboxHD.dtb -V 16 kuroboxHD.dts" | |||
33 | PowerPC,603e { /* Really 8241 */ | 33 | PowerPC,603e { /* Really 8241 */ |
34 | device_type = "cpu"; | 34 | device_type = "cpu"; |
35 | reg = <0>; | 35 | reg = <0>; |
36 | clock-frequency = <bebc200>; /* Fixed by bootwrapper */ | 36 | clock-frequency = <bebc200>; /* Fixed by bootloader */ |
37 | timebase-frequency = <1743000>; /* Fixed by bootwrapper */ | 37 | timebase-frequency = <1743000>; /* Fixed by bootloader */ |
38 | bus-frequency = <0>; /* From bootloader */ | 38 | bus-frequency = <0>; /* Fixed by bootloader */ |
39 | /* Following required by dtc but not used */ | 39 | /* Following required by dtc but not used */ |
40 | i-cache-line-size = <0>; | ||
41 | d-cache-line-size = <0>; | ||
42 | i-cache-size = <4000>; | 40 | i-cache-size = <4000>; |
43 | d-cache-size = <4000>; | 41 | d-cache-size = <4000>; |
44 | }; | 42 | }; |
@@ -64,11 +62,19 @@ build with: "dtc -f -I dts -O dtb -o kuroboxHD.dtb -V 16 kuroboxHD.dts" | |||
64 | fef00000 fef00000 00100000>; /* pci iack */ | 62 | fef00000 fef00000 00100000>; /* pci iack */ |
65 | 63 | ||
66 | i2c@80003000 { | 64 | i2c@80003000 { |
65 | #address-cells = <1>; | ||
66 | #size-cells = <0>; | ||
67 | device_type = "i2c"; | 67 | device_type = "i2c"; |
68 | compatible = "fsl-i2c"; | 68 | compatible = "fsl-i2c"; |
69 | reg = <80003000 1000>; | 69 | reg = <80003000 1000>; |
70 | interrupts = <5 2>; | 70 | interrupts = <5 2>; |
71 | interrupt-parent = <&mpic>; | 71 | interrupt-parent = <&mpic>; |
72 | |||
73 | rtc@32 { | ||
74 | device_type = "rtc"; | ||
75 | compatible = "ricoh,rs5c372b"; | ||
76 | reg = <32>; | ||
77 | }; | ||
72 | }; | 78 | }; |
73 | 79 | ||
74 | serial@80004500 { | 80 | serial@80004500 { |
@@ -91,7 +97,7 @@ build with: "dtc -f -I dts -O dtb -o kuroboxHD.dtb -V 16 kuroboxHD.dts" | |||
91 | interrupt-parent = <&mpic>; | 97 | interrupt-parent = <&mpic>; |
92 | }; | 98 | }; |
93 | 99 | ||
94 | mpic: pic@80040000 { | 100 | mpic: interrupt-controller@80040000 { |
95 | #interrupt-cells = <2>; | 101 | #interrupt-cells = <2>; |
96 | #address-cells = <0>; | 102 | #address-cells = <0>; |
97 | device_type = "open-pic"; | 103 | device_type = "open-pic"; |
diff --git a/arch/powerpc/boot/dts/kuroboxHG.dts b/arch/powerpc/boot/dts/kuroboxHG.dts index 5cf42dc022df..579aa8b967d9 100644 --- a/arch/powerpc/boot/dts/kuroboxHG.dts +++ b/arch/powerpc/boot/dts/kuroboxHG.dts | |||
@@ -33,12 +33,10 @@ build with: "dtc -f -I dts -O dtb -o kuroboxHG.dtb -V 16 kuroboxHG.dts" | |||
33 | PowerPC,603e { /* Really 8241 */ | 33 | PowerPC,603e { /* Really 8241 */ |
34 | device_type = "cpu"; | 34 | device_type = "cpu"; |
35 | reg = <0>; | 35 | reg = <0>; |
36 | clock-frequency = <fdad680>; /* Fixed by bootwrapper */ | 36 | clock-frequency = <fdad680>; /* Fixed by bootloader */ |
37 | timebase-frequency = <1F04000>; /* Fixed by bootwrapper */ | 37 | timebase-frequency = <1F04000>; /* Fixed by bootloader */ |
38 | bus-frequency = <0>; /* From bootloader */ | 38 | bus-frequency = <0>; /* Fixed by bootloader */ |
39 | /* Following required by dtc but not used */ | 39 | /* Following required by dtc but not used */ |
40 | i-cache-line-size = <0>; | ||
41 | d-cache-line-size = <0>; | ||
42 | i-cache-size = <4000>; | 40 | i-cache-size = <4000>; |
43 | d-cache-size = <4000>; | 41 | d-cache-size = <4000>; |
44 | }; | 42 | }; |
@@ -64,11 +62,19 @@ build with: "dtc -f -I dts -O dtb -o kuroboxHG.dtb -V 16 kuroboxHG.dts" | |||
64 | fef00000 fef00000 00100000>; /* pci iack */ | 62 | fef00000 fef00000 00100000>; /* pci iack */ |
65 | 63 | ||
66 | i2c@80003000 { | 64 | i2c@80003000 { |
65 | #address-cells = <1>; | ||
66 | #size-cells = <0>; | ||
67 | device_type = "i2c"; | 67 | device_type = "i2c"; |
68 | compatible = "fsl-i2c"; | 68 | compatible = "fsl-i2c"; |
69 | reg = <80003000 1000>; | 69 | reg = <80003000 1000>; |
70 | interrupts = <5 2>; | 70 | interrupts = <5 2>; |
71 | interrupt-parent = <&mpic>; | 71 | interrupt-parent = <&mpic>; |
72 | |||
73 | rtc@32 { | ||
74 | device_type = "rtc"; | ||
75 | compatible = "ricoh,rs5c372b"; | ||
76 | reg = <32>; | ||
77 | }; | ||
72 | }; | 78 | }; |
73 | 79 | ||
74 | serial@80004500 { | 80 | serial@80004500 { |
@@ -91,8 +97,7 @@ build with: "dtc -f -I dts -O dtb -o kuroboxHG.dtb -V 16 kuroboxHG.dts" | |||
91 | interrupt-parent = <&mpic>; | 97 | interrupt-parent = <&mpic>; |
92 | }; | 98 | }; |
93 | 99 | ||
94 | mpic: pic@80040000 { | 100 | mpic: interrupt-controller@80040000 { |
95 | interrupt-parent = <&mpic>; | ||
96 | #interrupt-cells = <2>; | 101 | #interrupt-cells = <2>; |
97 | #address-cells = <0>; | 102 | #address-cells = <0>; |
98 | device_type = "open-pic"; | 103 | device_type = "open-pic"; |
diff --git a/arch/powerpc/boot/dts/mpc7448hpc2.dts b/arch/powerpc/boot/dts/mpc7448hpc2.dts index 0e3d314a7158..b9158eb2797e 100644 --- a/arch/powerpc/boot/dts/mpc7448hpc2.dts +++ b/arch/powerpc/boot/dts/mpc7448hpc2.dts | |||
@@ -45,7 +45,7 @@ | |||
45 | #address-cells = <1>; | 45 | #address-cells = <1>; |
46 | #size-cells = <1>; | 46 | #size-cells = <1>; |
47 | #interrupt-cells = <2>; | 47 | #interrupt-cells = <2>; |
48 | device_type = "tsi108-bridge"; | 48 | device_type = "tsi-bridge"; |
49 | ranges = <00000000 c0000000 00010000>; | 49 | ranges = <00000000 c0000000 00010000>; |
50 | reg = <c0000000 00010000>; | 50 | reg = <c0000000 00010000>; |
51 | bus-frequency = <0>; | 51 | bus-frequency = <0>; |
diff --git a/arch/powerpc/boot/dts/mpc8313erdb.dts b/arch/powerpc/boot/dts/mpc8313erdb.dts index a1533cc07d09..c5adbe40364e 100644 --- a/arch/powerpc/boot/dts/mpc8313erdb.dts +++ b/arch/powerpc/boot/dts/mpc8313erdb.dts | |||
@@ -178,7 +178,7 @@ | |||
178 | #size-cells = <2>; | 178 | #size-cells = <2>; |
179 | #address-cells = <3>; | 179 | #address-cells = <3>; |
180 | reg = <8500 100>; | 180 | reg = <8500 100>; |
181 | compatible = "83xx"; | 181 | compatible = "fsl,mpc8349-pci"; |
182 | device_type = "pci"; | 182 | device_type = "pci"; |
183 | }; | 183 | }; |
184 | 184 | ||
diff --git a/arch/powerpc/boot/dts/mpc832x_mds.dts b/arch/powerpc/boot/dts/mpc832x_mds.dts index 4fc0c4d34aa8..f158ed781ba8 100644 --- a/arch/powerpc/boot/dts/mpc832x_mds.dts +++ b/arch/powerpc/boot/dts/mpc832x_mds.dts | |||
@@ -154,7 +154,7 @@ | |||
154 | #size-cells = <2>; | 154 | #size-cells = <2>; |
155 | #address-cells = <3>; | 155 | #address-cells = <3>; |
156 | reg = <8500 100>; | 156 | reg = <8500 100>; |
157 | compatible = "83xx"; | 157 | compatible = "fsl,mpc8349-pci"; |
158 | device_type = "pci"; | 158 | device_type = "pci"; |
159 | }; | 159 | }; |
160 | 160 | ||
diff --git a/arch/powerpc/boot/dts/mpc832x_rdb.dts b/arch/powerpc/boot/dts/mpc832x_rdb.dts index 447c03ffabbc..7c4beff3e200 100644 --- a/arch/powerpc/boot/dts/mpc832x_rdb.dts +++ b/arch/powerpc/boot/dts/mpc832x_rdb.dts | |||
@@ -123,7 +123,7 @@ | |||
123 | #size-cells = <2>; | 123 | #size-cells = <2>; |
124 | #address-cells = <3>; | 124 | #address-cells = <3>; |
125 | reg = <8500 100>; | 125 | reg = <8500 100>; |
126 | compatible = "83xx"; | 126 | compatible = "fsl,mpc8349-pci"; |
127 | device_type = "pci"; | 127 | device_type = "pci"; |
128 | }; | 128 | }; |
129 | 129 | ||
diff --git a/arch/powerpc/boot/dts/mpc8349emitx.dts b/arch/powerpc/boot/dts/mpc8349emitx.dts index ae9bca575453..502f47c01797 100644 --- a/arch/powerpc/boot/dts/mpc8349emitx.dts +++ b/arch/powerpc/boot/dts/mpc8349emitx.dts | |||
@@ -197,7 +197,7 @@ | |||
197 | #size-cells = <2>; | 197 | #size-cells = <2>; |
198 | #address-cells = <3>; | 198 | #address-cells = <3>; |
199 | reg = <8500 100>; | 199 | reg = <8500 100>; |
200 | compatible = "83xx"; | 200 | compatible = "fsl,mpc8349-pci"; |
201 | device_type = "pci"; | 201 | device_type = "pci"; |
202 | }; | 202 | }; |
203 | 203 | ||
@@ -222,7 +222,7 @@ | |||
222 | #size-cells = <2>; | 222 | #size-cells = <2>; |
223 | #address-cells = <3>; | 223 | #address-cells = <3>; |
224 | reg = <8600 100>; | 224 | reg = <8600 100>; |
225 | compatible = "83xx"; | 225 | compatible = "fsl,mpc8349-pci"; |
226 | device_type = "pci"; | 226 | device_type = "pci"; |
227 | }; | 227 | }; |
228 | 228 | ||
diff --git a/arch/powerpc/boot/dts/mpc8349emitxgp.dts b/arch/powerpc/boot/dts/mpc8349emitxgp.dts index f636528a3c72..0b8387141d88 100644 --- a/arch/powerpc/boot/dts/mpc8349emitxgp.dts +++ b/arch/powerpc/boot/dts/mpc8349emitxgp.dts | |||
@@ -154,7 +154,7 @@ | |||
154 | #size-cells = <2>; | 154 | #size-cells = <2>; |
155 | #address-cells = <3>; | 155 | #address-cells = <3>; |
156 | reg = <8600 100>; | 156 | reg = <8600 100>; |
157 | compatible = "83xx"; | 157 | compatible = "fsl,mpc8349-pci"; |
158 | device_type = "pci"; | 158 | device_type = "pci"; |
159 | }; | 159 | }; |
160 | 160 | ||
diff --git a/arch/powerpc/boot/dts/mpc834x_mds.dts b/arch/powerpc/boot/dts/mpc834x_mds.dts index 310e877826b4..481099756e44 100644 --- a/arch/powerpc/boot/dts/mpc834x_mds.dts +++ b/arch/powerpc/boot/dts/mpc834x_mds.dts | |||
@@ -241,7 +241,7 @@ | |||
241 | #size-cells = <2>; | 241 | #size-cells = <2>; |
242 | #address-cells = <3>; | 242 | #address-cells = <3>; |
243 | reg = <8500 100>; | 243 | reg = <8500 100>; |
244 | compatible = "83xx"; | 244 | compatible = "fsl,mpc8349-pci"; |
245 | device_type = "pci"; | 245 | device_type = "pci"; |
246 | }; | 246 | }; |
247 | 247 | ||
@@ -301,7 +301,7 @@ | |||
301 | #size-cells = <2>; | 301 | #size-cells = <2>; |
302 | #address-cells = <3>; | 302 | #address-cells = <3>; |
303 | reg = <8600 100>; | 303 | reg = <8600 100>; |
304 | compatible = "83xx"; | 304 | compatible = "fsl,mpc8349-pci"; |
305 | device_type = "pci"; | 305 | device_type = "pci"; |
306 | }; | 306 | }; |
307 | 307 | ||
diff --git a/arch/powerpc/boot/dts/mpc836x_mds.dts b/arch/powerpc/boot/dts/mpc836x_mds.dts index 1e914f31dd92..e3f7c1282068 100644 --- a/arch/powerpc/boot/dts/mpc836x_mds.dts +++ b/arch/powerpc/boot/dts/mpc836x_mds.dts | |||
@@ -169,7 +169,7 @@ | |||
169 | #size-cells = <2>; | 169 | #size-cells = <2>; |
170 | #address-cells = <3>; | 170 | #address-cells = <3>; |
171 | reg = <8500 100>; | 171 | reg = <8500 100>; |
172 | compatible = "83xx"; | 172 | compatible = "fsl,mpc8349-pci"; |
173 | device_type = "pci"; | 173 | device_type = "pci"; |
174 | }; | 174 | }; |
175 | 175 | ||
diff --git a/arch/powerpc/boot/dts/mpc8540ads.dts b/arch/powerpc/boot/dts/mpc8540ads.dts index 364a969f5c2f..fc8dff9f6201 100644 --- a/arch/powerpc/boot/dts/mpc8540ads.dts +++ b/arch/powerpc/boot/dts/mpc8540ads.dts | |||
@@ -258,7 +258,7 @@ | |||
258 | #size-cells = <2>; | 258 | #size-cells = <2>; |
259 | #address-cells = <3>; | 259 | #address-cells = <3>; |
260 | reg = <8000 1000>; | 260 | reg = <8000 1000>; |
261 | compatible = "85xx"; | 261 | compatible = "fsl,mpc8540-pcix", "fsl,mpc8540-pci"; |
262 | device_type = "pci"; | 262 | device_type = "pci"; |
263 | }; | 263 | }; |
264 | 264 | ||
diff --git a/arch/powerpc/boot/dts/mpc8541cds.dts b/arch/powerpc/boot/dts/mpc8541cds.dts index 070206fffe88..fb0b647f8c2a 100644 --- a/arch/powerpc/boot/dts/mpc8541cds.dts +++ b/arch/powerpc/boot/dts/mpc8541cds.dts | |||
@@ -193,7 +193,7 @@ | |||
193 | #size-cells = <2>; | 193 | #size-cells = <2>; |
194 | #address-cells = <3>; | 194 | #address-cells = <3>; |
195 | reg = <8000 1000>; | 195 | reg = <8000 1000>; |
196 | compatible = "85xx"; | 196 | compatible = "fsl,mpc8540-pci"; |
197 | device_type = "pci"; | 197 | device_type = "pci"; |
198 | 198 | ||
199 | i8259@19000 { | 199 | i8259@19000 { |
@@ -230,7 +230,7 @@ | |||
230 | #size-cells = <2>; | 230 | #size-cells = <2>; |
231 | #address-cells = <3>; | 231 | #address-cells = <3>; |
232 | reg = <9000 1000>; | 232 | reg = <9000 1000>; |
233 | compatible = "85xx"; | 233 | compatible = "fsl,mpc8540-pci"; |
234 | device_type = "pci"; | 234 | device_type = "pci"; |
235 | }; | 235 | }; |
236 | 236 | ||
diff --git a/arch/powerpc/boot/dts/mpc8544ds.dts b/arch/powerpc/boot/dts/mpc8544ds.dts index 828592592460..4680e2010887 100644 --- a/arch/powerpc/boot/dts/mpc8544ds.dts +++ b/arch/powerpc/boot/dts/mpc8544ds.dts | |||
@@ -104,6 +104,7 @@ | |||
104 | interrupts = <1d 2 1e 2 22 2>; | 104 | interrupts = <1d 2 1e 2 22 2>; |
105 | interrupt-parent = <&mpic>; | 105 | interrupt-parent = <&mpic>; |
106 | phy-handle = <&phy0>; | 106 | phy-handle = <&phy0>; |
107 | phy-connection-type = "rgmii-id"; | ||
107 | }; | 108 | }; |
108 | 109 | ||
109 | ethernet@26000 { | 110 | ethernet@26000 { |
@@ -117,6 +118,7 @@ | |||
117 | interrupts = <1f 2 20 2 21 2>; | 118 | interrupts = <1f 2 20 2 21 2>; |
118 | interrupt-parent = <&mpic>; | 119 | interrupt-parent = <&mpic>; |
119 | phy-handle = <&phy1>; | 120 | phy-handle = <&phy1>; |
121 | phy-connection-type = "rgmii-id"; | ||
120 | }; | 122 | }; |
121 | 123 | ||
122 | serial@4500 { | 124 | serial@4500 { |
@@ -137,6 +139,223 @@ | |||
137 | interrupt-parent = <&mpic>; | 139 | interrupt-parent = <&mpic>; |
138 | }; | 140 | }; |
139 | 141 | ||
142 | pci@8000 { | ||
143 | compatible = "fsl,mpc8540-pci"; | ||
144 | device_type = "pci"; | ||
145 | interrupt-map-mask = <f800 0 0 7>; | ||
146 | interrupt-map = < | ||
147 | |||
148 | /* IDSEL 0x11 J17 Slot 1 */ | ||
149 | 8800 0 0 1 &mpic 2 1 | ||
150 | 8800 0 0 2 &mpic 3 1 | ||
151 | 8800 0 0 3 &mpic 4 1 | ||
152 | 8800 0 0 4 &mpic 1 1 | ||
153 | |||
154 | /* IDSEL 0x12 J16 Slot 2 */ | ||
155 | |||
156 | 9000 0 0 1 &mpic 3 1 | ||
157 | 9000 0 0 2 &mpic 4 1 | ||
158 | 9000 0 0 3 &mpic 2 1 | ||
159 | 9000 0 0 4 &mpic 1 1>; | ||
160 | |||
161 | interrupt-parent = <&mpic>; | ||
162 | interrupts = <18 2>; | ||
163 | bus-range = <0 ff>; | ||
164 | ranges = <02000000 0 80000000 80000000 0 10000000 | ||
165 | 01000000 0 00000000 e2000000 0 00800000>; | ||
166 | clock-frequency = <3f940aa>; | ||
167 | #interrupt-cells = <1>; | ||
168 | #size-cells = <2>; | ||
169 | #address-cells = <3>; | ||
170 | reg = <8000 1000>; | ||
171 | }; | ||
172 | |||
173 | pcie@9000 { | ||
174 | compatible = "fsl,mpc8548-pcie"; | ||
175 | device_type = "pci"; | ||
176 | #interrupt-cells = <1>; | ||
177 | #size-cells = <2>; | ||
178 | #address-cells = <3>; | ||
179 | reg = <9000 1000>; | ||
180 | bus-range = <0 ff>; | ||
181 | ranges = <02000000 0 90000000 90000000 0 10000000 | ||
182 | 01000000 0 00000000 e3000000 0 00800000>; | ||
183 | clock-frequency = <1fca055>; | ||
184 | interrupt-parent = <&mpic>; | ||
185 | interrupts = <1a 2>; | ||
186 | interrupt-map-mask = <f800 0 0 7>; | ||
187 | interrupt-map = < | ||
188 | /* IDSEL 0x0 */ | ||
189 | 0000 0 0 1 &mpic 4 1 | ||
190 | 0000 0 0 2 &mpic 5 1 | ||
191 | 0000 0 0 3 &mpic 6 1 | ||
192 | 0000 0 0 4 &mpic 7 1 | ||
193 | >; | ||
194 | }; | ||
195 | |||
196 | pcie@a000 { | ||
197 | compatible = "fsl,mpc8548-pcie"; | ||
198 | device_type = "pci"; | ||
199 | #interrupt-cells = <1>; | ||
200 | #size-cells = <2>; | ||
201 | #address-cells = <3>; | ||
202 | reg = <a000 1000>; | ||
203 | bus-range = <0 ff>; | ||
204 | ranges = <02000000 0 a0000000 a0000000 0 10000000 | ||
205 | 01000000 0 00000000 e2800000 0 00800000>; | ||
206 | clock-frequency = <1fca055>; | ||
207 | interrupt-parent = <&mpic>; | ||
208 | interrupts = <19 2>; | ||
209 | interrupt-map-mask = <f800 0 0 7>; | ||
210 | interrupt-map = < | ||
211 | /* IDSEL 0x0 */ | ||
212 | 0000 0 0 1 &mpic 0 1 | ||
213 | 0000 0 0 2 &mpic 1 1 | ||
214 | 0000 0 0 3 &mpic 2 1 | ||
215 | 0000 0 0 4 &mpic 3 1 | ||
216 | >; | ||
217 | }; | ||
218 | |||
219 | pcie@b000 { | ||
220 | compatible = "fsl,mpc8548-pcie"; | ||
221 | device_type = "pci"; | ||
222 | #interrupt-cells = <1>; | ||
223 | #size-cells = <2>; | ||
224 | #address-cells = <3>; | ||
225 | reg = <b000 1000>; | ||
226 | bus-range = <0 ff>; | ||
227 | ranges = <02000000 0 b0000000 b0000000 0 10000000 | ||
228 | 01000000 0 00000000 e3800000 0 00800000>; | ||
229 | clock-frequency = <1fca055>; | ||
230 | interrupt-parent = <&mpic>; | ||
231 | interrupts = <1b 2>; | ||
232 | interrupt-map-mask = <f800 0 0 7>; | ||
233 | interrupt-map = < | ||
234 | |||
235 | // IDSEL 0x1a | ||
236 | d000 0 0 1 &i8259 6 2 | ||
237 | d000 0 0 2 &i8259 3 2 | ||
238 | d000 0 0 3 &i8259 4 2 | ||
239 | d000 0 0 4 &i8259 5 2 | ||
240 | |||
241 | // IDSEL 0x1b | ||
242 | d800 0 0 1 &i8259 5 2 | ||
243 | d800 0 0 2 &i8259 0 0 | ||
244 | d800 0 0 3 &i8259 0 0 | ||
245 | d800 0 0 4 &i8259 0 0 | ||
246 | |||
247 | // IDSEL 0x1c USB | ||
248 | e000 0 0 1 &i8259 9 2 | ||
249 | e000 0 0 2 &i8259 a 2 | ||
250 | e000 0 0 3 &i8259 c 2 | ||
251 | e000 0 0 4 &i8259 7 2 | ||
252 | |||
253 | // IDSEL 0x1d Audio | ||
254 | e800 0 0 1 &i8259 9 2 | ||
255 | e800 0 0 2 &i8259 a 2 | ||
256 | e800 0 0 3 &i8259 b 2 | ||
257 | e800 0 0 4 &i8259 0 0 | ||
258 | |||
259 | // IDSEL 0x1e Legacy | ||
260 | f000 0 0 1 &i8259 c 2 | ||
261 | f000 0 0 2 &i8259 0 0 | ||
262 | f000 0 0 3 &i8259 0 0 | ||
263 | f000 0 0 4 &i8259 0 0 | ||
264 | |||
265 | // IDSEL 0x1f IDE/SATA | ||
266 | f800 0 0 1 &i8259 6 2 | ||
267 | f800 0 0 2 &i8259 0 0 | ||
268 | f800 0 0 3 &i8259 0 0 | ||
269 | f800 0 0 4 &i8259 0 0 | ||
270 | >; | ||
271 | uli1575@0 { | ||
272 | reg = <0 0 0 0 0>; | ||
273 | #size-cells = <2>; | ||
274 | #address-cells = <3>; | ||
275 | ranges = <02000000 0 b0000000 | ||
276 | 02000000 0 b0000000 | ||
277 | 0 10000000 | ||
278 | 01000000 0 00000000 | ||
279 | 01000000 0 00000000 | ||
280 | 0 00080000>; | ||
281 | |||
282 | pci_bridge@0 { | ||
283 | reg = <0 0 0 0 0>; | ||
284 | #size-cells = <2>; | ||
285 | #address-cells = <3>; | ||
286 | ranges = <02000000 0 b0000000 | ||
287 | 02000000 0 b0000000 | ||
288 | 0 20000000 | ||
289 | 01000000 0 00000000 | ||
290 | 01000000 0 00000000 | ||
291 | 0 00100000>; | ||
292 | |||
293 | isa@1e { | ||
294 | device_type = "isa"; | ||
295 | #interrupt-cells = <2>; | ||
296 | #size-cells = <1>; | ||
297 | #address-cells = <2>; | ||
298 | reg = <f000 0 0 0 0>; | ||
299 | ranges = <1 0 01000000 0 0 | ||
300 | 00001000>; | ||
301 | interrupt-parent = <&i8259>; | ||
302 | |||
303 | i8259: interrupt-controller@20 { | ||
304 | reg = <1 20 2 | ||
305 | 1 a0 2 | ||
306 | 1 4d0 2>; | ||
307 | clock-frequency = <0>; | ||
308 | interrupt-controller; | ||
309 | device_type = "interrupt-controller"; | ||
310 | #address-cells = <0>; | ||
311 | #interrupt-cells = <2>; | ||
312 | built-in; | ||
313 | compatible = "chrp,iic"; | ||
314 | interrupts = <9 2>; | ||
315 | interrupt-parent = | ||
316 | <&mpic>; | ||
317 | }; | ||
318 | |||
319 | i8042@60 { | ||
320 | #size-cells = <0>; | ||
321 | #address-cells = <1>; | ||
322 | reg = <1 60 1 1 64 1>; | ||
323 | interrupts = <1 3 c 3>; | ||
324 | interrupt-parent = | ||
325 | <&i8259>; | ||
326 | |||
327 | keyboard@0 { | ||
328 | reg = <0>; | ||
329 | compatible = "pnpPNP,303"; | ||
330 | }; | ||
331 | |||
332 | mouse@1 { | ||
333 | reg = <1>; | ||
334 | compatible = "pnpPNP,f03"; | ||
335 | }; | ||
336 | }; | ||
337 | |||
338 | rtc@70 { | ||
339 | compatible = | ||
340 | "pnpPNP,b00"; | ||
341 | reg = <1 70 2>; | ||
342 | }; | ||
343 | |||
344 | gpio@400 { | ||
345 | reg = <1 400 80>; | ||
346 | }; | ||
347 | }; | ||
348 | }; | ||
349 | }; | ||
350 | |||
351 | }; | ||
352 | |||
353 | global-utilities@e0000 { //global utilities block | ||
354 | compatible = "fsl,mpc8548-guts"; | ||
355 | reg = <e0000 1000>; | ||
356 | fsl,has-rstcr; | ||
357 | }; | ||
358 | |||
140 | mpic: pic@40000 { | 359 | mpic: pic@40000 { |
141 | clock-frequency = <0>; | 360 | clock-frequency = <0>; |
142 | interrupt-controller; | 361 | interrupt-controller; |
diff --git a/arch/powerpc/boot/dts/mpc8548cds.dts b/arch/powerpc/boot/dts/mpc8548cds.dts index 9d0b84b66cd4..d215d21fff42 100644 --- a/arch/powerpc/boot/dts/mpc8548cds.dts +++ b/arch/powerpc/boot/dts/mpc8548cds.dts | |||
@@ -1,5 +1,5 @@ | |||
1 | /* | 1 | /* |
2 | * MPC8555 CDS Device Tree Source | 2 | * MPC8548 CDS Device Tree Source |
3 | * | 3 | * |
4 | * Copyright 2006 Freescale Semiconductor Inc. | 4 | * Copyright 2006 Freescale Semiconductor Inc. |
5 | * | 5 | * |
@@ -44,8 +44,14 @@ | |||
44 | #size-cells = <1>; | 44 | #size-cells = <1>; |
45 | #interrupt-cells = <2>; | 45 | #interrupt-cells = <2>; |
46 | device_type = "soc"; | 46 | device_type = "soc"; |
47 | ranges = <0 e0000000 00100000>; | 47 | ranges = <00001000 e0001000 000ff000 |
48 | reg = <e0000000 00100000>; // CCSRBAR 1M | 48 | 80000000 80000000 10000000 |
49 | e2000000 e2000000 00800000 | ||
50 | 90000000 90000000 10000000 | ||
51 | e2800000 e2800000 00800000 | ||
52 | a0000000 a0000000 20000000 | ||
53 | e3000000 e3000000 01000000>; | ||
54 | reg = <e0000000 00001000>; // CCSRBAR | ||
49 | bus-frequency = <0>; | 55 | bus-frequency = <0>; |
50 | 56 | ||
51 | memory-controller@2000 { | 57 | memory-controller@2000 { |
@@ -162,8 +168,8 @@ | |||
162 | serial@4500 { | 168 | serial@4500 { |
163 | device_type = "serial"; | 169 | device_type = "serial"; |
164 | compatible = "ns16550"; | 170 | compatible = "ns16550"; |
165 | reg = <4500 100>; // reg base, size | 171 | reg = <4500 100>; // reg base, size |
166 | clock-frequency = <0>; // should we fill in in uboot? | 172 | clock-frequency = <0>; // should we fill in in uboot? |
167 | interrupts = <2a 2>; | 173 | interrupts = <2a 2>; |
168 | interrupt-parent = <&mpic>; | 174 | interrupt-parent = <&mpic>; |
169 | }; | 175 | }; |
@@ -172,7 +178,7 @@ | |||
172 | device_type = "serial"; | 178 | device_type = "serial"; |
173 | compatible = "ns16550"; | 179 | compatible = "ns16550"; |
174 | reg = <4600 100>; // reg base, size | 180 | reg = <4600 100>; // reg base, size |
175 | clock-frequency = <0>; // should we fill in in uboot? | 181 | clock-frequency = <0>; // should we fill in in uboot? |
176 | interrupts = <2a 2>; | 182 | interrupts = <2a 2>; |
177 | interrupt-parent = <&mpic>; | 183 | interrupt-parent = <&mpic>; |
178 | }; | 184 | }; |
@@ -183,77 +189,154 @@ | |||
183 | fsl,has-rstcr; | 189 | fsl,has-rstcr; |
184 | }; | 190 | }; |
185 | 191 | ||
186 | pci1: pci@8000 { | 192 | pci@8000 { |
187 | interrupt-map-mask = <1f800 0 0 7>; | 193 | interrupt-map-mask = <f800 0 0 7>; |
188 | interrupt-map = < | 194 | interrupt-map = < |
195 | /* IDSEL 0x4 (PCIX Slot 2) */ | ||
196 | 02000 0 0 1 &mpic 0 1 | ||
197 | 02000 0 0 2 &mpic 1 1 | ||
198 | 02000 0 0 3 &mpic 2 1 | ||
199 | 02000 0 0 4 &mpic 3 1 | ||
200 | |||
201 | /* IDSEL 0x5 (PCIX Slot 3) */ | ||
202 | 02800 0 0 1 &mpic 1 1 | ||
203 | 02800 0 0 2 &mpic 2 1 | ||
204 | 02800 0 0 3 &mpic 3 1 | ||
205 | 02800 0 0 4 &mpic 0 1 | ||
206 | |||
207 | /* IDSEL 0x6 (PCIX Slot 4) */ | ||
208 | 03000 0 0 1 &mpic 2 1 | ||
209 | 03000 0 0 2 &mpic 3 1 | ||
210 | 03000 0 0 3 &mpic 0 1 | ||
211 | 03000 0 0 4 &mpic 1 1 | ||
212 | |||
213 | /* IDSEL 0x8 (PCIX Slot 5) */ | ||
214 | 04000 0 0 1 &mpic 0 1 | ||
215 | 04000 0 0 2 &mpic 1 1 | ||
216 | 04000 0 0 3 &mpic 2 1 | ||
217 | 04000 0 0 4 &mpic 3 1 | ||
218 | |||
219 | /* IDSEL 0xC (Tsi310 bridge) */ | ||
220 | 06000 0 0 1 &mpic 0 1 | ||
221 | 06000 0 0 2 &mpic 1 1 | ||
222 | 06000 0 0 3 &mpic 2 1 | ||
223 | 06000 0 0 4 &mpic 3 1 | ||
224 | |||
225 | /* IDSEL 0x14 (Slot 2) */ | ||
226 | 0a000 0 0 1 &mpic 0 1 | ||
227 | 0a000 0 0 2 &mpic 1 1 | ||
228 | 0a000 0 0 3 &mpic 2 1 | ||
229 | 0a000 0 0 4 &mpic 3 1 | ||
230 | |||
231 | /* IDSEL 0x15 (Slot 3) */ | ||
232 | 0a800 0 0 1 &mpic 1 1 | ||
233 | 0a800 0 0 2 &mpic 2 1 | ||
234 | 0a800 0 0 3 &mpic 3 1 | ||
235 | 0a800 0 0 4 &mpic 0 1 | ||
236 | |||
237 | /* IDSEL 0x16 (Slot 4) */ | ||
238 | 0b000 0 0 1 &mpic 2 1 | ||
239 | 0b000 0 0 2 &mpic 3 1 | ||
240 | 0b000 0 0 3 &mpic 0 1 | ||
241 | 0b000 0 0 4 &mpic 1 1 | ||
242 | |||
243 | /* IDSEL 0x18 (Slot 5) */ | ||
244 | 0c000 0 0 1 &mpic 0 1 | ||
245 | 0c000 0 0 2 &mpic 1 1 | ||
246 | 0c000 0 0 3 &mpic 2 1 | ||
247 | 0c000 0 0 4 &mpic 3 1 | ||
248 | |||
249 | /* IDSEL 0x1C (Tsi310 bridge PCI primary) */ | ||
250 | 0E000 0 0 1 &mpic 0 1 | ||
251 | 0E000 0 0 2 &mpic 1 1 | ||
252 | 0E000 0 0 3 &mpic 2 1 | ||
253 | 0E000 0 0 4 &mpic 3 1>; | ||
189 | 254 | ||
190 | /* IDSEL 0x10 */ | ||
191 | 08000 0 0 1 &mpic 0 1 | ||
192 | 08000 0 0 2 &mpic 1 1 | ||
193 | 08000 0 0 3 &mpic 2 1 | ||
194 | 08000 0 0 4 &mpic 3 1 | ||
195 | |||
196 | /* IDSEL 0x11 */ | ||
197 | 08800 0 0 1 &mpic 0 1 | ||
198 | 08800 0 0 2 &mpic 1 1 | ||
199 | 08800 0 0 3 &mpic 2 1 | ||
200 | 08800 0 0 4 &mpic 3 1 | ||
201 | |||
202 | /* IDSEL 0x12 (Slot 1) */ | ||
203 | 09000 0 0 1 &mpic 0 1 | ||
204 | 09000 0 0 2 &mpic 1 1 | ||
205 | 09000 0 0 3 &mpic 2 1 | ||
206 | 09000 0 0 4 &mpic 3 1 | ||
207 | |||
208 | /* IDSEL 0x13 (Slot 2) */ | ||
209 | 09800 0 0 1 &mpic 1 1 | ||
210 | 09800 0 0 2 &mpic 2 1 | ||
211 | 09800 0 0 3 &mpic 3 1 | ||
212 | 09800 0 0 4 &mpic 0 1 | ||
213 | |||
214 | /* IDSEL 0x14 (Slot 3) */ | ||
215 | 0a000 0 0 1 &mpic 2 1 | ||
216 | 0a000 0 0 2 &mpic 3 1 | ||
217 | 0a000 0 0 3 &mpic 0 1 | ||
218 | 0a000 0 0 4 &mpic 1 1 | ||
219 | |||
220 | /* IDSEL 0x15 (Slot 4) */ | ||
221 | 0a800 0 0 1 &mpic 3 1 | ||
222 | 0a800 0 0 2 &mpic 0 1 | ||
223 | 0a800 0 0 3 &mpic 1 1 | ||
224 | 0a800 0 0 4 &mpic 2 1 | ||
225 | |||
226 | /* Bus 1 (Tundra Bridge) */ | ||
227 | /* IDSEL 0x12 (ISA bridge) */ | ||
228 | 19000 0 0 1 &mpic 0 1 | ||
229 | 19000 0 0 2 &mpic 1 1 | ||
230 | 19000 0 0 3 &mpic 2 1 | ||
231 | 19000 0 0 4 &mpic 3 1>; | ||
232 | interrupt-parent = <&mpic>; | 255 | interrupt-parent = <&mpic>; |
233 | interrupts = <18 2>; | 256 | interrupts = <18 2>; |
234 | bus-range = <0 0>; | 257 | bus-range = <0 0>; |
235 | ranges = <02000000 0 80000000 80000000 0 20000000 | 258 | ranges = <02000000 0 80000000 80000000 0 10000000 |
236 | 01000000 0 00000000 e2000000 0 00100000>; | 259 | 01000000 0 00000000 e2000000 0 00800000>; |
237 | clock-frequency = <3f940aa>; | 260 | clock-frequency = <3f940aa>; |
238 | #interrupt-cells = <1>; | 261 | #interrupt-cells = <1>; |
239 | #size-cells = <2>; | 262 | #size-cells = <2>; |
240 | #address-cells = <3>; | 263 | #address-cells = <3>; |
241 | reg = <8000 1000>; | 264 | reg = <8000 1000>; |
242 | compatible = "85xx"; | 265 | compatible = "fsl,mpc8540-pcix", "fsl,mpc8540-pci"; |
243 | device_type = "pci"; | 266 | device_type = "pci"; |
244 | 267 | ||
245 | i8259@19000 { | 268 | pci_bridge@1c { |
246 | clock-frequency = <0>; | 269 | interrupt-map-mask = <f800 0 0 7>; |
247 | interrupt-controller; | 270 | interrupt-map = < |
248 | device_type = "interrupt-controller"; | 271 | |
249 | reg = <19000 0 0 0 1>; | 272 | /* IDSEL 0x00 (PrPMC Site) */ |
250 | #address-cells = <0>; | 273 | 0000 0 0 1 &mpic 0 1 |
251 | #interrupt-cells = <2>; | 274 | 0000 0 0 2 &mpic 1 1 |
252 | built-in; | 275 | 0000 0 0 3 &mpic 2 1 |
253 | compatible = "chrp,iic"; | 276 | 0000 0 0 4 &mpic 3 1 |
254 | big-endian; | 277 | |
255 | interrupts = <1>; | 278 | /* IDSEL 0x04 (VIA chip) */ |
256 | interrupt-parent = <&pci1>; | 279 | 2000 0 0 1 &mpic 0 1 |
280 | 2000 0 0 2 &mpic 1 1 | ||
281 | 2000 0 0 3 &mpic 2 1 | ||
282 | 2000 0 0 4 &mpic 3 1 | ||
283 | |||
284 | /* IDSEL 0x05 (8139) */ | ||
285 | 2800 0 0 1 &mpic 1 1 | ||
286 | |||
287 | /* IDSEL 0x06 (Slot 6) */ | ||
288 | 3000 0 0 1 &mpic 2 1 | ||
289 | 3000 0 0 2 &mpic 3 1 | ||
290 | 3000 0 0 3 &mpic 0 1 | ||
291 | 3000 0 0 4 &mpic 1 1 | ||
292 | |||
293 | /* IDESL 0x07 (Slot 7) */ | ||
294 | 3800 0 0 1 &mpic 3 1 | ||
295 | 3800 0 0 2 &mpic 0 1 | ||
296 | 3800 0 0 3 &mpic 1 1 | ||
297 | 3800 0 0 4 &mpic 2 1>; | ||
298 | |||
299 | reg = <e000 0 0 0 0>; | ||
300 | #interrupt-cells = <1>; | ||
301 | #size-cells = <2>; | ||
302 | #address-cells = <3>; | ||
303 | ranges = <02000000 0 80000000 | ||
304 | 02000000 0 80000000 | ||
305 | 0 20000000 | ||
306 | 01000000 0 00000000 | ||
307 | 01000000 0 00000000 | ||
308 | 0 00080000>; | ||
309 | clock-frequency = <1fca055>; | ||
310 | |||
311 | isa@4 { | ||
312 | device_type = "isa"; | ||
313 | #interrupt-cells = <2>; | ||
314 | #size-cells = <1>; | ||
315 | #address-cells = <2>; | ||
316 | reg = <2000 0 0 0 0>; | ||
317 | ranges = <1 0 01000000 0 0 00001000>; | ||
318 | interrupt-parent = <&i8259>; | ||
319 | |||
320 | i8259: interrupt-controller@20 { | ||
321 | clock-frequency = <0>; | ||
322 | interrupt-controller; | ||
323 | device_type = "interrupt-controller"; | ||
324 | reg = <1 20 2 | ||
325 | 1 a0 2 | ||
326 | 1 4d0 2>; | ||
327 | #address-cells = <0>; | ||
328 | #interrupt-cells = <2>; | ||
329 | built-in; | ||
330 | compatible = "chrp,iic"; | ||
331 | interrupts = <0 1>; | ||
332 | interrupt-parent = <&mpic>; | ||
333 | }; | ||
334 | |||
335 | rtc@70 { | ||
336 | compatible = "pnpPNP,b00"; | ||
337 | reg = <1 70 2>; | ||
338 | }; | ||
339 | }; | ||
257 | }; | 340 | }; |
258 | }; | 341 | }; |
259 | 342 | ||
@@ -263,20 +346,45 @@ | |||
263 | 346 | ||
264 | /* IDSEL 0x15 */ | 347 | /* IDSEL 0x15 */ |
265 | a800 0 0 1 &mpic b 1 | 348 | a800 0 0 1 &mpic b 1 |
266 | a800 0 0 2 &mpic b 1 | 349 | a800 0 0 2 &mpic 1 1 |
267 | a800 0 0 3 &mpic b 1 | 350 | a800 0 0 3 &mpic 2 1 |
268 | a800 0 0 4 &mpic b 1>; | 351 | a800 0 0 4 &mpic 3 1>; |
352 | |||
269 | interrupt-parent = <&mpic>; | 353 | interrupt-parent = <&mpic>; |
270 | interrupts = <19 2>; | 354 | interrupts = <19 2>; |
271 | bus-range = <0 0>; | 355 | bus-range = <0 0>; |
272 | ranges = <02000000 0 a0000000 a0000000 0 20000000 | 356 | ranges = <02000000 0 90000000 90000000 0 10000000 |
273 | 01000000 0 00000000 e3000000 0 00100000>; | 357 | 01000000 0 00000000 e2800000 0 00800000>; |
274 | clock-frequency = <3f940aa>; | 358 | clock-frequency = <3f940aa>; |
275 | #interrupt-cells = <1>; | 359 | #interrupt-cells = <1>; |
276 | #size-cells = <2>; | 360 | #size-cells = <2>; |
277 | #address-cells = <3>; | 361 | #address-cells = <3>; |
278 | reg = <9000 1000>; | 362 | reg = <9000 1000>; |
279 | compatible = "85xx"; | 363 | compatible = "fsl,mpc8540-pci"; |
364 | device_type = "pci"; | ||
365 | }; | ||
366 | /* PCI Express */ | ||
367 | pcie@a000 { | ||
368 | interrupt-map-mask = <f800 0 0 7>; | ||
369 | interrupt-map = < | ||
370 | |||
371 | /* IDSEL 0x0 (PEX) */ | ||
372 | 00000 0 0 1 &mpic 0 1 | ||
373 | 00000 0 0 2 &mpic 1 1 | ||
374 | 00000 0 0 3 &mpic 2 1 | ||
375 | 00000 0 0 4 &mpic 3 1>; | ||
376 | |||
377 | interrupt-parent = <&mpic>; | ||
378 | interrupts = <1a 2>; | ||
379 | bus-range = <0 ff>; | ||
380 | ranges = <02000000 0 a0000000 a0000000 0 20000000 | ||
381 | 01000000 0 00000000 e3000000 0 08000000>; | ||
382 | clock-frequency = <1fca055>; | ||
383 | #interrupt-cells = <1>; | ||
384 | #size-cells = <2>; | ||
385 | #address-cells = <3>; | ||
386 | reg = <a000 1000>; | ||
387 | compatible = "fsl,mpc8548-pcie"; | ||
280 | device_type = "pci"; | 388 | device_type = "pci"; |
281 | }; | 389 | }; |
282 | 390 | ||
diff --git a/arch/powerpc/boot/dts/mpc8555cds.dts b/arch/powerpc/boot/dts/mpc8555cds.dts index 17e45d9a382a..c3c888252121 100644 --- a/arch/powerpc/boot/dts/mpc8555cds.dts +++ b/arch/powerpc/boot/dts/mpc8555cds.dts | |||
@@ -193,7 +193,7 @@ | |||
193 | #size-cells = <2>; | 193 | #size-cells = <2>; |
194 | #address-cells = <3>; | 194 | #address-cells = <3>; |
195 | reg = <8000 1000>; | 195 | reg = <8000 1000>; |
196 | compatible = "85xx"; | 196 | compatible = "fsl,mpc8540-pci"; |
197 | device_type = "pci"; | 197 | device_type = "pci"; |
198 | 198 | ||
199 | i8259@19000 { | 199 | i8259@19000 { |
@@ -230,7 +230,7 @@ | |||
230 | #size-cells = <2>; | 230 | #size-cells = <2>; |
231 | #address-cells = <3>; | 231 | #address-cells = <3>; |
232 | reg = <9000 1000>; | 232 | reg = <9000 1000>; |
233 | compatible = "85xx"; | 233 | compatible = "fsl,mpc8540-pci"; |
234 | device_type = "pci"; | 234 | device_type = "pci"; |
235 | }; | 235 | }; |
236 | 236 | ||
diff --git a/arch/powerpc/boot/dts/mpc8560ads.dts b/arch/powerpc/boot/dts/mpc8560ads.dts index 21ccaaa27993..16dbe848cecf 100644 --- a/arch/powerpc/boot/dts/mpc8560ads.dts +++ b/arch/powerpc/boot/dts/mpc8560ads.dts | |||
@@ -136,7 +136,7 @@ | |||
136 | #interrupt-cells = <1>; | 136 | #interrupt-cells = <1>; |
137 | #size-cells = <2>; | 137 | #size-cells = <2>; |
138 | #address-cells = <3>; | 138 | #address-cells = <3>; |
139 | compatible = "85xx"; | 139 | compatible = "fsl,mpc8540-pcix", "fsl,mpc8540-pci"; |
140 | device_type = "pci"; | 140 | device_type = "pci"; |
141 | reg = <8000 1000>; | 141 | reg = <8000 1000>; |
142 | clock-frequency = <3f940aa>; | 142 | clock-frequency = <3f940aa>; |
diff --git a/arch/powerpc/boot/dts/mpc8568mds.dts b/arch/powerpc/boot/dts/mpc8568mds.dts index 6bb18f2807a8..b1dcfbe8c1f8 100644 --- a/arch/powerpc/boot/dts/mpc8568mds.dts +++ b/arch/powerpc/boot/dts/mpc8568mds.dts | |||
@@ -170,6 +170,66 @@ | |||
170 | interrupt-parent = <&mpic>; | 170 | interrupt-parent = <&mpic>; |
171 | }; | 171 | }; |
172 | 172 | ||
173 | global-utilities@e0000 { //global utilities block | ||
174 | compatible = "fsl,mpc8548-guts"; | ||
175 | reg = <e0000 1000>; | ||
176 | fsl,has-rstcr; | ||
177 | }; | ||
178 | |||
179 | pci@8000 { | ||
180 | interrupt-map-mask = <f800 0 0 7>; | ||
181 | interrupt-map = < | ||
182 | /* IDSEL 0x12 AD18 */ | ||
183 | 9000 0 0 1 &mpic 5 1 | ||
184 | 9000 0 0 2 &mpic 6 1 | ||
185 | 9000 0 0 3 &mpic 7 1 | ||
186 | 9000 0 0 4 &mpic 4 1 | ||
187 | |||
188 | /* IDSEL 0x13 AD19 */ | ||
189 | 9800 0 0 1 &mpic 6 1 | ||
190 | 9800 0 0 2 &mpic 7 1 | ||
191 | 9800 0 0 3 &mpic 4 1 | ||
192 | 9800 0 0 4 &mpic 5 1>; | ||
193 | |||
194 | interrupt-parent = <&mpic>; | ||
195 | interrupts = <18 2>; | ||
196 | bus-range = <0 ff>; | ||
197 | ranges = <02000000 0 80000000 80000000 0 20000000 | ||
198 | 01000000 0 00000000 e2000000 0 00800000>; | ||
199 | clock-frequency = <3f940aa>; | ||
200 | #interrupt-cells = <1>; | ||
201 | #size-cells = <2>; | ||
202 | #address-cells = <3>; | ||
203 | reg = <8000 1000>; | ||
204 | compatible = "fsl,mpc8540-pci"; | ||
205 | device_type = "pci"; | ||
206 | }; | ||
207 | |||
208 | /* PCI Express */ | ||
209 | pcie@a000 { | ||
210 | interrupt-map-mask = <f800 0 0 7>; | ||
211 | interrupt-map = < | ||
212 | |||
213 | /* IDSEL 0x0 (PEX) */ | ||
214 | 00000 0 0 1 &mpic 0 1 | ||
215 | 00000 0 0 2 &mpic 1 1 | ||
216 | 00000 0 0 3 &mpic 2 1 | ||
217 | 00000 0 0 4 &mpic 3 1>; | ||
218 | |||
219 | interrupt-parent = <&mpic>; | ||
220 | interrupts = <1a 2>; | ||
221 | bus-range = <0 ff>; | ||
222 | ranges = <02000000 0 a0000000 a0000000 0 20000000 | ||
223 | 01000000 0 00000000 e3000000 0 08000000>; | ||
224 | clock-frequency = <1fca055>; | ||
225 | #interrupt-cells = <1>; | ||
226 | #size-cells = <2>; | ||
227 | #address-cells = <3>; | ||
228 | reg = <a000 1000>; | ||
229 | compatible = "fsl,mpc8548-pcie"; | ||
230 | device_type = "pci"; | ||
231 | }; | ||
232 | |||
173 | serial@4600 { | 233 | serial@4600 { |
174 | device_type = "serial"; | 234 | device_type = "serial"; |
175 | compatible = "ns16550"; | 235 | compatible = "ns16550"; |
diff --git a/arch/powerpc/boot/dts/mpc8641_hpcn.dts b/arch/powerpc/boot/dts/mpc8641_hpcn.dts index 6a78a2b37c08..5d82709cfcbb 100644 --- a/arch/powerpc/boot/dts/mpc8641_hpcn.dts +++ b/arch/powerpc/boot/dts/mpc8641_hpcn.dts | |||
@@ -211,8 +211,8 @@ | |||
211 | interrupt-parent = <&mpic>; | 211 | interrupt-parent = <&mpic>; |
212 | }; | 212 | }; |
213 | 213 | ||
214 | pci@8000 { | 214 | pcie@8000 { |
215 | compatible = "86xx"; | 215 | compatible = "fsl,mpc8641-pcie"; |
216 | device_type = "pci"; | 216 | device_type = "pci"; |
217 | #interrupt-cells = <1>; | 217 | #interrupt-cells = <1>; |
218 | #size-cells = <2>; | 218 | #size-cells = <2>; |
@@ -399,8 +399,8 @@ | |||
399 | 399 | ||
400 | }; | 400 | }; |
401 | 401 | ||
402 | pci@9000 { | 402 | pcie@9000 { |
403 | compatible = "86xx"; | 403 | compatible = "fsl,mpc8641-pcie"; |
404 | device_type = "pci"; | 404 | device_type = "pci"; |
405 | #interrupt-cells = <1>; | 405 | #interrupt-cells = <1>; |
406 | #size-cells = <2>; | 406 | #size-cells = <2>; |
diff --git a/arch/powerpc/configs/mpc8544_ds_defconfig b/arch/powerpc/configs/mpc8544_ds_defconfig index c40a25a79cbb..7995231def26 100644 --- a/arch/powerpc/configs/mpc8544_ds_defconfig +++ b/arch/powerpc/configs/mpc8544_ds_defconfig | |||
@@ -1,9 +1,26 @@ | |||
1 | # | 1 | # |
2 | # Automatically generated make config: don't edit | 2 | # Automatically generated make config: don't edit |
3 | # Linux kernel version: 2.6.22-rc7 | 3 | # Linux kernel version: 2.6.22 |
4 | # Sun Jul 1 23:56:58 2007 | 4 | # Fri Jul 20 14:09:13 2007 |
5 | # | 5 | # |
6 | # CONFIG_PPC64 is not set | 6 | # CONFIG_PPC64 is not set |
7 | |||
8 | # | ||
9 | # Processor support | ||
10 | # | ||
11 | # CONFIG_6xx is not set | ||
12 | CONFIG_PPC_85xx=y | ||
13 | # CONFIG_PPC_8xx is not set | ||
14 | # CONFIG_40x is not set | ||
15 | # CONFIG_44x is not set | ||
16 | # CONFIG_E200 is not set | ||
17 | CONFIG_85xx=y | ||
18 | CONFIG_E500=y | ||
19 | CONFIG_BOOKE=y | ||
20 | CONFIG_FSL_BOOKE=y | ||
21 | # CONFIG_PHYS_64BIT is not set | ||
22 | # CONFIG_SPE is not set | ||
23 | # CONFIG_PPC_MM_SLICES is not set | ||
7 | CONFIG_PPC32=y | 24 | CONFIG_PPC32=y |
8 | CONFIG_PPC_MERGE=y | 25 | CONFIG_PPC_MERGE=y |
9 | CONFIG_MMU=y | 26 | CONFIG_MMU=y |
@@ -14,6 +31,7 @@ CONFIG_ARCH_HAS_ILOG2_U32=y | |||
14 | CONFIG_GENERIC_HWEIGHT=y | 31 | CONFIG_GENERIC_HWEIGHT=y |
15 | CONFIG_GENERIC_CALIBRATE_DELAY=y | 32 | CONFIG_GENERIC_CALIBRATE_DELAY=y |
16 | CONFIG_GENERIC_FIND_NEXT_BIT=y | 33 | CONFIG_GENERIC_FIND_NEXT_BIT=y |
34 | # CONFIG_ARCH_NO_VIRT_TO_BUS is not set | ||
17 | CONFIG_PPC=y | 35 | CONFIG_PPC=y |
18 | CONFIG_EARLY_PRINTK=y | 36 | CONFIG_EARLY_PRINTK=y |
19 | CONFIG_GENERIC_NVRAM=y | 37 | CONFIG_GENERIC_NVRAM=y |
@@ -25,28 +43,8 @@ CONFIG_PPC_UDBG_16550=y | |||
25 | CONFIG_AUDIT_ARCH=y | 43 | CONFIG_AUDIT_ARCH=y |
26 | CONFIG_GENERIC_BUG=y | 44 | CONFIG_GENERIC_BUG=y |
27 | CONFIG_DEFAULT_UIMAGE=y | 45 | CONFIG_DEFAULT_UIMAGE=y |
28 | |||
29 | # | ||
30 | # Processor support | ||
31 | # | ||
32 | # CONFIG_CLASSIC32 is not set | ||
33 | # CONFIG_PPC_82xx is not set | ||
34 | # CONFIG_PPC_83xx is not set | ||
35 | CONFIG_PPC_85xx=y | ||
36 | # CONFIG_PPC_86xx is not set | ||
37 | # CONFIG_PPC_8xx is not set | ||
38 | # CONFIG_40x is not set | ||
39 | # CONFIG_44x is not set | ||
40 | # CONFIG_E200 is not set | ||
41 | CONFIG_85xx=y | ||
42 | CONFIG_E500=y | ||
43 | # CONFIG_PPC_DCR_NATIVE is not set | 46 | # CONFIG_PPC_DCR_NATIVE is not set |
44 | # CONFIG_PPC_DCR_MMIO is not set | 47 | # CONFIG_PPC_DCR_MMIO is not set |
45 | CONFIG_BOOKE=y | ||
46 | CONFIG_FSL_BOOKE=y | ||
47 | # CONFIG_PHYS_64BIT is not set | ||
48 | # CONFIG_SPE is not set | ||
49 | # CONFIG_PPC_MM_SLICES is not set | ||
50 | CONFIG_DEFCONFIG_LIST="/lib/modules/$UNAME_RELEASE/.config" | 48 | CONFIG_DEFCONFIG_LIST="/lib/modules/$UNAME_RELEASE/.config" |
51 | 49 | ||
52 | # | 50 | # |
@@ -63,13 +61,12 @@ CONFIG_LOCALVERSION="" | |||
63 | CONFIG_LOCALVERSION_AUTO=y | 61 | CONFIG_LOCALVERSION_AUTO=y |
64 | CONFIG_SWAP=y | 62 | CONFIG_SWAP=y |
65 | CONFIG_SYSVIPC=y | 63 | CONFIG_SYSVIPC=y |
66 | CONFIG_IPC_NS=y | ||
67 | CONFIG_SYSVIPC_SYSCTL=y | 64 | CONFIG_SYSVIPC_SYSCTL=y |
68 | CONFIG_POSIX_MQUEUE=y | 65 | CONFIG_POSIX_MQUEUE=y |
69 | CONFIG_BSD_PROCESS_ACCT=y | 66 | CONFIG_BSD_PROCESS_ACCT=y |
70 | # CONFIG_BSD_PROCESS_ACCT_V3 is not set | 67 | # CONFIG_BSD_PROCESS_ACCT_V3 is not set |
71 | # CONFIG_TASKSTATS is not set | 68 | # CONFIG_TASKSTATS is not set |
72 | # CONFIG_UTS_NS is not set | 69 | # CONFIG_USER_NS is not set |
73 | CONFIG_AUDIT=y | 70 | CONFIG_AUDIT=y |
74 | # CONFIG_AUDITSYSCALL is not set | 71 | # CONFIG_AUDITSYSCALL is not set |
75 | CONFIG_IKCONFIG=y | 72 | CONFIG_IKCONFIG=y |
@@ -86,7 +83,7 @@ CONFIG_SYSCTL_SYSCALL=y | |||
86 | CONFIG_KALLSYMS=y | 83 | CONFIG_KALLSYMS=y |
87 | CONFIG_KALLSYMS_ALL=y | 84 | CONFIG_KALLSYMS_ALL=y |
88 | # CONFIG_KALLSYMS_EXTRA_PASS is not set | 85 | # CONFIG_KALLSYMS_EXTRA_PASS is not set |
89 | # CONFIG_HOTPLUG is not set | 86 | CONFIG_HOTPLUG=y |
90 | CONFIG_PRINTK=y | 87 | CONFIG_PRINTK=y |
91 | CONFIG_BUG=y | 88 | CONFIG_BUG=y |
92 | CONFIG_ELF_CORE=y | 89 | CONFIG_ELF_CORE=y |
@@ -105,24 +102,17 @@ CONFIG_SLAB=y | |||
105 | CONFIG_RT_MUTEXES=y | 102 | CONFIG_RT_MUTEXES=y |
106 | # CONFIG_TINY_SHMEM is not set | 103 | # CONFIG_TINY_SHMEM is not set |
107 | CONFIG_BASE_SMALL=0 | 104 | CONFIG_BASE_SMALL=0 |
108 | |||
109 | # | ||
110 | # Loadable module support | ||
111 | # | ||
112 | CONFIG_MODULES=y | 105 | CONFIG_MODULES=y |
113 | CONFIG_MODULE_UNLOAD=y | 106 | CONFIG_MODULE_UNLOAD=y |
114 | CONFIG_MODULE_FORCE_UNLOAD=y | 107 | CONFIG_MODULE_FORCE_UNLOAD=y |
115 | CONFIG_MODVERSIONS=y | 108 | CONFIG_MODVERSIONS=y |
116 | # CONFIG_MODULE_SRCVERSION_ALL is not set | 109 | # CONFIG_MODULE_SRCVERSION_ALL is not set |
117 | CONFIG_KMOD=y | 110 | CONFIG_KMOD=y |
118 | |||
119 | # | ||
120 | # Block layer | ||
121 | # | ||
122 | CONFIG_BLOCK=y | 111 | CONFIG_BLOCK=y |
123 | CONFIG_LBD=y | 112 | CONFIG_LBD=y |
124 | # CONFIG_BLK_DEV_IO_TRACE is not set | 113 | # CONFIG_BLK_DEV_IO_TRACE is not set |
125 | # CONFIG_LSF is not set | 114 | # CONFIG_LSF is not set |
115 | # CONFIG_BLK_DEV_BSG is not set | ||
126 | 116 | ||
127 | # | 117 | # |
128 | # IO Schedulers | 118 | # IO Schedulers |
@@ -153,7 +143,7 @@ CONFIG_MPC8544_DS=y | |||
153 | CONFIG_MPC85xx=y | 143 | CONFIG_MPC85xx=y |
154 | CONFIG_MPIC=y | 144 | CONFIG_MPIC=y |
155 | # CONFIG_MPIC_WEIRD is not set | 145 | # CONFIG_MPIC_WEIRD is not set |
156 | # CONFIG_PPC_I8259 is not set | 146 | CONFIG_PPC_I8259=y |
157 | # CONFIG_PPC_RTAS is not set | 147 | # CONFIG_PPC_RTAS is not set |
158 | # CONFIG_MMIO_NVRAM is not set | 148 | # CONFIG_MMIO_NVRAM is not set |
159 | # CONFIG_PPC_MPC106 is not set | 149 | # CONFIG_PPC_MPC106 is not set |
@@ -191,6 +181,8 @@ CONFIG_FLAT_NODE_MEM_MAP=y | |||
191 | CONFIG_SPLIT_PTLOCK_CPUS=4 | 181 | CONFIG_SPLIT_PTLOCK_CPUS=4 |
192 | # CONFIG_RESOURCES_64BIT is not set | 182 | # CONFIG_RESOURCES_64BIT is not set |
193 | CONFIG_ZONE_DMA_FLAG=1 | 183 | CONFIG_ZONE_DMA_FLAG=1 |
184 | CONFIG_BOUNCE=y | ||
185 | CONFIG_VIRT_TO_BUS=y | ||
194 | CONFIG_PROC_DEVICETREE=y | 186 | CONFIG_PROC_DEVICETREE=y |
195 | CONFIG_CMDLINE_BOOL=y | 187 | CONFIG_CMDLINE_BOOL=y |
196 | CONFIG_CMDLINE="root=/dev/sda3 rw console=ttyS0,115200" | 188 | CONFIG_CMDLINE="root=/dev/sda3 rw console=ttyS0,115200" |
@@ -205,15 +197,21 @@ CONFIG_ISA_DMA_API=y | |||
205 | # | 197 | # |
206 | CONFIG_ZONE_DMA=y | 198 | CONFIG_ZONE_DMA=y |
207 | CONFIG_PPC_INDIRECT_PCI=y | 199 | CONFIG_PPC_INDIRECT_PCI=y |
208 | CONFIG_PPC_INDIRECT_PCI_BE=y | ||
209 | CONFIG_FSL_SOC=y | 200 | CONFIG_FSL_SOC=y |
210 | # CONFIG_PCI is not set | 201 | CONFIG_FSL_PCI=y |
211 | # CONFIG_PCI_DOMAINS is not set | 202 | CONFIG_PCI=y |
212 | # CONFIG_ARCH_SUPPORTS_MSI is not set | 203 | CONFIG_PCI_DOMAINS=y |
204 | CONFIG_PCI_SYSCALL=y | ||
205 | # CONFIG_PCIEPORTBUS is not set | ||
206 | CONFIG_ARCH_SUPPORTS_MSI=y | ||
207 | # CONFIG_PCI_MSI is not set | ||
208 | # CONFIG_PCI_DEBUG is not set | ||
213 | 209 | ||
214 | # | 210 | # |
215 | # PCCARD (PCMCIA/CardBus) support | 211 | # PCCARD (PCMCIA/CardBus) support |
216 | # | 212 | # |
213 | # CONFIG_PCCARD is not set | ||
214 | # CONFIG_HOTPLUG_PCI is not set | ||
217 | 215 | ||
218 | # | 216 | # |
219 | # Advanced setup | 217 | # Advanced setup |
@@ -254,7 +252,6 @@ CONFIG_ASK_IP_FIB_HASH=y | |||
254 | CONFIG_IP_FIB_HASH=y | 252 | CONFIG_IP_FIB_HASH=y |
255 | CONFIG_IP_MULTIPLE_TABLES=y | 253 | CONFIG_IP_MULTIPLE_TABLES=y |
256 | CONFIG_IP_ROUTE_MULTIPATH=y | 254 | CONFIG_IP_ROUTE_MULTIPATH=y |
257 | # CONFIG_IP_ROUTE_MULTIPATH_CACHED is not set | ||
258 | CONFIG_IP_ROUTE_VERBOSE=y | 255 | CONFIG_IP_ROUTE_VERBOSE=y |
259 | CONFIG_IP_PNP=y | 256 | CONFIG_IP_PNP=y |
260 | CONFIG_IP_PNP_DHCP=y | 257 | CONFIG_IP_PNP_DHCP=y |
@@ -330,6 +327,7 @@ CONFIG_FIB_RULES=y | |||
330 | # CONFIG_MAC80211 is not set | 327 | # CONFIG_MAC80211 is not set |
331 | # CONFIG_IEEE80211 is not set | 328 | # CONFIG_IEEE80211 is not set |
332 | # CONFIG_RFKILL is not set | 329 | # CONFIG_RFKILL is not set |
330 | # CONFIG_NET_9P is not set | ||
333 | 331 | ||
334 | # | 332 | # |
335 | # Device Drivers | 333 | # Device Drivers |
@@ -340,45 +338,35 @@ CONFIG_FIB_RULES=y | |||
340 | # | 338 | # |
341 | CONFIG_STANDALONE=y | 339 | CONFIG_STANDALONE=y |
342 | CONFIG_PREVENT_FIRMWARE_BUILD=y | 340 | CONFIG_PREVENT_FIRMWARE_BUILD=y |
341 | # CONFIG_FW_LOADER is not set | ||
343 | # CONFIG_DEBUG_DRIVER is not set | 342 | # CONFIG_DEBUG_DRIVER is not set |
344 | # CONFIG_DEBUG_DEVRES is not set | 343 | # CONFIG_DEBUG_DEVRES is not set |
345 | # CONFIG_SYS_HYPERVISOR is not set | 344 | # CONFIG_SYS_HYPERVISOR is not set |
346 | |||
347 | # | ||
348 | # Connector - unified userspace <-> kernelspace linker | ||
349 | # | ||
350 | # CONFIG_CONNECTOR is not set | 345 | # CONFIG_CONNECTOR is not set |
351 | # CONFIG_MTD is not set | 346 | # CONFIG_MTD is not set |
352 | |||
353 | # | ||
354 | # Parallel port support | ||
355 | # | ||
356 | # CONFIG_PARPORT is not set | 347 | # CONFIG_PARPORT is not set |
357 | 348 | CONFIG_BLK_DEV=y | |
358 | # | ||
359 | # Plug and Play support | ||
360 | # | ||
361 | # CONFIG_PNPACPI is not set | ||
362 | |||
363 | # | ||
364 | # Block devices | ||
365 | # | ||
366 | # CONFIG_BLK_DEV_FD is not set | 349 | # CONFIG_BLK_DEV_FD is not set |
350 | # CONFIG_BLK_CPQ_DA is not set | ||
351 | # CONFIG_BLK_CPQ_CISS_DA is not set | ||
352 | # CONFIG_BLK_DEV_DAC960 is not set | ||
353 | # CONFIG_BLK_DEV_UMEM is not set | ||
367 | # CONFIG_BLK_DEV_COW_COMMON is not set | 354 | # CONFIG_BLK_DEV_COW_COMMON is not set |
368 | CONFIG_BLK_DEV_LOOP=y | 355 | CONFIG_BLK_DEV_LOOP=y |
369 | # CONFIG_BLK_DEV_CRYPTOLOOP is not set | 356 | # CONFIG_BLK_DEV_CRYPTOLOOP is not set |
370 | CONFIG_BLK_DEV_NBD=y | 357 | CONFIG_BLK_DEV_NBD=y |
358 | # CONFIG_BLK_DEV_SX8 is not set | ||
371 | CONFIG_BLK_DEV_RAM=y | 359 | CONFIG_BLK_DEV_RAM=y |
372 | CONFIG_BLK_DEV_RAM_COUNT=2 | 360 | CONFIG_BLK_DEV_RAM_COUNT=2 |
373 | CONFIG_BLK_DEV_RAM_SIZE=16384 | 361 | CONFIG_BLK_DEV_RAM_SIZE=16384 |
374 | CONFIG_BLK_DEV_RAM_BLOCKSIZE=1024 | 362 | CONFIG_BLK_DEV_RAM_BLOCKSIZE=1024 |
375 | # CONFIG_CDROM_PKTCDVD is not set | 363 | # CONFIG_CDROM_PKTCDVD is not set |
376 | # CONFIG_ATA_OVER_ETH is not set | 364 | # CONFIG_ATA_OVER_ETH is not set |
377 | 365 | CONFIG_MISC_DEVICES=y | |
378 | # | 366 | # CONFIG_PHANTOM is not set |
379 | # Misc devices | 367 | # CONFIG_EEPROM_93CX6 is not set |
380 | # | 368 | # CONFIG_SGI_IOC4 is not set |
381 | # CONFIG_BLINK is not set | 369 | # CONFIG_TIFM_CORE is not set |
382 | # CONFIG_IDE is not set | 370 | # CONFIG_IDE is not set |
383 | 371 | ||
384 | # | 372 | # |
@@ -386,6 +374,7 @@ CONFIG_BLK_DEV_RAM_BLOCKSIZE=1024 | |||
386 | # | 374 | # |
387 | # CONFIG_RAID_ATTRS is not set | 375 | # CONFIG_RAID_ATTRS is not set |
388 | CONFIG_SCSI=y | 376 | CONFIG_SCSI=y |
377 | CONFIG_SCSI_DMA=y | ||
389 | # CONFIG_SCSI_TGT is not set | 378 | # CONFIG_SCSI_TGT is not set |
390 | # CONFIG_SCSI_NETLINK is not set | 379 | # CONFIG_SCSI_NETLINK is not set |
391 | CONFIG_SCSI_PROC_FS=y | 380 | CONFIG_SCSI_PROC_FS=y |
@@ -422,25 +411,120 @@ CONFIG_SCSI_WAIT_SCAN=m | |||
422 | # SCSI low-level drivers | 411 | # SCSI low-level drivers |
423 | # | 412 | # |
424 | # CONFIG_ISCSI_TCP is not set | 413 | # CONFIG_ISCSI_TCP is not set |
414 | # CONFIG_BLK_DEV_3W_XXXX_RAID is not set | ||
415 | # CONFIG_SCSI_3W_9XXX is not set | ||
416 | # CONFIG_SCSI_ACARD is not set | ||
417 | # CONFIG_SCSI_AACRAID is not set | ||
418 | # CONFIG_SCSI_AIC7XXX is not set | ||
419 | # CONFIG_SCSI_AIC7XXX_OLD is not set | ||
420 | # CONFIG_SCSI_AIC79XX is not set | ||
421 | # CONFIG_SCSI_AIC94XX is not set | ||
422 | # CONFIG_SCSI_DPT_I2O is not set | ||
423 | # CONFIG_SCSI_ARCMSR is not set | ||
424 | # CONFIG_MEGARAID_NEWGEN is not set | ||
425 | # CONFIG_MEGARAID_LEGACY is not set | ||
426 | # CONFIG_MEGARAID_SAS is not set | ||
427 | # CONFIG_SCSI_HPTIOP is not set | ||
428 | # CONFIG_SCSI_BUSLOGIC is not set | ||
429 | # CONFIG_SCSI_DMX3191D is not set | ||
430 | # CONFIG_SCSI_EATA is not set | ||
431 | # CONFIG_SCSI_FUTURE_DOMAIN is not set | ||
432 | # CONFIG_SCSI_GDTH is not set | ||
433 | # CONFIG_SCSI_IPS is not set | ||
434 | # CONFIG_SCSI_INITIO is not set | ||
435 | # CONFIG_SCSI_INIA100 is not set | ||
436 | # CONFIG_SCSI_STEX is not set | ||
437 | # CONFIG_SCSI_SYM53C8XX_2 is not set | ||
438 | # CONFIG_SCSI_IPR is not set | ||
439 | # CONFIG_SCSI_QLOGIC_1280 is not set | ||
440 | # CONFIG_SCSI_QLA_FC is not set | ||
441 | # CONFIG_SCSI_QLA_ISCSI is not set | ||
442 | # CONFIG_SCSI_LPFC is not set | ||
443 | # CONFIG_SCSI_DC395x is not set | ||
444 | # CONFIG_SCSI_DC390T is not set | ||
445 | # CONFIG_SCSI_NSP32 is not set | ||
425 | # CONFIG_SCSI_DEBUG is not set | 446 | # CONFIG_SCSI_DEBUG is not set |
447 | # CONFIG_SCSI_SRP is not set | ||
426 | CONFIG_ATA=y | 448 | CONFIG_ATA=y |
427 | # CONFIG_ATA_NONSTANDARD is not set | 449 | # CONFIG_ATA_NONSTANDARD is not set |
450 | # CONFIG_SATA_AHCI is not set | ||
451 | # CONFIG_SATA_SVW is not set | ||
452 | # CONFIG_ATA_PIIX is not set | ||
453 | # CONFIG_SATA_MV is not set | ||
454 | # CONFIG_SATA_NV is not set | ||
455 | # CONFIG_PDC_ADMA is not set | ||
456 | # CONFIG_SATA_QSTOR is not set | ||
457 | # CONFIG_SATA_PROMISE is not set | ||
458 | # CONFIG_SATA_SX4 is not set | ||
459 | # CONFIG_SATA_SIL is not set | ||
460 | # CONFIG_SATA_SIL24 is not set | ||
461 | # CONFIG_SATA_SIS is not set | ||
462 | # CONFIG_SATA_ULI is not set | ||
463 | # CONFIG_SATA_VIA is not set | ||
464 | # CONFIG_SATA_VITESSE is not set | ||
465 | # CONFIG_SATA_INIC162X is not set | ||
466 | # CONFIG_PATA_ALI is not set | ||
467 | # CONFIG_PATA_AMD is not set | ||
468 | # CONFIG_PATA_ARTOP is not set | ||
469 | # CONFIG_PATA_ATIIXP is not set | ||
470 | # CONFIG_PATA_CMD640_PCI is not set | ||
471 | # CONFIG_PATA_CMD64X is not set | ||
472 | # CONFIG_PATA_CS5520 is not set | ||
473 | # CONFIG_PATA_CS5530 is not set | ||
474 | # CONFIG_PATA_CYPRESS is not set | ||
475 | # CONFIG_PATA_EFAR is not set | ||
476 | # CONFIG_ATA_GENERIC is not set | ||
477 | # CONFIG_PATA_HPT366 is not set | ||
478 | # CONFIG_PATA_HPT37X is not set | ||
479 | # CONFIG_PATA_HPT3X2N is not set | ||
480 | # CONFIG_PATA_HPT3X3 is not set | ||
481 | # CONFIG_PATA_IT821X is not set | ||
482 | # CONFIG_PATA_IT8213 is not set | ||
483 | # CONFIG_PATA_JMICRON is not set | ||
484 | # CONFIG_PATA_TRIFLEX is not set | ||
485 | # CONFIG_PATA_MARVELL is not set | ||
486 | # CONFIG_PATA_MPIIX is not set | ||
487 | # CONFIG_PATA_OLDPIIX is not set | ||
488 | # CONFIG_PATA_NETCELL is not set | ||
489 | # CONFIG_PATA_NS87410 is not set | ||
490 | # CONFIG_PATA_OPTI is not set | ||
491 | # CONFIG_PATA_OPTIDMA is not set | ||
492 | # CONFIG_PATA_PDC_OLD is not set | ||
493 | # CONFIG_PATA_RADISYS is not set | ||
494 | # CONFIG_PATA_RZ1000 is not set | ||
495 | # CONFIG_PATA_SC1200 is not set | ||
496 | # CONFIG_PATA_SERVERWORKS is not set | ||
497 | # CONFIG_PATA_PDC2027X is not set | ||
498 | # CONFIG_PATA_SIL680 is not set | ||
499 | # CONFIG_PATA_SIS is not set | ||
500 | # CONFIG_PATA_VIA is not set | ||
501 | # CONFIG_PATA_WINBOND is not set | ||
428 | # CONFIG_PATA_PLATFORM is not set | 502 | # CONFIG_PATA_PLATFORM is not set |
503 | # CONFIG_MD is not set | ||
429 | 504 | ||
430 | # | 505 | # |
431 | # Multi-device support (RAID and LVM) | 506 | # Fusion MPT device support |
432 | # | 507 | # |
433 | # CONFIG_MD is not set | 508 | # CONFIG_FUSION is not set |
434 | # CONFIG_MACINTOSH_DRIVERS is not set | 509 | # CONFIG_FUSION_SPI is not set |
510 | # CONFIG_FUSION_FC is not set | ||
511 | # CONFIG_FUSION_SAS is not set | ||
435 | 512 | ||
436 | # | 513 | # |
437 | # Network device support | 514 | # IEEE 1394 (FireWire) support |
438 | # | 515 | # |
516 | # CONFIG_FIREWIRE is not set | ||
517 | # CONFIG_IEEE1394 is not set | ||
518 | # CONFIG_I2O is not set | ||
519 | # CONFIG_MACINTOSH_DRIVERS is not set | ||
439 | CONFIG_NETDEVICES=y | 520 | CONFIG_NETDEVICES=y |
521 | # CONFIG_NETDEVICES_MULTIQUEUE is not set | ||
440 | # CONFIG_DUMMY is not set | 522 | # CONFIG_DUMMY is not set |
441 | # CONFIG_BONDING is not set | 523 | # CONFIG_BONDING is not set |
524 | # CONFIG_MACVLAN is not set | ||
442 | # CONFIG_EQUALIZER is not set | 525 | # CONFIG_EQUALIZER is not set |
443 | # CONFIG_TUN is not set | 526 | # CONFIG_TUN is not set |
527 | # CONFIG_ARCNET is not set | ||
444 | CONFIG_PHYLIB=y | 528 | CONFIG_PHYLIB=y |
445 | 529 | ||
446 | # | 530 | # |
@@ -454,17 +538,44 @@ CONFIG_PHYLIB=y | |||
454 | CONFIG_VITESSE_PHY=y | 538 | CONFIG_VITESSE_PHY=y |
455 | # CONFIG_SMSC_PHY is not set | 539 | # CONFIG_SMSC_PHY is not set |
456 | # CONFIG_BROADCOM_PHY is not set | 540 | # CONFIG_BROADCOM_PHY is not set |
541 | # CONFIG_ICPLUS_PHY is not set | ||
457 | # CONFIG_FIXED_PHY is not set | 542 | # CONFIG_FIXED_PHY is not set |
458 | |||
459 | # | ||
460 | # Ethernet (10 or 100Mbit) | ||
461 | # | ||
462 | CONFIG_NET_ETHERNET=y | 543 | CONFIG_NET_ETHERNET=y |
463 | CONFIG_MII=y | 544 | CONFIG_MII=y |
545 | # CONFIG_HAPPYMEAL is not set | ||
546 | # CONFIG_SUNGEM is not set | ||
547 | # CONFIG_CASSINI is not set | ||
548 | # CONFIG_NET_VENDOR_3COM is not set | ||
549 | # CONFIG_NET_TULIP is not set | ||
550 | # CONFIG_HP100 is not set | ||
551 | # CONFIG_NET_PCI is not set | ||
464 | CONFIG_NETDEV_1000=y | 552 | CONFIG_NETDEV_1000=y |
553 | # CONFIG_ACENIC is not set | ||
554 | # CONFIG_DL2K is not set | ||
555 | # CONFIG_E1000 is not set | ||
556 | # CONFIG_NS83820 is not set | ||
557 | # CONFIG_HAMACHI is not set | ||
558 | # CONFIG_YELLOWFIN is not set | ||
559 | # CONFIG_R8169 is not set | ||
560 | # CONFIG_SIS190 is not set | ||
561 | # CONFIG_SKGE is not set | ||
562 | # CONFIG_SKY2 is not set | ||
563 | # CONFIG_VIA_VELOCITY is not set | ||
564 | # CONFIG_TIGON3 is not set | ||
565 | # CONFIG_BNX2 is not set | ||
465 | CONFIG_GIANFAR=y | 566 | CONFIG_GIANFAR=y |
466 | CONFIG_GFAR_NAPI=y | 567 | CONFIG_GFAR_NAPI=y |
568 | # CONFIG_QLA3XXX is not set | ||
569 | # CONFIG_ATL1 is not set | ||
467 | CONFIG_NETDEV_10000=y | 570 | CONFIG_NETDEV_10000=y |
571 | # CONFIG_CHELSIO_T1 is not set | ||
572 | # CONFIG_CHELSIO_T3 is not set | ||
573 | # CONFIG_IXGB is not set | ||
574 | # CONFIG_S2IO is not set | ||
575 | # CONFIG_MYRI10GE is not set | ||
576 | # CONFIG_NETXEN_NIC is not set | ||
577 | # CONFIG_MLX4_CORE is not set | ||
578 | # CONFIG_TR is not set | ||
468 | 579 | ||
469 | # | 580 | # |
470 | # Wireless LAN | 581 | # Wireless LAN |
@@ -472,21 +583,16 @@ CONFIG_NETDEV_10000=y | |||
472 | # CONFIG_WLAN_PRE80211 is not set | 583 | # CONFIG_WLAN_PRE80211 is not set |
473 | # CONFIG_WLAN_80211 is not set | 584 | # CONFIG_WLAN_80211 is not set |
474 | # CONFIG_WAN is not set | 585 | # CONFIG_WAN is not set |
586 | # CONFIG_FDDI is not set | ||
587 | # CONFIG_HIPPI is not set | ||
475 | # CONFIG_PPP is not set | 588 | # CONFIG_PPP is not set |
476 | # CONFIG_SLIP is not set | 589 | # CONFIG_SLIP is not set |
590 | # CONFIG_NET_FC is not set | ||
477 | # CONFIG_SHAPER is not set | 591 | # CONFIG_SHAPER is not set |
478 | # CONFIG_NETCONSOLE is not set | 592 | # CONFIG_NETCONSOLE is not set |
479 | # CONFIG_NETPOLL is not set | 593 | # CONFIG_NETPOLL is not set |
480 | # CONFIG_NET_POLL_CONTROLLER is not set | 594 | # CONFIG_NET_POLL_CONTROLLER is not set |
481 | |||
482 | # | ||
483 | # ISDN subsystem | ||
484 | # | ||
485 | # CONFIG_ISDN is not set | 595 | # CONFIG_ISDN is not set |
486 | |||
487 | # | ||
488 | # Telephony Support | ||
489 | # | ||
490 | # CONFIG_PHONE is not set | 596 | # CONFIG_PHONE is not set |
491 | 597 | ||
492 | # | 598 | # |
@@ -521,6 +627,7 @@ CONFIG_INPUT=y | |||
521 | CONFIG_SERIO=y | 627 | CONFIG_SERIO=y |
522 | CONFIG_SERIO_I8042=y | 628 | CONFIG_SERIO_I8042=y |
523 | CONFIG_SERIO_SERPORT=y | 629 | CONFIG_SERIO_SERPORT=y |
630 | # CONFIG_SERIO_PCIPS2 is not set | ||
524 | CONFIG_SERIO_LIBPS2=y | 631 | CONFIG_SERIO_LIBPS2=y |
525 | # CONFIG_SERIO_RAW is not set | 632 | # CONFIG_SERIO_RAW is not set |
526 | # CONFIG_GAMEPORT is not set | 633 | # CONFIG_GAMEPORT is not set |
@@ -539,6 +646,7 @@ CONFIG_HW_CONSOLE=y | |||
539 | # | 646 | # |
540 | CONFIG_SERIAL_8250=y | 647 | CONFIG_SERIAL_8250=y |
541 | CONFIG_SERIAL_8250_CONSOLE=y | 648 | CONFIG_SERIAL_8250_CONSOLE=y |
649 | CONFIG_SERIAL_8250_PCI=y | ||
542 | CONFIG_SERIAL_8250_NR_UARTS=4 | 650 | CONFIG_SERIAL_8250_NR_UARTS=4 |
543 | CONFIG_SERIAL_8250_RUNTIME_UARTS=4 | 651 | CONFIG_SERIAL_8250_RUNTIME_UARTS=4 |
544 | # CONFIG_SERIAL_8250_EXTENDED is not set | 652 | # CONFIG_SERIAL_8250_EXTENDED is not set |
@@ -550,14 +658,11 @@ CONFIG_SERIAL_8250_SHARE_IRQ=y | |||
550 | # CONFIG_SERIAL_UARTLITE is not set | 658 | # CONFIG_SERIAL_UARTLITE is not set |
551 | CONFIG_SERIAL_CORE=y | 659 | CONFIG_SERIAL_CORE=y |
552 | CONFIG_SERIAL_CORE_CONSOLE=y | 660 | CONFIG_SERIAL_CORE_CONSOLE=y |
661 | # CONFIG_SERIAL_JSM is not set | ||
553 | # CONFIG_SERIAL_OF_PLATFORM is not set | 662 | # CONFIG_SERIAL_OF_PLATFORM is not set |
554 | CONFIG_UNIX98_PTYS=y | 663 | CONFIG_UNIX98_PTYS=y |
555 | CONFIG_LEGACY_PTYS=y | 664 | CONFIG_LEGACY_PTYS=y |
556 | CONFIG_LEGACY_PTY_COUNT=256 | 665 | CONFIG_LEGACY_PTY_COUNT=256 |
557 | |||
558 | # | ||
559 | # IPMI | ||
560 | # | ||
561 | # CONFIG_IPMI_HANDLER is not set | 666 | # CONFIG_IPMI_HANDLER is not set |
562 | # CONFIG_WATCHDOG is not set | 667 | # CONFIG_WATCHDOG is not set |
563 | # CONFIG_HW_RANDOM is not set | 668 | # CONFIG_HW_RANDOM is not set |
@@ -565,12 +670,12 @@ CONFIG_NVRAM=y | |||
565 | CONFIG_GEN_RTC=y | 670 | CONFIG_GEN_RTC=y |
566 | CONFIG_GEN_RTC_X=y | 671 | CONFIG_GEN_RTC_X=y |
567 | # CONFIG_R3964 is not set | 672 | # CONFIG_R3964 is not set |
673 | # CONFIG_APPLICOM is not set | ||
674 | # CONFIG_AGP is not set | ||
675 | # CONFIG_DRM is not set | ||
568 | # CONFIG_RAW_DRIVER is not set | 676 | # CONFIG_RAW_DRIVER is not set |
569 | |||
570 | # | ||
571 | # TPM devices | ||
572 | # | ||
573 | # CONFIG_TCG_TPM is not set | 677 | # CONFIG_TCG_TPM is not set |
678 | CONFIG_DEVPORT=y | ||
574 | # CONFIG_I2C is not set | 679 | # CONFIG_I2C is not set |
575 | 680 | ||
576 | # | 681 | # |
@@ -578,11 +683,8 @@ CONFIG_GEN_RTC_X=y | |||
578 | # | 683 | # |
579 | # CONFIG_SPI is not set | 684 | # CONFIG_SPI is not set |
580 | # CONFIG_SPI_MASTER is not set | 685 | # CONFIG_SPI_MASTER is not set |
581 | |||
582 | # | ||
583 | # Dallas's 1-wire bus | ||
584 | # | ||
585 | # CONFIG_W1 is not set | 686 | # CONFIG_W1 is not set |
687 | # CONFIG_POWER_SUPPLY is not set | ||
586 | # CONFIG_HWMON is not set | 688 | # CONFIG_HWMON is not set |
587 | 689 | ||
588 | # | 690 | # |
@@ -655,19 +757,14 @@ CONFIG_DUMMY_CONSOLE=y | |||
655 | # Sound | 757 | # Sound |
656 | # | 758 | # |
657 | # CONFIG_SOUND is not set | 759 | # CONFIG_SOUND is not set |
658 | 760 | CONFIG_HID_SUPPORT=y | |
659 | # | ||
660 | # HID Devices | ||
661 | # | ||
662 | CONFIG_HID=y | 761 | CONFIG_HID=y |
663 | # CONFIG_HID_DEBUG is not set | 762 | # CONFIG_HID_DEBUG is not set |
664 | 763 | CONFIG_USB_SUPPORT=y | |
665 | # | 764 | CONFIG_USB_ARCH_HAS_HCD=y |
666 | # USB support | 765 | CONFIG_USB_ARCH_HAS_OHCI=y |
667 | # | 766 | CONFIG_USB_ARCH_HAS_EHCI=y |
668 | # CONFIG_USB_ARCH_HAS_HCD is not set | 767 | # CONFIG_USB is not set |
669 | # CONFIG_USB_ARCH_HAS_OHCI is not set | ||
670 | # CONFIG_USB_ARCH_HAS_EHCI is not set | ||
671 | 768 | ||
672 | # | 769 | # |
673 | # NOTE: USB_STORAGE enables SCSI, and 'SCSI disk support' | 770 | # NOTE: USB_STORAGE enables SCSI, and 'SCSI disk support' |
@@ -691,14 +788,7 @@ CONFIG_HID=y | |||
691 | # | 788 | # |
692 | # LED Triggers | 789 | # LED Triggers |
693 | # | 790 | # |
694 | 791 | # CONFIG_INFINIBAND is not set | |
695 | # | ||
696 | # InfiniBand support | ||
697 | # | ||
698 | |||
699 | # | ||
700 | # EDAC - error detection and reporting (RAS) (EXPERIMENTAL) | ||
701 | # | ||
702 | 792 | ||
703 | # | 793 | # |
704 | # Real Time Clock | 794 | # Real Time Clock |
@@ -719,19 +809,13 @@ CONFIG_RTC_INTF_DEV=y | |||
719 | # CONFIG_RTC_DRV_TEST is not set | 809 | # CONFIG_RTC_DRV_TEST is not set |
720 | 810 | ||
721 | # | 811 | # |
722 | # I2C RTC drivers | ||
723 | # | ||
724 | |||
725 | # | ||
726 | # SPI RTC drivers | ||
727 | # | ||
728 | |||
729 | # | ||
730 | # Platform RTC drivers | 812 | # Platform RTC drivers |
731 | # | 813 | # |
814 | # CONFIG_RTC_DRV_CMOS is not set | ||
732 | # CONFIG_RTC_DRV_DS1553 is not set | 815 | # CONFIG_RTC_DRV_DS1553 is not set |
733 | # CONFIG_RTC_DRV_DS1742 is not set | 816 | # CONFIG_RTC_DRV_DS1742 is not set |
734 | # CONFIG_RTC_DRV_M48T86 is not set | 817 | # CONFIG_RTC_DRV_M48T86 is not set |
818 | # CONFIG_RTC_DRV_M48T59 is not set | ||
735 | # CONFIG_RTC_DRV_V3020 is not set | 819 | # CONFIG_RTC_DRV_V3020 is not set |
736 | 820 | ||
737 | # | 821 | # |
@@ -752,6 +836,11 @@ CONFIG_RTC_INTF_DEV=y | |||
752 | # | 836 | # |
753 | 837 | ||
754 | # | 838 | # |
839 | # Userspace I/O | ||
840 | # | ||
841 | # CONFIG_UIO is not set | ||
842 | |||
843 | # | ||
755 | # File systems | 844 | # File systems |
756 | # | 845 | # |
757 | CONFIG_EXT2_FS=y | 846 | CONFIG_EXT2_FS=y |
@@ -859,7 +948,6 @@ CONFIG_RPCSEC_GSS_KRB5=y | |||
859 | # CONFIG_NCP_FS is not set | 948 | # CONFIG_NCP_FS is not set |
860 | # CONFIG_CODA_FS is not set | 949 | # CONFIG_CODA_FS is not set |
861 | # CONFIG_AFS_FS is not set | 950 | # CONFIG_AFS_FS is not set |
862 | # CONFIG_9P_FS is not set | ||
863 | 951 | ||
864 | # | 952 | # |
865 | # Partition Types | 953 | # Partition Types |
@@ -941,6 +1029,7 @@ CONFIG_BITREVERSE=y | |||
941 | # CONFIG_CRC16 is not set | 1029 | # CONFIG_CRC16 is not set |
942 | # CONFIG_CRC_ITU_T is not set | 1030 | # CONFIG_CRC_ITU_T is not set |
943 | CONFIG_CRC32=y | 1031 | CONFIG_CRC32=y |
1032 | # CONFIG_CRC7 is not set | ||
944 | CONFIG_LIBCRC32C=m | 1033 | CONFIG_LIBCRC32C=m |
945 | CONFIG_ZLIB_INFLATE=y | 1034 | CONFIG_ZLIB_INFLATE=y |
946 | CONFIG_PLIST=y | 1035 | CONFIG_PLIST=y |
@@ -965,6 +1054,7 @@ CONFIG_ENABLE_MUST_CHECK=y | |||
965 | CONFIG_DEBUG_KERNEL=y | 1054 | CONFIG_DEBUG_KERNEL=y |
966 | # CONFIG_DEBUG_SHIRQ is not set | 1055 | # CONFIG_DEBUG_SHIRQ is not set |
967 | CONFIG_DETECT_SOFTLOCKUP=y | 1056 | CONFIG_DETECT_SOFTLOCKUP=y |
1057 | CONFIG_SCHED_DEBUG=y | ||
968 | # CONFIG_SCHEDSTATS is not set | 1058 | # CONFIG_SCHEDSTATS is not set |
969 | # CONFIG_TIMER_STATS is not set | 1059 | # CONFIG_TIMER_STATS is not set |
970 | # CONFIG_DEBUG_SLAB is not set | 1060 | # CONFIG_DEBUG_SLAB is not set |
@@ -996,10 +1086,6 @@ CONFIG_FORCED_INLINING=y | |||
996 | # | 1086 | # |
997 | # CONFIG_KEYS is not set | 1087 | # CONFIG_KEYS is not set |
998 | # CONFIG_SECURITY is not set | 1088 | # CONFIG_SECURITY is not set |
999 | |||
1000 | # | ||
1001 | # Cryptographic options | ||
1002 | # | ||
1003 | CONFIG_CRYPTO=y | 1089 | CONFIG_CRYPTO=y |
1004 | CONFIG_CRYPTO_ALGAPI=y | 1090 | CONFIG_CRYPTO_ALGAPI=y |
1005 | CONFIG_CRYPTO_BLKCIPHER=y | 1091 | CONFIG_CRYPTO_BLKCIPHER=y |
@@ -1038,7 +1124,4 @@ CONFIG_CRYPTO_DES=y | |||
1038 | # CONFIG_CRYPTO_CRC32C is not set | 1124 | # CONFIG_CRYPTO_CRC32C is not set |
1039 | # CONFIG_CRYPTO_CAMELLIA is not set | 1125 | # CONFIG_CRYPTO_CAMELLIA is not set |
1040 | # CONFIG_CRYPTO_TEST is not set | 1126 | # CONFIG_CRYPTO_TEST is not set |
1041 | 1127 | CONFIG_CRYPTO_HW=y | |
1042 | # | ||
1043 | # Hardware crypto devices | ||
1044 | # | ||
diff --git a/arch/powerpc/configs/mpc8568mds_defconfig b/arch/powerpc/configs/mpc8568mds_defconfig index 6451d4dd28a0..417d3e6abcdf 100644 --- a/arch/powerpc/configs/mpc8568mds_defconfig +++ b/arch/powerpc/configs/mpc8568mds_defconfig | |||
@@ -1,9 +1,26 @@ | |||
1 | # | 1 | # |
2 | # Automatically generated make config: don't edit | 2 | # Automatically generated make config: don't edit |
3 | # Linux kernel version: 2.6.22-rc7 | 3 | # Linux kernel version: 2.6.22 |
4 | # Sun Jul 1 23:56:59 2007 | 4 | # Fri Jul 20 13:55:04 2007 |
5 | # | 5 | # |
6 | # CONFIG_PPC64 is not set | 6 | # CONFIG_PPC64 is not set |
7 | |||
8 | # | ||
9 | # Processor support | ||
10 | # | ||
11 | # CONFIG_6xx is not set | ||
12 | CONFIG_PPC_85xx=y | ||
13 | # CONFIG_PPC_8xx is not set | ||
14 | # CONFIG_40x is not set | ||
15 | # CONFIG_44x is not set | ||
16 | # CONFIG_E200 is not set | ||
17 | CONFIG_85xx=y | ||
18 | CONFIG_E500=y | ||
19 | CONFIG_BOOKE=y | ||
20 | CONFIG_FSL_BOOKE=y | ||
21 | # CONFIG_PHYS_64BIT is not set | ||
22 | CONFIG_SPE=y | ||
23 | # CONFIG_PPC_MM_SLICES is not set | ||
7 | CONFIG_PPC32=y | 24 | CONFIG_PPC32=y |
8 | CONFIG_PPC_MERGE=y | 25 | CONFIG_PPC_MERGE=y |
9 | CONFIG_MMU=y | 26 | CONFIG_MMU=y |
@@ -14,6 +31,7 @@ CONFIG_ARCH_HAS_ILOG2_U32=y | |||
14 | CONFIG_GENERIC_HWEIGHT=y | 31 | CONFIG_GENERIC_HWEIGHT=y |
15 | CONFIG_GENERIC_CALIBRATE_DELAY=y | 32 | CONFIG_GENERIC_CALIBRATE_DELAY=y |
16 | CONFIG_GENERIC_FIND_NEXT_BIT=y | 33 | CONFIG_GENERIC_FIND_NEXT_BIT=y |
34 | # CONFIG_ARCH_NO_VIRT_TO_BUS is not set | ||
17 | CONFIG_PPC=y | 35 | CONFIG_PPC=y |
18 | CONFIG_EARLY_PRINTK=y | 36 | CONFIG_EARLY_PRINTK=y |
19 | CONFIG_GENERIC_NVRAM=y | 37 | CONFIG_GENERIC_NVRAM=y |
@@ -25,28 +43,8 @@ CONFIG_PPC_UDBG_16550=y | |||
25 | CONFIG_AUDIT_ARCH=y | 43 | CONFIG_AUDIT_ARCH=y |
26 | CONFIG_GENERIC_BUG=y | 44 | CONFIG_GENERIC_BUG=y |
27 | CONFIG_DEFAULT_UIMAGE=y | 45 | CONFIG_DEFAULT_UIMAGE=y |
28 | |||
29 | # | ||
30 | # Processor support | ||
31 | # | ||
32 | # CONFIG_CLASSIC32 is not set | ||
33 | # CONFIG_PPC_82xx is not set | ||
34 | # CONFIG_PPC_83xx is not set | ||
35 | CONFIG_PPC_85xx=y | ||
36 | # CONFIG_PPC_86xx is not set | ||
37 | # CONFIG_PPC_8xx is not set | ||
38 | # CONFIG_40x is not set | ||
39 | # CONFIG_44x is not set | ||
40 | # CONFIG_E200 is not set | ||
41 | CONFIG_85xx=y | ||
42 | CONFIG_E500=y | ||
43 | # CONFIG_PPC_DCR_NATIVE is not set | 46 | # CONFIG_PPC_DCR_NATIVE is not set |
44 | # CONFIG_PPC_DCR_MMIO is not set | 47 | # CONFIG_PPC_DCR_MMIO is not set |
45 | CONFIG_BOOKE=y | ||
46 | CONFIG_FSL_BOOKE=y | ||
47 | # CONFIG_PHYS_64BIT is not set | ||
48 | CONFIG_SPE=y | ||
49 | # CONFIG_PPC_MM_SLICES is not set | ||
50 | CONFIG_DEFCONFIG_LIST="/lib/modules/$UNAME_RELEASE/.config" | 48 | CONFIG_DEFCONFIG_LIST="/lib/modules/$UNAME_RELEASE/.config" |
51 | 49 | ||
52 | # | 50 | # |
@@ -63,12 +61,11 @@ CONFIG_LOCALVERSION="" | |||
63 | CONFIG_LOCALVERSION_AUTO=y | 61 | CONFIG_LOCALVERSION_AUTO=y |
64 | CONFIG_SWAP=y | 62 | CONFIG_SWAP=y |
65 | CONFIG_SYSVIPC=y | 63 | CONFIG_SYSVIPC=y |
66 | # CONFIG_IPC_NS is not set | ||
67 | CONFIG_SYSVIPC_SYSCTL=y | 64 | CONFIG_SYSVIPC_SYSCTL=y |
68 | # CONFIG_POSIX_MQUEUE is not set | 65 | # CONFIG_POSIX_MQUEUE is not set |
69 | # CONFIG_BSD_PROCESS_ACCT is not set | 66 | # CONFIG_BSD_PROCESS_ACCT is not set |
70 | # CONFIG_TASKSTATS is not set | 67 | # CONFIG_TASKSTATS is not set |
71 | # CONFIG_UTS_NS is not set | 68 | # CONFIG_USER_NS is not set |
72 | # CONFIG_AUDIT is not set | 69 | # CONFIG_AUDIT is not set |
73 | # CONFIG_IKCONFIG is not set | 70 | # CONFIG_IKCONFIG is not set |
74 | CONFIG_LOG_BUF_SHIFT=14 | 71 | CONFIG_LOG_BUF_SHIFT=14 |
@@ -100,24 +97,17 @@ CONFIG_SLAB=y | |||
100 | CONFIG_RT_MUTEXES=y | 97 | CONFIG_RT_MUTEXES=y |
101 | # CONFIG_TINY_SHMEM is not set | 98 | # CONFIG_TINY_SHMEM is not set |
102 | CONFIG_BASE_SMALL=0 | 99 | CONFIG_BASE_SMALL=0 |
103 | |||
104 | # | ||
105 | # Loadable module support | ||
106 | # | ||
107 | CONFIG_MODULES=y | 100 | CONFIG_MODULES=y |
108 | CONFIG_MODULE_UNLOAD=y | 101 | CONFIG_MODULE_UNLOAD=y |
109 | # CONFIG_MODULE_FORCE_UNLOAD is not set | 102 | # CONFIG_MODULE_FORCE_UNLOAD is not set |
110 | # CONFIG_MODVERSIONS is not set | 103 | # CONFIG_MODVERSIONS is not set |
111 | # CONFIG_MODULE_SRCVERSION_ALL is not set | 104 | # CONFIG_MODULE_SRCVERSION_ALL is not set |
112 | # CONFIG_KMOD is not set | 105 | # CONFIG_KMOD is not set |
113 | |||
114 | # | ||
115 | # Block layer | ||
116 | # | ||
117 | CONFIG_BLOCK=y | 106 | CONFIG_BLOCK=y |
118 | # CONFIG_LBD is not set | 107 | # CONFIG_LBD is not set |
119 | # CONFIG_BLK_DEV_IO_TRACE is not set | 108 | # CONFIG_BLK_DEV_IO_TRACE is not set |
120 | # CONFIG_LSF is not set | 109 | # CONFIG_LSF is not set |
110 | # CONFIG_BLK_DEV_BSG is not set | ||
121 | 111 | ||
122 | # | 112 | # |
123 | # IO Schedulers | 113 | # IO Schedulers |
@@ -186,6 +176,8 @@ CONFIG_FLAT_NODE_MEM_MAP=y | |||
186 | CONFIG_SPLIT_PTLOCK_CPUS=4 | 176 | CONFIG_SPLIT_PTLOCK_CPUS=4 |
187 | # CONFIG_RESOURCES_64BIT is not set | 177 | # CONFIG_RESOURCES_64BIT is not set |
188 | CONFIG_ZONE_DMA_FLAG=1 | 178 | CONFIG_ZONE_DMA_FLAG=1 |
179 | CONFIG_BOUNCE=y | ||
180 | CONFIG_VIRT_TO_BUS=y | ||
189 | CONFIG_PROC_DEVICETREE=y | 181 | CONFIG_PROC_DEVICETREE=y |
190 | # CONFIG_CMDLINE_BOOL is not set | 182 | # CONFIG_CMDLINE_BOOL is not set |
191 | # CONFIG_PM is not set | 183 | # CONFIG_PM is not set |
@@ -201,14 +193,20 @@ CONFIG_ZONE_DMA=y | |||
201 | CONFIG_PPC_INDIRECT_PCI=y | 193 | CONFIG_PPC_INDIRECT_PCI=y |
202 | CONFIG_PPC_INDIRECT_PCI_BE=y | 194 | CONFIG_PPC_INDIRECT_PCI_BE=y |
203 | CONFIG_FSL_SOC=y | 195 | CONFIG_FSL_SOC=y |
204 | # CONFIG_PCI is not set | 196 | CONFIG_FSL_PCI=y |
205 | # CONFIG_PCI_DOMAINS is not set | 197 | CONFIG_PCI=y |
206 | # CONFIG_ARCH_SUPPORTS_MSI is not set | 198 | CONFIG_PCI_DOMAINS=y |
199 | CONFIG_PCI_SYSCALL=y | ||
200 | # CONFIG_PCIEPORTBUS is not set | ||
201 | CONFIG_ARCH_SUPPORTS_MSI=y | ||
202 | # CONFIG_PCI_MSI is not set | ||
203 | # CONFIG_PCI_DEBUG is not set | ||
207 | 204 | ||
208 | # | 205 | # |
209 | # PCCARD (PCMCIA/CardBus) support | 206 | # PCCARD (PCMCIA/CardBus) support |
210 | # | 207 | # |
211 | # CONFIG_PCCARD is not set | 208 | # CONFIG_PCCARD is not set |
209 | # CONFIG_HOTPLUG_PCI is not set | ||
212 | 210 | ||
213 | # | 211 | # |
214 | # Advanced setup | 212 | # Advanced setup |
@@ -309,6 +307,7 @@ CONFIG_DEFAULT_TCP_CONG="cubic" | |||
309 | # CONFIG_MAC80211 is not set | 307 | # CONFIG_MAC80211 is not set |
310 | # CONFIG_IEEE80211 is not set | 308 | # CONFIG_IEEE80211 is not set |
311 | # CONFIG_RFKILL is not set | 309 | # CONFIG_RFKILL is not set |
310 | # CONFIG_NET_9P is not set | ||
312 | 311 | ||
313 | # | 312 | # |
314 | # Device Drivers | 313 | # Device Drivers |
@@ -323,42 +322,31 @@ CONFIG_PREVENT_FIRMWARE_BUILD=y | |||
323 | # CONFIG_DEBUG_DRIVER is not set | 322 | # CONFIG_DEBUG_DRIVER is not set |
324 | # CONFIG_DEBUG_DEVRES is not set | 323 | # CONFIG_DEBUG_DEVRES is not set |
325 | # CONFIG_SYS_HYPERVISOR is not set | 324 | # CONFIG_SYS_HYPERVISOR is not set |
326 | |||
327 | # | ||
328 | # Connector - unified userspace <-> kernelspace linker | ||
329 | # | ||
330 | # CONFIG_CONNECTOR is not set | 325 | # CONFIG_CONNECTOR is not set |
331 | # CONFIG_MTD is not set | 326 | # CONFIG_MTD is not set |
332 | |||
333 | # | ||
334 | # Parallel port support | ||
335 | # | ||
336 | # CONFIG_PARPORT is not set | 327 | # CONFIG_PARPORT is not set |
337 | 328 | CONFIG_BLK_DEV=y | |
338 | # | ||
339 | # Plug and Play support | ||
340 | # | ||
341 | # CONFIG_PNPACPI is not set | ||
342 | |||
343 | # | ||
344 | # Block devices | ||
345 | # | ||
346 | # CONFIG_BLK_DEV_FD is not set | 329 | # CONFIG_BLK_DEV_FD is not set |
330 | # CONFIG_BLK_CPQ_DA is not set | ||
331 | # CONFIG_BLK_CPQ_CISS_DA is not set | ||
332 | # CONFIG_BLK_DEV_DAC960 is not set | ||
333 | # CONFIG_BLK_DEV_UMEM is not set | ||
347 | # CONFIG_BLK_DEV_COW_COMMON is not set | 334 | # CONFIG_BLK_DEV_COW_COMMON is not set |
348 | CONFIG_BLK_DEV_LOOP=y | 335 | CONFIG_BLK_DEV_LOOP=y |
349 | # CONFIG_BLK_DEV_CRYPTOLOOP is not set | 336 | # CONFIG_BLK_DEV_CRYPTOLOOP is not set |
350 | # CONFIG_BLK_DEV_NBD is not set | 337 | # CONFIG_BLK_DEV_NBD is not set |
338 | # CONFIG_BLK_DEV_SX8 is not set | ||
351 | CONFIG_BLK_DEV_RAM=y | 339 | CONFIG_BLK_DEV_RAM=y |
352 | CONFIG_BLK_DEV_RAM_COUNT=16 | 340 | CONFIG_BLK_DEV_RAM_COUNT=16 |
353 | CONFIG_BLK_DEV_RAM_SIZE=32768 | 341 | CONFIG_BLK_DEV_RAM_SIZE=32768 |
354 | CONFIG_BLK_DEV_RAM_BLOCKSIZE=1024 | 342 | CONFIG_BLK_DEV_RAM_BLOCKSIZE=1024 |
355 | # CONFIG_CDROM_PKTCDVD is not set | 343 | # CONFIG_CDROM_PKTCDVD is not set |
356 | # CONFIG_ATA_OVER_ETH is not set | 344 | # CONFIG_ATA_OVER_ETH is not set |
357 | 345 | CONFIG_MISC_DEVICES=y | |
358 | # | 346 | # CONFIG_PHANTOM is not set |
359 | # Misc devices | 347 | # CONFIG_EEPROM_93CX6 is not set |
360 | # | 348 | # CONFIG_SGI_IOC4 is not set |
361 | # CONFIG_BLINK is not set | 349 | # CONFIG_TIFM_CORE is not set |
362 | # CONFIG_IDE is not set | 350 | # CONFIG_IDE is not set |
363 | 351 | ||
364 | # | 352 | # |
@@ -366,6 +354,7 @@ CONFIG_BLK_DEV_RAM_BLOCKSIZE=1024 | |||
366 | # | 354 | # |
367 | # CONFIG_RAID_ATTRS is not set | 355 | # CONFIG_RAID_ATTRS is not set |
368 | CONFIG_SCSI=y | 356 | CONFIG_SCSI=y |
357 | CONFIG_SCSI_DMA=y | ||
369 | # CONFIG_SCSI_TGT is not set | 358 | # CONFIG_SCSI_TGT is not set |
370 | # CONFIG_SCSI_NETLINK is not set | 359 | # CONFIG_SCSI_NETLINK is not set |
371 | CONFIG_SCSI_PROC_FS=y | 360 | CONFIG_SCSI_PROC_FS=y |
@@ -402,23 +391,65 @@ CONFIG_SCSI_WAIT_SCAN=m | |||
402 | # SCSI low-level drivers | 391 | # SCSI low-level drivers |
403 | # | 392 | # |
404 | # CONFIG_ISCSI_TCP is not set | 393 | # CONFIG_ISCSI_TCP is not set |
394 | # CONFIG_BLK_DEV_3W_XXXX_RAID is not set | ||
395 | # CONFIG_SCSI_3W_9XXX is not set | ||
396 | # CONFIG_SCSI_ACARD is not set | ||
397 | # CONFIG_SCSI_AACRAID is not set | ||
398 | # CONFIG_SCSI_AIC7XXX is not set | ||
399 | # CONFIG_SCSI_AIC7XXX_OLD is not set | ||
400 | # CONFIG_SCSI_AIC79XX is not set | ||
401 | # CONFIG_SCSI_AIC94XX is not set | ||
402 | # CONFIG_SCSI_DPT_I2O is not set | ||
403 | # CONFIG_SCSI_ARCMSR is not set | ||
404 | # CONFIG_MEGARAID_NEWGEN is not set | ||
405 | # CONFIG_MEGARAID_LEGACY is not set | ||
406 | # CONFIG_MEGARAID_SAS is not set | ||
407 | # CONFIG_SCSI_HPTIOP is not set | ||
408 | # CONFIG_SCSI_BUSLOGIC is not set | ||
409 | # CONFIG_SCSI_DMX3191D is not set | ||
410 | # CONFIG_SCSI_EATA is not set | ||
411 | # CONFIG_SCSI_FUTURE_DOMAIN is not set | ||
412 | # CONFIG_SCSI_GDTH is not set | ||
413 | # CONFIG_SCSI_IPS is not set | ||
414 | # CONFIG_SCSI_INITIO is not set | ||
415 | # CONFIG_SCSI_INIA100 is not set | ||
416 | # CONFIG_SCSI_STEX is not set | ||
417 | # CONFIG_SCSI_SYM53C8XX_2 is not set | ||
418 | # CONFIG_SCSI_QLOGIC_1280 is not set | ||
419 | # CONFIG_SCSI_QLA_FC is not set | ||
420 | # CONFIG_SCSI_QLA_ISCSI is not set | ||
421 | # CONFIG_SCSI_LPFC is not set | ||
422 | # CONFIG_SCSI_DC395x is not set | ||
423 | # CONFIG_SCSI_DC390T is not set | ||
424 | # CONFIG_SCSI_NSP32 is not set | ||
405 | # CONFIG_SCSI_DEBUG is not set | 425 | # CONFIG_SCSI_DEBUG is not set |
426 | # CONFIG_SCSI_SRP is not set | ||
406 | # CONFIG_ATA is not set | 427 | # CONFIG_ATA is not set |
428 | # CONFIG_MD is not set | ||
407 | 429 | ||
408 | # | 430 | # |
409 | # Multi-device support (RAID and LVM) | 431 | # Fusion MPT device support |
410 | # | 432 | # |
411 | # CONFIG_MD is not set | 433 | # CONFIG_FUSION is not set |
412 | # CONFIG_MACINTOSH_DRIVERS is not set | 434 | # CONFIG_FUSION_SPI is not set |
435 | # CONFIG_FUSION_FC is not set | ||
436 | # CONFIG_FUSION_SAS is not set | ||
413 | 437 | ||
414 | # | 438 | # |
415 | # Network device support | 439 | # IEEE 1394 (FireWire) support |
416 | # | 440 | # |
441 | # CONFIG_FIREWIRE is not set | ||
442 | # CONFIG_IEEE1394 is not set | ||
443 | # CONFIG_I2O is not set | ||
444 | # CONFIG_MACINTOSH_DRIVERS is not set | ||
417 | CONFIG_NETDEVICES=y | 445 | CONFIG_NETDEVICES=y |
446 | # CONFIG_NETDEVICES_MULTIQUEUE is not set | ||
418 | # CONFIG_DUMMY is not set | 447 | # CONFIG_DUMMY is not set |
419 | # CONFIG_BONDING is not set | 448 | # CONFIG_BONDING is not set |
449 | # CONFIG_MACVLAN is not set | ||
420 | # CONFIG_EQUALIZER is not set | 450 | # CONFIG_EQUALIZER is not set |
421 | # CONFIG_TUN is not set | 451 | # CONFIG_TUN is not set |
452 | # CONFIG_ARCNET is not set | ||
422 | CONFIG_PHYLIB=y | 453 | CONFIG_PHYLIB=y |
423 | 454 | ||
424 | # | 455 | # |
@@ -432,17 +463,44 @@ CONFIG_MARVELL_PHY=y | |||
432 | # CONFIG_VITESSE_PHY is not set | 463 | # CONFIG_VITESSE_PHY is not set |
433 | # CONFIG_SMSC_PHY is not set | 464 | # CONFIG_SMSC_PHY is not set |
434 | # CONFIG_BROADCOM_PHY is not set | 465 | # CONFIG_BROADCOM_PHY is not set |
466 | # CONFIG_ICPLUS_PHY is not set | ||
435 | # CONFIG_FIXED_PHY is not set | 467 | # CONFIG_FIXED_PHY is not set |
436 | |||
437 | # | ||
438 | # Ethernet (10 or 100Mbit) | ||
439 | # | ||
440 | CONFIG_NET_ETHERNET=y | 468 | CONFIG_NET_ETHERNET=y |
441 | CONFIG_MII=y | 469 | CONFIG_MII=y |
470 | # CONFIG_HAPPYMEAL is not set | ||
471 | # CONFIG_SUNGEM is not set | ||
472 | # CONFIG_CASSINI is not set | ||
473 | # CONFIG_NET_VENDOR_3COM is not set | ||
474 | # CONFIG_NET_TULIP is not set | ||
475 | # CONFIG_HP100 is not set | ||
476 | # CONFIG_NET_PCI is not set | ||
442 | CONFIG_NETDEV_1000=y | 477 | CONFIG_NETDEV_1000=y |
478 | # CONFIG_ACENIC is not set | ||
479 | # CONFIG_DL2K is not set | ||
480 | # CONFIG_E1000 is not set | ||
481 | # CONFIG_NS83820 is not set | ||
482 | # CONFIG_HAMACHI is not set | ||
483 | # CONFIG_YELLOWFIN is not set | ||
484 | # CONFIG_R8169 is not set | ||
485 | # CONFIG_SIS190 is not set | ||
486 | # CONFIG_SKGE is not set | ||
487 | # CONFIG_SKY2 is not set | ||
488 | # CONFIG_VIA_VELOCITY is not set | ||
489 | # CONFIG_TIGON3 is not set | ||
490 | # CONFIG_BNX2 is not set | ||
443 | CONFIG_GIANFAR=y | 491 | CONFIG_GIANFAR=y |
444 | CONFIG_GFAR_NAPI=y | 492 | CONFIG_GFAR_NAPI=y |
493 | # CONFIG_QLA3XXX is not set | ||
494 | # CONFIG_ATL1 is not set | ||
445 | CONFIG_NETDEV_10000=y | 495 | CONFIG_NETDEV_10000=y |
496 | # CONFIG_CHELSIO_T1 is not set | ||
497 | # CONFIG_CHELSIO_T3 is not set | ||
498 | # CONFIG_IXGB is not set | ||
499 | # CONFIG_S2IO is not set | ||
500 | # CONFIG_MYRI10GE is not set | ||
501 | # CONFIG_NETXEN_NIC is not set | ||
502 | # CONFIG_MLX4_CORE is not set | ||
503 | # CONFIG_TR is not set | ||
446 | 504 | ||
447 | # | 505 | # |
448 | # Wireless LAN | 506 | # Wireless LAN |
@@ -450,21 +508,16 @@ CONFIG_NETDEV_10000=y | |||
450 | # CONFIG_WLAN_PRE80211 is not set | 508 | # CONFIG_WLAN_PRE80211 is not set |
451 | # CONFIG_WLAN_80211 is not set | 509 | # CONFIG_WLAN_80211 is not set |
452 | # CONFIG_WAN is not set | 510 | # CONFIG_WAN is not set |
511 | # CONFIG_FDDI is not set | ||
512 | # CONFIG_HIPPI is not set | ||
453 | # CONFIG_PPP is not set | 513 | # CONFIG_PPP is not set |
454 | # CONFIG_SLIP is not set | 514 | # CONFIG_SLIP is not set |
515 | # CONFIG_NET_FC is not set | ||
455 | # CONFIG_SHAPER is not set | 516 | # CONFIG_SHAPER is not set |
456 | # CONFIG_NETCONSOLE is not set | 517 | # CONFIG_NETCONSOLE is not set |
457 | # CONFIG_NETPOLL is not set | 518 | # CONFIG_NETPOLL is not set |
458 | # CONFIG_NET_POLL_CONTROLLER is not set | 519 | # CONFIG_NET_POLL_CONTROLLER is not set |
459 | |||
460 | # | ||
461 | # ISDN subsystem | ||
462 | # | ||
463 | # CONFIG_ISDN is not set | 520 | # CONFIG_ISDN is not set |
464 | |||
465 | # | ||
466 | # Telephony Support | ||
467 | # | ||
468 | # CONFIG_PHONE is not set | 521 | # CONFIG_PHONE is not set |
469 | 522 | ||
470 | # | 523 | # |
@@ -510,6 +563,7 @@ CONFIG_INPUT=y | |||
510 | # | 563 | # |
511 | CONFIG_SERIAL_8250=y | 564 | CONFIG_SERIAL_8250=y |
512 | CONFIG_SERIAL_8250_CONSOLE=y | 565 | CONFIG_SERIAL_8250_CONSOLE=y |
566 | CONFIG_SERIAL_8250_PCI=y | ||
513 | CONFIG_SERIAL_8250_NR_UARTS=4 | 567 | CONFIG_SERIAL_8250_NR_UARTS=4 |
514 | CONFIG_SERIAL_8250_RUNTIME_UARTS=4 | 568 | CONFIG_SERIAL_8250_RUNTIME_UARTS=4 |
515 | # CONFIG_SERIAL_8250_EXTENDED is not set | 569 | # CONFIG_SERIAL_8250_EXTENDED is not set |
@@ -521,14 +575,11 @@ CONFIG_SERIAL_8250_SHARE_IRQ=y | |||
521 | # CONFIG_SERIAL_UARTLITE is not set | 575 | # CONFIG_SERIAL_UARTLITE is not set |
522 | CONFIG_SERIAL_CORE=y | 576 | CONFIG_SERIAL_CORE=y |
523 | CONFIG_SERIAL_CORE_CONSOLE=y | 577 | CONFIG_SERIAL_CORE_CONSOLE=y |
578 | # CONFIG_SERIAL_JSM is not set | ||
524 | # CONFIG_SERIAL_OF_PLATFORM is not set | 579 | # CONFIG_SERIAL_OF_PLATFORM is not set |
525 | CONFIG_UNIX98_PTYS=y | 580 | CONFIG_UNIX98_PTYS=y |
526 | CONFIG_LEGACY_PTYS=y | 581 | CONFIG_LEGACY_PTYS=y |
527 | CONFIG_LEGACY_PTY_COUNT=256 | 582 | CONFIG_LEGACY_PTY_COUNT=256 |
528 | |||
529 | # | ||
530 | # IPMI | ||
531 | # | ||
532 | # CONFIG_IPMI_HANDLER is not set | 583 | # CONFIG_IPMI_HANDLER is not set |
533 | CONFIG_WATCHDOG=y | 584 | CONFIG_WATCHDOG=y |
534 | # CONFIG_WATCHDOG_NOWAYOUT is not set | 585 | # CONFIG_WATCHDOG_NOWAYOUT is not set |
@@ -538,17 +589,23 @@ CONFIG_WATCHDOG=y | |||
538 | # | 589 | # |
539 | # CONFIG_SOFT_WATCHDOG is not set | 590 | # CONFIG_SOFT_WATCHDOG is not set |
540 | # CONFIG_BOOKE_WDT is not set | 591 | # CONFIG_BOOKE_WDT is not set |
592 | |||
593 | # | ||
594 | # PCI-based Watchdog Cards | ||
595 | # | ||
596 | # CONFIG_PCIPCWATCHDOG is not set | ||
597 | # CONFIG_WDTPCI is not set | ||
541 | CONFIG_HW_RANDOM=y | 598 | CONFIG_HW_RANDOM=y |
542 | # CONFIG_NVRAM is not set | 599 | # CONFIG_NVRAM is not set |
543 | CONFIG_GEN_RTC=y | 600 | CONFIG_GEN_RTC=y |
544 | # CONFIG_GEN_RTC_X is not set | 601 | # CONFIG_GEN_RTC_X is not set |
545 | # CONFIG_R3964 is not set | 602 | # CONFIG_R3964 is not set |
603 | # CONFIG_APPLICOM is not set | ||
604 | # CONFIG_AGP is not set | ||
605 | # CONFIG_DRM is not set | ||
546 | # CONFIG_RAW_DRIVER is not set | 606 | # CONFIG_RAW_DRIVER is not set |
547 | |||
548 | # | ||
549 | # TPM devices | ||
550 | # | ||
551 | # CONFIG_TCG_TPM is not set | 607 | # CONFIG_TCG_TPM is not set |
608 | CONFIG_DEVPORT=y | ||
552 | CONFIG_I2C=y | 609 | CONFIG_I2C=y |
553 | CONFIG_I2C_BOARDINFO=y | 610 | CONFIG_I2C_BOARDINFO=y |
554 | CONFIG_I2C_CHARDEV=y | 611 | CONFIG_I2C_CHARDEV=y |
@@ -563,23 +620,43 @@ CONFIG_I2C_CHARDEV=y | |||
563 | # | 620 | # |
564 | # I2C Hardware Bus support | 621 | # I2C Hardware Bus support |
565 | # | 622 | # |
623 | # CONFIG_I2C_ALI1535 is not set | ||
624 | # CONFIG_I2C_ALI1563 is not set | ||
625 | # CONFIG_I2C_ALI15X3 is not set | ||
626 | # CONFIG_I2C_AMD756 is not set | ||
627 | # CONFIG_I2C_AMD8111 is not set | ||
628 | # CONFIG_I2C_I801 is not set | ||
629 | # CONFIG_I2C_I810 is not set | ||
630 | # CONFIG_I2C_PIIX4 is not set | ||
566 | CONFIG_I2C_MPC=y | 631 | CONFIG_I2C_MPC=y |
632 | # CONFIG_I2C_NFORCE2 is not set | ||
567 | # CONFIG_I2C_OCORES is not set | 633 | # CONFIG_I2C_OCORES is not set |
568 | # CONFIG_I2C_PARPORT_LIGHT is not set | 634 | # CONFIG_I2C_PARPORT_LIGHT is not set |
635 | # CONFIG_I2C_PROSAVAGE is not set | ||
636 | # CONFIG_I2C_SAVAGE4 is not set | ||
569 | # CONFIG_I2C_SIMTEC is not set | 637 | # CONFIG_I2C_SIMTEC is not set |
638 | # CONFIG_I2C_SIS5595 is not set | ||
639 | # CONFIG_I2C_SIS630 is not set | ||
640 | # CONFIG_I2C_SIS96X is not set | ||
641 | # CONFIG_I2C_TAOS_EVM is not set | ||
570 | # CONFIG_I2C_STUB is not set | 642 | # CONFIG_I2C_STUB is not set |
643 | # CONFIG_I2C_VIA is not set | ||
644 | # CONFIG_I2C_VIAPRO is not set | ||
645 | # CONFIG_I2C_VOODOO3 is not set | ||
571 | 646 | ||
572 | # | 647 | # |
573 | # Miscellaneous I2C Chip support | 648 | # Miscellaneous I2C Chip support |
574 | # | 649 | # |
575 | # CONFIG_SENSORS_DS1337 is not set | 650 | # CONFIG_SENSORS_DS1337 is not set |
576 | # CONFIG_SENSORS_DS1374 is not set | 651 | # CONFIG_SENSORS_DS1374 is not set |
652 | # CONFIG_DS1682 is not set | ||
577 | # CONFIG_SENSORS_EEPROM is not set | 653 | # CONFIG_SENSORS_EEPROM is not set |
578 | # CONFIG_SENSORS_PCF8574 is not set | 654 | # CONFIG_SENSORS_PCF8574 is not set |
579 | # CONFIG_SENSORS_PCA9539 is not set | 655 | # CONFIG_SENSORS_PCA9539 is not set |
580 | # CONFIG_SENSORS_PCF8591 is not set | 656 | # CONFIG_SENSORS_PCF8591 is not set |
581 | # CONFIG_SENSORS_M41T00 is not set | 657 | # CONFIG_SENSORS_M41T00 is not set |
582 | # CONFIG_SENSORS_MAX6875 is not set | 658 | # CONFIG_SENSORS_MAX6875 is not set |
659 | # CONFIG_SENSORS_TSL2550 is not set | ||
583 | # CONFIG_I2C_DEBUG_CORE is not set | 660 | # CONFIG_I2C_DEBUG_CORE is not set |
584 | # CONFIG_I2C_DEBUG_ALGO is not set | 661 | # CONFIG_I2C_DEBUG_ALGO is not set |
585 | # CONFIG_I2C_DEBUG_BUS is not set | 662 | # CONFIG_I2C_DEBUG_BUS is not set |
@@ -590,11 +667,8 @@ CONFIG_I2C_MPC=y | |||
590 | # | 667 | # |
591 | # CONFIG_SPI is not set | 668 | # CONFIG_SPI is not set |
592 | # CONFIG_SPI_MASTER is not set | 669 | # CONFIG_SPI_MASTER is not set |
593 | |||
594 | # | ||
595 | # Dallas's 1-wire bus | ||
596 | # | ||
597 | # CONFIG_W1 is not set | 670 | # CONFIG_W1 is not set |
671 | # CONFIG_POWER_SUPPLY is not set | ||
598 | CONFIG_HWMON=y | 672 | CONFIG_HWMON=y |
599 | # CONFIG_HWMON_VID is not set | 673 | # CONFIG_HWMON_VID is not set |
600 | # CONFIG_SENSORS_ABITUGURU is not set | 674 | # CONFIG_SENSORS_ABITUGURU is not set |
@@ -628,10 +702,13 @@ CONFIG_HWMON=y | |||
628 | # CONFIG_SENSORS_MAX6650 is not set | 702 | # CONFIG_SENSORS_MAX6650 is not set |
629 | # CONFIG_SENSORS_PC87360 is not set | 703 | # CONFIG_SENSORS_PC87360 is not set |
630 | # CONFIG_SENSORS_PC87427 is not set | 704 | # CONFIG_SENSORS_PC87427 is not set |
705 | # CONFIG_SENSORS_SIS5595 is not set | ||
631 | # CONFIG_SENSORS_SMSC47M1 is not set | 706 | # CONFIG_SENSORS_SMSC47M1 is not set |
632 | # CONFIG_SENSORS_SMSC47M192 is not set | 707 | # CONFIG_SENSORS_SMSC47M192 is not set |
633 | # CONFIG_SENSORS_SMSC47B397 is not set | 708 | # CONFIG_SENSORS_SMSC47B397 is not set |
709 | # CONFIG_SENSORS_VIA686A is not set | ||
634 | # CONFIG_SENSORS_VT1211 is not set | 710 | # CONFIG_SENSORS_VT1211 is not set |
711 | # CONFIG_SENSORS_VT8231 is not set | ||
635 | # CONFIG_SENSORS_W83781D is not set | 712 | # CONFIG_SENSORS_W83781D is not set |
636 | # CONFIG_SENSORS_W83791D is not set | 713 | # CONFIG_SENSORS_W83791D is not set |
637 | # CONFIG_SENSORS_W83792D is not set | 714 | # CONFIG_SENSORS_W83792D is not set |
@@ -670,19 +747,14 @@ CONFIG_DAB=y | |||
670 | # Sound | 747 | # Sound |
671 | # | 748 | # |
672 | # CONFIG_SOUND is not set | 749 | # CONFIG_SOUND is not set |
673 | 750 | CONFIG_HID_SUPPORT=y | |
674 | # | ||
675 | # HID Devices | ||
676 | # | ||
677 | CONFIG_HID=y | 751 | CONFIG_HID=y |
678 | # CONFIG_HID_DEBUG is not set | 752 | # CONFIG_HID_DEBUG is not set |
679 | 753 | CONFIG_USB_SUPPORT=y | |
680 | # | 754 | CONFIG_USB_ARCH_HAS_HCD=y |
681 | # USB support | 755 | CONFIG_USB_ARCH_HAS_OHCI=y |
682 | # | 756 | CONFIG_USB_ARCH_HAS_EHCI=y |
683 | # CONFIG_USB_ARCH_HAS_HCD is not set | 757 | # CONFIG_USB is not set |
684 | # CONFIG_USB_ARCH_HAS_OHCI is not set | ||
685 | # CONFIG_USB_ARCH_HAS_EHCI is not set | ||
686 | 758 | ||
687 | # | 759 | # |
688 | # NOTE: USB_STORAGE enables SCSI, and 'SCSI disk support' | 760 | # NOTE: USB_STORAGE enables SCSI, and 'SCSI disk support' |
@@ -706,14 +778,7 @@ CONFIG_HID=y | |||
706 | # | 778 | # |
707 | # LED Triggers | 779 | # LED Triggers |
708 | # | 780 | # |
709 | 781 | # CONFIG_INFINIBAND is not set | |
710 | # | ||
711 | # InfiniBand support | ||
712 | # | ||
713 | |||
714 | # | ||
715 | # EDAC - error detection and reporting (RAS) (EXPERIMENTAL) | ||
716 | # | ||
717 | 782 | ||
718 | # | 783 | # |
719 | # Real Time Clock | 784 | # Real Time Clock |
@@ -734,6 +799,11 @@ CONFIG_HID=y | |||
734 | # | 799 | # |
735 | 800 | ||
736 | # | 801 | # |
802 | # Userspace I/O | ||
803 | # | ||
804 | # CONFIG_UIO is not set | ||
805 | |||
806 | # | ||
737 | # File systems | 807 | # File systems |
738 | # | 808 | # |
739 | CONFIG_EXT2_FS=y | 809 | CONFIG_EXT2_FS=y |
@@ -829,7 +899,6 @@ CONFIG_RPCSEC_GSS_KRB5=y | |||
829 | # CONFIG_NCP_FS is not set | 899 | # CONFIG_NCP_FS is not set |
830 | # CONFIG_CODA_FS is not set | 900 | # CONFIG_CODA_FS is not set |
831 | # CONFIG_AFS_FS is not set | 901 | # CONFIG_AFS_FS is not set |
832 | # CONFIG_9P_FS is not set | ||
833 | 902 | ||
834 | # | 903 | # |
835 | # Partition Types | 904 | # Partition Types |
@@ -868,6 +937,7 @@ CONFIG_BITREVERSE=y | |||
868 | # CONFIG_CRC16 is not set | 937 | # CONFIG_CRC16 is not set |
869 | # CONFIG_CRC_ITU_T is not set | 938 | # CONFIG_CRC_ITU_T is not set |
870 | CONFIG_CRC32=y | 939 | CONFIG_CRC32=y |
940 | # CONFIG_CRC7 is not set | ||
871 | # CONFIG_LIBCRC32C is not set | 941 | # CONFIG_LIBCRC32C is not set |
872 | CONFIG_PLIST=y | 942 | CONFIG_PLIST=y |
873 | CONFIG_HAS_IOMEM=y | 943 | CONFIG_HAS_IOMEM=y |
@@ -892,6 +962,7 @@ CONFIG_ENABLE_MUST_CHECK=y | |||
892 | CONFIG_DEBUG_KERNEL=y | 962 | CONFIG_DEBUG_KERNEL=y |
893 | # CONFIG_DEBUG_SHIRQ is not set | 963 | # CONFIG_DEBUG_SHIRQ is not set |
894 | CONFIG_DETECT_SOFTLOCKUP=y | 964 | CONFIG_DETECT_SOFTLOCKUP=y |
965 | CONFIG_SCHED_DEBUG=y | ||
895 | # CONFIG_SCHEDSTATS is not set | 966 | # CONFIG_SCHEDSTATS is not set |
896 | # CONFIG_TIMER_STATS is not set | 967 | # CONFIG_TIMER_STATS is not set |
897 | # CONFIG_DEBUG_SLAB is not set | 968 | # CONFIG_DEBUG_SLAB is not set |
@@ -915,7 +986,7 @@ CONFIG_FORCED_INLINING=y | |||
915 | CONFIG_DEBUGGER=y | 986 | CONFIG_DEBUGGER=y |
916 | # CONFIG_XMON is not set | 987 | # CONFIG_XMON is not set |
917 | # CONFIG_BDI_SWITCH is not set | 988 | # CONFIG_BDI_SWITCH is not set |
918 | CONFIG_BOOTX_TEXT=y | 989 | # CONFIG_BOOTX_TEXT is not set |
919 | CONFIG_PPC_EARLY_DEBUG=y | 990 | CONFIG_PPC_EARLY_DEBUG=y |
920 | # CONFIG_PPC_EARLY_DEBUG_LPAR is not set | 991 | # CONFIG_PPC_EARLY_DEBUG_LPAR is not set |
921 | # CONFIG_PPC_EARLY_DEBUG_G5 is not set | 992 | # CONFIG_PPC_EARLY_DEBUG_G5 is not set |
@@ -932,10 +1003,6 @@ CONFIG_PPC_EARLY_DEBUG=y | |||
932 | # | 1003 | # |
933 | # CONFIG_KEYS is not set | 1004 | # CONFIG_KEYS is not set |
934 | # CONFIG_SECURITY is not set | 1005 | # CONFIG_SECURITY is not set |
935 | |||
936 | # | ||
937 | # Cryptographic options | ||
938 | # | ||
939 | CONFIG_CRYPTO=y | 1006 | CONFIG_CRYPTO=y |
940 | CONFIG_CRYPTO_ALGAPI=y | 1007 | CONFIG_CRYPTO_ALGAPI=y |
941 | CONFIG_CRYPTO_BLKCIPHER=y | 1008 | CONFIG_CRYPTO_BLKCIPHER=y |
@@ -973,7 +1040,4 @@ CONFIG_CRYPTO_DES=y | |||
973 | # CONFIG_CRYPTO_CRC32C is not set | 1040 | # CONFIG_CRYPTO_CRC32C is not set |
974 | # CONFIG_CRYPTO_CAMELLIA is not set | 1041 | # CONFIG_CRYPTO_CAMELLIA is not set |
975 | # CONFIG_CRYPTO_TEST is not set | 1042 | # CONFIG_CRYPTO_TEST is not set |
976 | 1043 | CONFIG_CRYPTO_HW=y | |
977 | # | ||
978 | # Hardware crypto devices | ||
979 | # | ||
diff --git a/arch/powerpc/kernel/head_64.S b/arch/powerpc/kernel/head_64.S index 8cdd48ea4391..1448af92c6a9 100644 --- a/arch/powerpc/kernel/head_64.S +++ b/arch/powerpc/kernel/head_64.S | |||
@@ -809,8 +809,9 @@ system_reset_iSeries: | |||
809 | mtmsrd r24 /* RI on */ | 809 | mtmsrd r24 /* RI on */ |
810 | lhz r24,PACAPACAINDEX(r13) /* Get processor # */ | 810 | lhz r24,PACAPACAINDEX(r13) /* Get processor # */ |
811 | cmpwi 0,r24,0 /* Are we processor 0? */ | 811 | cmpwi 0,r24,0 /* Are we processor 0? */ |
812 | beq .__start_initialization_iSeries /* Start up the first processor */ | 812 | bne 1f |
813 | mfspr r4,SPRN_CTRLF | 813 | b .__start_initialization_iSeries /* Start up the first processor */ |
814 | 1: mfspr r4,SPRN_CTRLF | ||
814 | li r5,CTRL_RUNLATCH /* Turn off the run light */ | 815 | li r5,CTRL_RUNLATCH /* Turn off the run light */ |
815 | andc r4,r4,r5 | 816 | andc r4,r4,r5 |
816 | mtspr SPRN_CTRLT,r4 | 817 | mtspr SPRN_CTRLT,r4 |
@@ -1611,7 +1612,7 @@ _GLOBAL(generic_secondary_smp_init) | |||
1611 | #endif | 1612 | #endif |
1612 | 1613 | ||
1613 | #ifdef CONFIG_PPC_ISERIES | 1614 | #ifdef CONFIG_PPC_ISERIES |
1614 | _STATIC(__start_initialization_iSeries) | 1615 | _INIT_STATIC(__start_initialization_iSeries) |
1615 | /* Clear out the BSS */ | 1616 | /* Clear out the BSS */ |
1616 | LOAD_REG_IMMEDIATE(r11,__bss_stop) | 1617 | LOAD_REG_IMMEDIATE(r11,__bss_stop) |
1617 | LOAD_REG_IMMEDIATE(r8,__bss_start) | 1618 | LOAD_REG_IMMEDIATE(r8,__bss_start) |
diff --git a/arch/powerpc/kernel/iomap.c b/arch/powerpc/kernel/iomap.c index 601ef79a5916..2a5cf8680370 100644 --- a/arch/powerpc/kernel/iomap.c +++ b/arch/powerpc/kernel/iomap.c | |||
@@ -7,6 +7,7 @@ | |||
7 | #include <linux/pci.h> | 7 | #include <linux/pci.h> |
8 | #include <linux/mm.h> | 8 | #include <linux/mm.h> |
9 | #include <asm/io.h> | 9 | #include <asm/io.h> |
10 | #include <asm/pci-bridge.h> | ||
10 | 11 | ||
11 | /* | 12 | /* |
12 | * Here comes the ppc64 implementation of the IOMAP | 13 | * Here comes the ppc64 implementation of the IOMAP |
@@ -136,7 +137,12 @@ void __iomem *pci_iomap(struct pci_dev *dev, int bar, unsigned long max) | |||
136 | 137 | ||
137 | void pci_iounmap(struct pci_dev *dev, void __iomem *addr) | 138 | void pci_iounmap(struct pci_dev *dev, void __iomem *addr) |
138 | { | 139 | { |
139 | /* Nothing to do */ | 140 | if (isa_vaddr_is_ioport(addr)) |
141 | return; | ||
142 | if (pcibios_vaddr_is_ioport(addr)) | ||
143 | return; | ||
144 | iounmap(addr); | ||
140 | } | 145 | } |
146 | |||
141 | EXPORT_SYMBOL(pci_iomap); | 147 | EXPORT_SYMBOL(pci_iomap); |
142 | EXPORT_SYMBOL(pci_iounmap); | 148 | EXPORT_SYMBOL(pci_iounmap); |
diff --git a/arch/powerpc/kernel/irq.c b/arch/powerpc/kernel/irq.c index 2fc87862146c..24bea97c736c 100644 --- a/arch/powerpc/kernel/irq.c +++ b/arch/powerpc/kernel/irq.c | |||
@@ -418,10 +418,10 @@ irq_hw_number_t virq_to_hw(unsigned int virq) | |||
418 | } | 418 | } |
419 | EXPORT_SYMBOL_GPL(virq_to_hw); | 419 | EXPORT_SYMBOL_GPL(virq_to_hw); |
420 | 420 | ||
421 | struct irq_host *irq_alloc_host(unsigned int revmap_type, | 421 | __init_refok struct irq_host *irq_alloc_host(unsigned int revmap_type, |
422 | unsigned int revmap_arg, | 422 | unsigned int revmap_arg, |
423 | struct irq_host_ops *ops, | 423 | struct irq_host_ops *ops, |
424 | irq_hw_number_t inval_irq) | 424 | irq_hw_number_t inval_irq) |
425 | { | 425 | { |
426 | struct irq_host *host; | 426 | struct irq_host *host; |
427 | unsigned int size = sizeof(struct irq_host); | 427 | unsigned int size = sizeof(struct irq_host); |
diff --git a/arch/powerpc/kernel/pci-common.c b/arch/powerpc/kernel/pci-common.c index fe7d1255e11e..083cfbdbe0b2 100644 --- a/arch/powerpc/kernel/pci-common.c +++ b/arch/powerpc/kernel/pci-common.c | |||
@@ -65,7 +65,7 @@ static void __devinit pci_setup_pci_controller(struct pci_controller *hose) | |||
65 | spin_unlock(&hose_spinlock); | 65 | spin_unlock(&hose_spinlock); |
66 | } | 66 | } |
67 | 67 | ||
68 | struct pci_controller * pcibios_alloc_controller(struct device_node *dev) | 68 | __init_refok struct pci_controller * pcibios_alloc_controller(struct device_node *dev) |
69 | { | 69 | { |
70 | struct pci_controller *phb; | 70 | struct pci_controller *phb; |
71 | 71 | ||
@@ -101,6 +101,29 @@ void pcibios_free_controller(struct pci_controller *phb) | |||
101 | kfree(phb); | 101 | kfree(phb); |
102 | } | 102 | } |
103 | 103 | ||
104 | int pcibios_vaddr_is_ioport(void __iomem *address) | ||
105 | { | ||
106 | int ret = 0; | ||
107 | struct pci_controller *hose; | ||
108 | unsigned long size; | ||
109 | |||
110 | spin_lock(&hose_spinlock); | ||
111 | list_for_each_entry(hose, &hose_list, list_node) { | ||
112 | #ifdef CONFIG_PPC64 | ||
113 | size = hose->pci_io_size; | ||
114 | #else | ||
115 | size = hose->io_resource.end - hose->io_resource.start + 1; | ||
116 | #endif | ||
117 | if (address >= hose->io_base_virt && | ||
118 | address < (hose->io_base_virt + size)) { | ||
119 | ret = 1; | ||
120 | break; | ||
121 | } | ||
122 | } | ||
123 | spin_unlock(&hose_spinlock); | ||
124 | return ret; | ||
125 | } | ||
126 | |||
104 | /* | 127 | /* |
105 | * Return the domain number for this bus. | 128 | * Return the domain number for this bus. |
106 | */ | 129 | */ |
diff --git a/arch/powerpc/kernel/pci_32.c b/arch/powerpc/kernel/pci_32.c index 0adf077f3f3a..cd35c969bb28 100644 --- a/arch/powerpc/kernel/pci_32.c +++ b/arch/powerpc/kernel/pci_32.c | |||
@@ -60,6 +60,24 @@ LIST_HEAD(hose_list); | |||
60 | static int pci_bus_count; | 60 | static int pci_bus_count; |
61 | 61 | ||
62 | static void | 62 | static void |
63 | fixup_hide_host_resource_fsl(struct pci_dev* dev) | ||
64 | { | ||
65 | int i, class = dev->class >> 8; | ||
66 | |||
67 | if ((class == PCI_CLASS_PROCESSOR_POWERPC) && | ||
68 | (dev->hdr_type == PCI_HEADER_TYPE_NORMAL) && | ||
69 | (dev->bus->parent == NULL)) { | ||
70 | for (i = 0; i < DEVICE_COUNT_RESOURCE; i++) { | ||
71 | dev->resource[i].start = 0; | ||
72 | dev->resource[i].end = 0; | ||
73 | dev->resource[i].flags = 0; | ||
74 | } | ||
75 | } | ||
76 | } | ||
77 | DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_MOTOROLA, PCI_ANY_ID, fixup_hide_host_resource_fsl); | ||
78 | DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_FREESCALE, PCI_ANY_ID, fixup_hide_host_resource_fsl); | ||
79 | |||
80 | static void | ||
63 | fixup_broken_pcnet32(struct pci_dev* dev) | 81 | fixup_broken_pcnet32(struct pci_dev* dev) |
64 | { | 82 | { |
65 | if ((dev->class>>8 == PCI_CLASS_NETWORK_ETHERNET)) { | 83 | if ((dev->class>>8 == PCI_CLASS_NETWORK_ETHERNET)) { |
@@ -415,15 +433,13 @@ probe_resource(struct pci_bus *parent, struct resource *pr, | |||
415 | return 0; | 433 | return 0; |
416 | } | 434 | } |
417 | 435 | ||
418 | static void __init | 436 | void __init |
419 | update_bridge_base(struct pci_bus *bus, int i) | 437 | update_bridge_resource(struct pci_dev *dev, struct resource *res) |
420 | { | 438 | { |
421 | struct resource *res = bus->resource[i]; | ||
422 | u8 io_base_lo, io_limit_lo; | 439 | u8 io_base_lo, io_limit_lo; |
423 | u16 mem_base, mem_limit; | 440 | u16 mem_base, mem_limit; |
424 | u16 cmd; | 441 | u16 cmd; |
425 | unsigned long start, end, off; | 442 | unsigned long start, end, off; |
426 | struct pci_dev *dev = bus->self; | ||
427 | struct pci_controller *hose = dev->sysdata; | 443 | struct pci_controller *hose = dev->sysdata; |
428 | 444 | ||
429 | if (!hose) { | 445 | if (!hose) { |
@@ -467,12 +483,20 @@ update_bridge_base(struct pci_bus *bus, int i) | |||
467 | pci_write_config_word(dev, PCI_PREF_MEMORY_LIMIT, mem_limit); | 483 | pci_write_config_word(dev, PCI_PREF_MEMORY_LIMIT, mem_limit); |
468 | 484 | ||
469 | } else { | 485 | } else { |
470 | DBG(KERN_ERR "PCI: ugh, bridge %s res %d has flags=%lx\n", | 486 | DBG(KERN_ERR "PCI: ugh, bridge %s res has flags=%lx\n", |
471 | pci_name(dev), i, res->flags); | 487 | pci_name(dev), res->flags); |
472 | } | 488 | } |
473 | pci_write_config_word(dev, PCI_COMMAND, cmd); | 489 | pci_write_config_word(dev, PCI_COMMAND, cmd); |
474 | } | 490 | } |
475 | 491 | ||
492 | static void __init | ||
493 | update_bridge_base(struct pci_bus *bus, int i) | ||
494 | { | ||
495 | struct resource *res = bus->resource[i]; | ||
496 | struct pci_dev *dev = bus->self; | ||
497 | update_bridge_resource(dev, res); | ||
498 | } | ||
499 | |||
476 | static inline void alloc_resource(struct pci_dev *dev, int idx) | 500 | static inline void alloc_resource(struct pci_dev *dev, int idx) |
477 | { | 501 | { |
478 | struct resource *pr, *r = &dev->resource[idx]; | 502 | struct resource *pr, *r = &dev->resource[idx]; |
@@ -1223,7 +1247,7 @@ pcibios_init(void) | |||
1223 | 1247 | ||
1224 | subsys_initcall(pcibios_init); | 1248 | subsys_initcall(pcibios_init); |
1225 | 1249 | ||
1226 | void __init pcibios_fixup_bus(struct pci_bus *bus) | 1250 | void pcibios_fixup_bus(struct pci_bus *bus) |
1227 | { | 1251 | { |
1228 | struct pci_controller *hose = (struct pci_controller *) bus->sysdata; | 1252 | struct pci_controller *hose = (struct pci_controller *) bus->sysdata; |
1229 | unsigned long io_offset; | 1253 | unsigned long io_offset; |
@@ -1468,3 +1492,10 @@ EARLY_PCI_OP(read, dword, u32 *) | |||
1468 | EARLY_PCI_OP(write, byte, u8) | 1492 | EARLY_PCI_OP(write, byte, u8) |
1469 | EARLY_PCI_OP(write, word, u16) | 1493 | EARLY_PCI_OP(write, word, u16) |
1470 | EARLY_PCI_OP(write, dword, u32) | 1494 | EARLY_PCI_OP(write, dword, u32) |
1495 | |||
1496 | extern int pci_bus_find_capability (struct pci_bus *bus, unsigned int devfn, int cap); | ||
1497 | int early_find_capability(struct pci_controller *hose, int bus, int devfn, | ||
1498 | int cap) | ||
1499 | { | ||
1500 | return pci_bus_find_capability(fake_pci_bus(hose, bus), devfn, cap); | ||
1501 | } | ||
diff --git a/arch/powerpc/kernel/process.c b/arch/powerpc/kernel/process.c index 84f000a45e36..a83727b308a0 100644 --- a/arch/powerpc/kernel/process.c +++ b/arch/powerpc/kernel/process.c | |||
@@ -423,7 +423,11 @@ void show_regs(struct pt_regs * regs) | |||
423 | printk(" CR: %08lx XER: %08lx\n", regs->ccr, regs->xer); | 423 | printk(" CR: %08lx XER: %08lx\n", regs->ccr, regs->xer); |
424 | trap = TRAP(regs); | 424 | trap = TRAP(regs); |
425 | if (trap == 0x300 || trap == 0x600) | 425 | if (trap == 0x300 || trap == 0x600) |
426 | #if defined(CONFIG_4xx) || defined(CONFIG_BOOKE) | ||
427 | printk("DEAR: "REG", ESR: "REG"\n", regs->dar, regs->dsisr); | ||
428 | #else | ||
426 | printk("DAR: "REG", DSISR: "REG"\n", regs->dar, regs->dsisr); | 429 | printk("DAR: "REG", DSISR: "REG"\n", regs->dar, regs->dsisr); |
430 | #endif | ||
427 | printk("TASK = %p[%d] '%s' THREAD: %p", | 431 | printk("TASK = %p[%d] '%s' THREAD: %p", |
428 | current, current->pid, current->comm, task_thread_info(current)); | 432 | current, current->pid, current->comm, task_thread_info(current)); |
429 | 433 | ||
diff --git a/arch/powerpc/kernel/prom_parse.c b/arch/powerpc/kernel/prom_parse.c index 3786dcc8a7b6..b5c96af955c6 100644 --- a/arch/powerpc/kernel/prom_parse.c +++ b/arch/powerpc/kernel/prom_parse.c | |||
@@ -24,7 +24,7 @@ | |||
24 | /* Max address size we deal with */ | 24 | /* Max address size we deal with */ |
25 | #define OF_MAX_ADDR_CELLS 4 | 25 | #define OF_MAX_ADDR_CELLS 4 |
26 | #define OF_CHECK_COUNTS(na, ns) ((na) > 0 && (na) <= OF_MAX_ADDR_CELLS && \ | 26 | #define OF_CHECK_COUNTS(na, ns) ((na) > 0 && (na) <= OF_MAX_ADDR_CELLS && \ |
27 | (ns) >= 0) | 27 | (ns) > 0) |
28 | 28 | ||
29 | static struct of_bus *of_match_bus(struct device_node *np); | 29 | static struct of_bus *of_match_bus(struct device_node *np); |
30 | static int __of_address_to_resource(struct device_node *dev, | 30 | static int __of_address_to_resource(struct device_node *dev, |
diff --git a/arch/powerpc/kernel/setup-common.c b/arch/powerpc/kernel/setup-common.c index 4924c48cb1ff..50ef38cffdbf 100644 --- a/arch/powerpc/kernel/setup-common.c +++ b/arch/powerpc/kernel/setup-common.c | |||
@@ -496,6 +496,10 @@ int check_legacy_ioport(unsigned long base_port) | |||
496 | break; | 496 | break; |
497 | } | 497 | } |
498 | np = of_find_node_by_type(NULL, "8042"); | 498 | np = of_find_node_by_type(NULL, "8042"); |
499 | /* Pegasos has no device_type on its 8042 node, look for the | ||
500 | * name instead */ | ||
501 | if (!np) | ||
502 | np = of_find_node_by_name(NULL, "8042"); | ||
499 | break; | 503 | break; |
500 | case FDC_BASE: /* FDC1 */ | 504 | case FDC_BASE: /* FDC1 */ |
501 | np = of_find_node_by_type(NULL, "fdc"); | 505 | np = of_find_node_by_type(NULL, "fdc"); |
diff --git a/arch/powerpc/kernel/udbg.c b/arch/powerpc/kernel/udbg.c index cbca1df8bc60..0f9b4eadfbcb 100644 --- a/arch/powerpc/kernel/udbg.c +++ b/arch/powerpc/kernel/udbg.c | |||
@@ -155,7 +155,7 @@ static int early_console_initialized; | |||
155 | * Called by setup_system after ppc_md->probe and ppc_md->early_init. | 155 | * Called by setup_system after ppc_md->probe and ppc_md->early_init. |
156 | * Call it again after setting udbg_putc in ppc_md->setup_arch. | 156 | * Call it again after setting udbg_putc in ppc_md->setup_arch. |
157 | */ | 157 | */ |
158 | void register_early_udbg_console(void) | 158 | void __init register_early_udbg_console(void) |
159 | { | 159 | { |
160 | if (early_console_initialized) | 160 | if (early_console_initialized) |
161 | return; | 161 | return; |
diff --git a/arch/powerpc/mm/lmb.c b/arch/powerpc/mm/lmb.c index e3a1e8dc536a..8f4d2dc4cafb 100644 --- a/arch/powerpc/mm/lmb.c +++ b/arch/powerpc/mm/lmb.c | |||
@@ -138,8 +138,8 @@ void __init lmb_analyze(void) | |||
138 | static long __init lmb_add_region(struct lmb_region *rgn, unsigned long base, | 138 | static long __init lmb_add_region(struct lmb_region *rgn, unsigned long base, |
139 | unsigned long size) | 139 | unsigned long size) |
140 | { | 140 | { |
141 | unsigned long i, coalesced = 0; | 141 | unsigned long coalesced = 0; |
142 | long adjacent; | 142 | long adjacent, i; |
143 | 143 | ||
144 | /* First try and coalesce this LMB with another. */ | 144 | /* First try and coalesce this LMB with another. */ |
145 | for (i=0; i < rgn->cnt; i++) { | 145 | for (i=0; i < rgn->cnt; i++) { |
diff --git a/arch/powerpc/platforms/82xx/mpc82xx_ads.c b/arch/powerpc/platforms/82xx/mpc82xx_ads.c index da20832b27f1..2d1b05b9f8ef 100644 --- a/arch/powerpc/platforms/82xx/mpc82xx_ads.c +++ b/arch/powerpc/platforms/82xx/mpc82xx_ads.c | |||
@@ -553,7 +553,8 @@ static void __init mpc82xx_add_bridge(struct device_node *np) | |||
553 | 553 | ||
554 | setup_indirect_pci(hose, | 554 | setup_indirect_pci(hose, |
555 | r.start + offsetof(pci_cpm2_t, pci_cfg_addr), | 555 | r.start + offsetof(pci_cpm2_t, pci_cfg_addr), |
556 | r.start + offsetof(pci_cpm2_t, pci_cfg_data)); | 556 | r.start + offsetof(pci_cpm2_t, pci_cfg_data), |
557 | 0); | ||
557 | 558 | ||
558 | pci_process_bridge_OF_ranges(hose, np, 1); | 559 | pci_process_bridge_OF_ranges(hose, np, 1); |
559 | } | 560 | } |
diff --git a/arch/powerpc/platforms/83xx/pci.c b/arch/powerpc/platforms/83xx/pci.c index c0e2b89154e5..92069469de20 100644 --- a/arch/powerpc/platforms/83xx/pci.c +++ b/arch/powerpc/platforms/83xx/pci.c | |||
@@ -74,11 +74,11 @@ int __init mpc83xx_add_bridge(struct device_node *dev) | |||
74 | */ | 74 | */ |
75 | /* PCI 1 */ | 75 | /* PCI 1 */ |
76 | if ((rsrc.start & 0xfffff) == 0x8500) { | 76 | if ((rsrc.start & 0xfffff) == 0x8500) { |
77 | setup_indirect_pci(hose, immr + 0x8300, immr + 0x8304); | 77 | setup_indirect_pci(hose, immr + 0x8300, immr + 0x8304, 0); |
78 | } | 78 | } |
79 | /* PCI 2 */ | 79 | /* PCI 2 */ |
80 | if ((rsrc.start & 0xfffff) == 0x8600) { | 80 | if ((rsrc.start & 0xfffff) == 0x8600) { |
81 | setup_indirect_pci(hose, immr + 0x8380, immr + 0x8384); | 81 | setup_indirect_pci(hose, immr + 0x8380, immr + 0x8384, 0); |
82 | primary = 0; | 82 | primary = 0; |
83 | } | 83 | } |
84 | 84 | ||
diff --git a/arch/powerpc/platforms/85xx/Kconfig b/arch/powerpc/platforms/85xx/Kconfig index 629926e01e90..f58184086c8c 100644 --- a/arch/powerpc/platforms/85xx/Kconfig +++ b/arch/powerpc/platforms/85xx/Kconfig | |||
@@ -18,6 +18,7 @@ config MPC8560_ADS | |||
18 | config MPC85xx_CDS | 18 | config MPC85xx_CDS |
19 | bool "Freescale MPC85xx CDS" | 19 | bool "Freescale MPC85xx CDS" |
20 | select DEFAULT_UIMAGE | 20 | select DEFAULT_UIMAGE |
21 | select PPC_I8259 | ||
21 | help | 22 | help |
22 | This option enables support for the MPC85xx CDS board | 23 | This option enables support for the MPC85xx CDS board |
23 | 24 | ||
@@ -30,6 +31,7 @@ config MPC85xx_MDS | |||
30 | 31 | ||
31 | config MPC8544_DS | 32 | config MPC8544_DS |
32 | bool "Freescale MPC8544 DS" | 33 | bool "Freescale MPC8544 DS" |
34 | select PPC_I8259 | ||
33 | select DEFAULT_UIMAGE | 35 | select DEFAULT_UIMAGE |
34 | help | 36 | help |
35 | This option enables support for the MPC8544 DS board | 37 | This option enables support for the MPC8544 DS board |
@@ -50,9 +52,9 @@ config MPC8560 | |||
50 | config MPC85xx | 52 | config MPC85xx |
51 | bool | 53 | bool |
52 | select PPC_UDBG_16550 | 54 | select PPC_UDBG_16550 |
53 | select PPC_INDIRECT_PCI | 55 | select PPC_INDIRECT_PCI if PCI |
54 | select PPC_INDIRECT_PCI_BE | ||
55 | select MPIC | 56 | select MPIC |
57 | select FSL_PCI if PCI | ||
56 | select SERIAL_8250_SHARE_IRQ if SERIAL_8250 | 58 | select SERIAL_8250_SHARE_IRQ if SERIAL_8250 |
57 | default y if MPC8540_ADS || MPC85xx_CDS || MPC8560_ADS \ | 59 | default y if MPC8540_ADS || MPC85xx_CDS || MPC8560_ADS \ |
58 | || MPC85xx_MDS || MPC8544_DS | 60 | || MPC85xx_MDS || MPC8544_DS |
diff --git a/arch/powerpc/platforms/85xx/Makefile b/arch/powerpc/platforms/85xx/Makefile index 4e02cbb14cf7..d70f2d0f9d36 100644 --- a/arch/powerpc/platforms/85xx/Makefile +++ b/arch/powerpc/platforms/85xx/Makefile | |||
@@ -1,7 +1,7 @@ | |||
1 | # | 1 | # |
2 | # Makefile for the PowerPC 85xx linux kernel. | 2 | # Makefile for the PowerPC 85xx linux kernel. |
3 | # | 3 | # |
4 | obj-$(CONFIG_PPC_85xx) += misc.o pci.o | 4 | obj-$(CONFIG_PPC_85xx) += misc.o |
5 | obj-$(CONFIG_MPC8540_ADS) += mpc85xx_ads.o | 5 | obj-$(CONFIG_MPC8540_ADS) += mpc85xx_ads.o |
6 | obj-$(CONFIG_MPC8560_ADS) += mpc85xx_ads.o | 6 | obj-$(CONFIG_MPC8560_ADS) += mpc85xx_ads.o |
7 | obj-$(CONFIG_MPC85xx_CDS) += mpc85xx_cds.o | 7 | obj-$(CONFIG_MPC85xx_CDS) += mpc85xx_cds.o |
diff --git a/arch/powerpc/platforms/85xx/mpc8544_ds.c b/arch/powerpc/platforms/85xx/mpc8544_ds.c index 6fb90aab879f..4905f6f8903b 100644 --- a/arch/powerpc/platforms/85xx/mpc8544_ds.c +++ b/arch/powerpc/platforms/85xx/mpc8544_ds.c | |||
@@ -2,6 +2,8 @@ | |||
2 | * MPC8544 DS Board Setup | 2 | * MPC8544 DS Board Setup |
3 | * | 3 | * |
4 | * Author Xianghua Xiao (x.xiao@freescale.com) | 4 | * Author Xianghua Xiao (x.xiao@freescale.com) |
5 | * Roy Zang <tie-fei.zang@freescale.com> | ||
6 | * - Add PCI/PCI Exprees support | ||
5 | * Copyright 2007 Freescale Semiconductor Inc. | 7 | * Copyright 2007 Freescale Semiconductor Inc. |
6 | * | 8 | * |
7 | * This program is free software; you can redistribute it and/or modify it | 9 | * This program is free software; you can redistribute it and/or modify it |
@@ -12,13 +14,16 @@ | |||
12 | 14 | ||
13 | #include <linux/stddef.h> | 15 | #include <linux/stddef.h> |
14 | #include <linux/kernel.h> | 16 | #include <linux/kernel.h> |
17 | #include <linux/pci.h> | ||
15 | #include <linux/kdev_t.h> | 18 | #include <linux/kdev_t.h> |
16 | #include <linux/delay.h> | 19 | #include <linux/delay.h> |
17 | #include <linux/seq_file.h> | 20 | #include <linux/seq_file.h> |
21 | #include <linux/interrupt.h> | ||
18 | 22 | ||
19 | #include <asm/system.h> | 23 | #include <asm/system.h> |
20 | #include <asm/time.h> | 24 | #include <asm/time.h> |
21 | #include <asm/machdep.h> | 25 | #include <asm/machdep.h> |
26 | #include <asm/pci-bridge.h> | ||
22 | #include <asm/mpc85xx.h> | 27 | #include <asm/mpc85xx.h> |
23 | #include <mm/mmu_decl.h> | 28 | #include <mm/mmu_decl.h> |
24 | #include <asm/prom.h> | 29 | #include <asm/prom.h> |
@@ -27,6 +32,7 @@ | |||
27 | #include <asm/i8259.h> | 32 | #include <asm/i8259.h> |
28 | 33 | ||
29 | #include <sysdev/fsl_soc.h> | 34 | #include <sysdev/fsl_soc.h> |
35 | #include <sysdev/fsl_pci.h> | ||
30 | #include "mpc85xx.h" | 36 | #include "mpc85xx.h" |
31 | 37 | ||
32 | #undef DEBUG | 38 | #undef DEBUG |
@@ -37,6 +43,17 @@ | |||
37 | #define DBG(fmt, args...) | 43 | #define DBG(fmt, args...) |
38 | #endif | 44 | #endif |
39 | 45 | ||
46 | #ifdef CONFIG_PPC_I8259 | ||
47 | static void mpc8544_8259_cascade(unsigned int irq, struct irq_desc *desc) | ||
48 | { | ||
49 | unsigned int cascade_irq = i8259_irq(); | ||
50 | |||
51 | if (cascade_irq != NO_IRQ) { | ||
52 | generic_handle_irq(cascade_irq); | ||
53 | } | ||
54 | desc->chip->eoi(irq); | ||
55 | } | ||
56 | #endif /* CONFIG_PPC_I8259 */ | ||
40 | 57 | ||
41 | void __init mpc8544_ds_pic_init(void) | 58 | void __init mpc8544_ds_pic_init(void) |
42 | { | 59 | { |
@@ -96,19 +113,240 @@ void __init mpc8544_ds_pic_init(void) | |||
96 | #endif /* CONFIG_PPC_I8259 */ | 113 | #endif /* CONFIG_PPC_I8259 */ |
97 | } | 114 | } |
98 | 115 | ||
116 | #ifdef CONFIG_PCI | ||
117 | enum pirq { PIRQA = 8, PIRQB, PIRQC, PIRQD, PIRQE, PIRQF, PIRQG, PIRQH }; | ||
118 | |||
119 | /* | ||
120 | * Value in table -- IRQ number | ||
121 | */ | ||
122 | const unsigned char uli1575_irq_route_table[16] = { | ||
123 | 0, /* 0: Reserved */ | ||
124 | 0x8, | ||
125 | 0, /* 2: Reserved */ | ||
126 | 0x2, | ||
127 | 0x4, | ||
128 | 0x5, | ||
129 | 0x7, | ||
130 | 0x6, | ||
131 | 0, /* 8: Reserved */ | ||
132 | 0x1, | ||
133 | 0x3, | ||
134 | 0x9, | ||
135 | 0xb, | ||
136 | 0, /* 13: Reserved */ | ||
137 | 0xd, | ||
138 | 0xf, | ||
139 | }; | ||
140 | |||
141 | static int __devinit | ||
142 | get_pci_irq_from_of(struct pci_controller *hose, int slot, int pin) | ||
143 | { | ||
144 | struct of_irq oirq; | ||
145 | u32 laddr[3]; | ||
146 | struct device_node *hosenode = hose ? hose->arch_data : NULL; | ||
147 | |||
148 | if (!hosenode) | ||
149 | return -EINVAL; | ||
150 | |||
151 | laddr[0] = (hose->first_busno << 16) | (PCI_DEVFN(slot, 0) << 8); | ||
152 | laddr[1] = laddr[2] = 0; | ||
153 | of_irq_map_raw(hosenode, &pin, 1, laddr, &oirq); | ||
154 | DBG("mpc8544_ds: pci irq addr %x, slot %d, pin %d, irq %d\n", | ||
155 | laddr[0], slot, pin, oirq.specifier[0]); | ||
156 | return oirq.specifier[0]; | ||
157 | } | ||
158 | |||
159 | /*8259*/ | ||
160 | static void __devinit quirk_uli1575(struct pci_dev *dev) | ||
161 | { | ||
162 | unsigned short temp; | ||
163 | struct pci_controller *hose = pci_bus_to_host(dev->bus); | ||
164 | unsigned char irq2pin[16]; | ||
165 | unsigned long pirq_map_word = 0; | ||
166 | u32 irq; | ||
167 | int i; | ||
168 | |||
169 | /* | ||
170 | * ULI1575 interrupts route setup | ||
171 | */ | ||
172 | memset(irq2pin, 0, 16); /* Initialize default value 0 */ | ||
173 | |||
174 | irq2pin[6]=PIRQA+3; /* enabled mapping for IRQ6 to PIRQD, used by SATA */ | ||
175 | |||
176 | /* | ||
177 | * PIRQE -> PIRQF mapping set manually | ||
178 | * | ||
179 | * IRQ pin IRQ# | ||
180 | * PIRQE ---- 9 | ||
181 | * PIRQF ---- 10 | ||
182 | * PIRQG ---- 11 | ||
183 | * PIRQH ---- 12 | ||
184 | */ | ||
185 | for (i = 0; i < 4; i++) | ||
186 | irq2pin[i + 9] = PIRQE + i; | ||
187 | |||
188 | /* Set IRQ-PIRQ Mapping to ULI1575 */ | ||
189 | for (i = 0; i < 16; i++) | ||
190 | if (irq2pin[i]) | ||
191 | pirq_map_word |= (uli1575_irq_route_table[i] & 0xf) | ||
192 | << ((irq2pin[i] - PIRQA) * 4); | ||
193 | |||
194 | pirq_map_word |= 1<<26; /* disable INTx in EP mode*/ | ||
195 | |||
196 | /* ULI1575 IRQ mapping conf register default value is 0xb9317542 */ | ||
197 | DBG("Setup ULI1575 IRQ mapping configuration register value = 0x%x\n", | ||
198 | (int)pirq_map_word); | ||
199 | pci_write_config_dword(dev, 0x48, pirq_map_word); | ||
200 | |||
201 | #define ULI1575_SET_DEV_IRQ(slot, pin, reg) \ | ||
202 | do { \ | ||
203 | int irq; \ | ||
204 | irq = get_pci_irq_from_of(hose, slot, pin); \ | ||
205 | if (irq > 0 && irq < 16) \ | ||
206 | pci_write_config_byte(dev, reg, irq2pin[irq]); \ | ||
207 | else \ | ||
208 | printk(KERN_WARNING "ULI1575 device" \ | ||
209 | "(slot %d, pin %d) irq %d is invalid.\n", \ | ||
210 | slot, pin, irq); \ | ||
211 | } while(0) | ||
212 | |||
213 | /* USB 1.1 OHCI controller 1, slot 28, pin 1 */ | ||
214 | ULI1575_SET_DEV_IRQ(28, 1, 0x86); | ||
215 | |||
216 | /* USB 1.1 OHCI controller 2, slot 28, pin 2 */ | ||
217 | ULI1575_SET_DEV_IRQ(28, 2, 0x87); | ||
218 | |||
219 | /* USB 1.1 OHCI controller 3, slot 28, pin 3 */ | ||
220 | ULI1575_SET_DEV_IRQ(28, 3, 0x88); | ||
221 | |||
222 | /* USB 2.0 controller, slot 28, pin 4 */ | ||
223 | irq = get_pci_irq_from_of(hose, 28, 4); | ||
224 | if (irq >= 0 && irq <= 15) | ||
225 | pci_write_config_dword(dev, 0x74, uli1575_irq_route_table[irq]); | ||
226 | |||
227 | /* Audio controller, slot 29, pin 1 */ | ||
228 | ULI1575_SET_DEV_IRQ(29, 1, 0x8a); | ||
229 | |||
230 | /* Modem controller, slot 29, pin 2 */ | ||
231 | ULI1575_SET_DEV_IRQ(29, 2, 0x8b); | ||
232 | |||
233 | /* HD audio controller, slot 29, pin 3 */ | ||
234 | ULI1575_SET_DEV_IRQ(29, 3, 0x8c); | ||
235 | |||
236 | /* SMB interrupt: slot 30, pin 1 */ | ||
237 | ULI1575_SET_DEV_IRQ(30, 1, 0x8e); | ||
238 | |||
239 | /* PMU ACPI SCI interrupt: slot 30, pin 2 */ | ||
240 | ULI1575_SET_DEV_IRQ(30, 2, 0x8f); | ||
241 | |||
242 | /* Serial ATA interrupt: slot 31, pin 1 */ | ||
243 | ULI1575_SET_DEV_IRQ(31, 1, 0x8d); | ||
244 | |||
245 | /* Primary PATA IDE IRQ: 14 | ||
246 | * Secondary PATA IDE IRQ: 15 | ||
247 | */ | ||
248 | pci_write_config_byte(dev, 0x44, 0x30 | uli1575_irq_route_table[14]); | ||
249 | pci_write_config_byte(dev, 0x75, uli1575_irq_route_table[15]); | ||
250 | |||
251 | /* Set IRQ14 and IRQ15 to legacy IRQs */ | ||
252 | pci_read_config_word(dev, 0x46, &temp); | ||
253 | temp |= 0xc000; | ||
254 | pci_write_config_word(dev, 0x46, temp); | ||
255 | |||
256 | /* Set i8259 interrupt trigger | ||
257 | * IRQ 3: Level | ||
258 | * IRQ 4: Level | ||
259 | * IRQ 5: Level | ||
260 | * IRQ 6: Level | ||
261 | * IRQ 7: Level | ||
262 | * IRQ 9: Level | ||
263 | * IRQ 10: Level | ||
264 | * IRQ 11: Level | ||
265 | * IRQ 12: Level | ||
266 | * IRQ 14: Edge | ||
267 | * IRQ 15: Edge | ||
268 | */ | ||
269 | outb(0xfa, 0x4d0); | ||
270 | outb(0x1e, 0x4d1); | ||
271 | |||
272 | #undef ULI1575_SET_DEV_IRQ | ||
273 | } | ||
274 | |||
275 | /* SATA */ | ||
276 | static void __devinit quirk_uli5288(struct pci_dev *dev) | ||
277 | { | ||
278 | unsigned char c; | ||
279 | |||
280 | pci_read_config_byte(dev, 0x83, &c); | ||
281 | c |= 0x80; /* read/write lock */ | ||
282 | pci_write_config_byte(dev, 0x83, c); | ||
283 | |||
284 | pci_write_config_byte(dev, 0x09, 0x01); /* Base class code: storage */ | ||
285 | pci_write_config_byte(dev, 0x0a, 0x06); /* IDE disk */ | ||
286 | |||
287 | pci_read_config_byte(dev, 0x83, &c); | ||
288 | c &= 0x7f; | ||
289 | pci_write_config_byte(dev, 0x83, c); | ||
290 | |||
291 | pci_read_config_byte(dev, 0x84, &c); | ||
292 | c |= 0x01; /* emulated PATA mode enabled */ | ||
293 | pci_write_config_byte(dev, 0x84, c); | ||
294 | } | ||
295 | |||
296 | /* PATA */ | ||
297 | static void __devinit quirk_uli5229(struct pci_dev *dev) | ||
298 | { | ||
299 | unsigned short temp; | ||
300 | pci_write_config_word(dev, 0x04, 0x0405); /* MEM IO MSI */ | ||
301 | pci_read_config_word(dev, 0x4a, &temp); | ||
302 | temp |= 0x1000; /* Enable Native IRQ 14/15 */ | ||
303 | pci_write_config_word(dev, 0x4a, temp); | ||
304 | } | ||
305 | |||
306 | /*Bridge*/ | ||
307 | static void __devinit early_uli5249(struct pci_dev *dev) | ||
308 | { | ||
309 | unsigned char temp; | ||
310 | pci_write_config_word(dev, 0x04, 0x0007); /* mem access */ | ||
311 | pci_read_config_byte(dev, 0x7c, &temp); | ||
312 | pci_write_config_byte(dev, 0x7c, 0x80); /* R/W lock control */ | ||
313 | pci_write_config_byte(dev, 0x09, 0x01); /* set as pci-pci bridge */ | ||
314 | pci_write_config_byte(dev, 0x7c, temp); /* restore pci bus debug control */ | ||
315 | dev->class |= 0x1; | ||
316 | } | ||
317 | |||
318 | DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_AL, 0x1575, quirk_uli1575); | ||
319 | DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_AL, 0x5288, quirk_uli5288); | ||
320 | DECLARE_PCI_FIXUP_HEADER(PCI_VENDOR_ID_AL, 0x5229, quirk_uli5229); | ||
321 | DECLARE_PCI_FIXUP_EARLY(PCI_VENDOR_ID_AL, 0x5249, early_uli5249); | ||
322 | #endif /* CONFIG_PCI */ | ||
99 | 323 | ||
100 | /* | 324 | /* |
101 | * Setup the architecture | 325 | * Setup the architecture |
102 | */ | 326 | */ |
103 | static void __init mpc8544_ds_setup_arch(void) | 327 | static void __init mpc8544_ds_setup_arch(void) |
104 | { | 328 | { |
329 | #ifdef CONFIG_PCI | ||
330 | struct device_node *np; | ||
331 | #endif | ||
332 | |||
105 | if (ppc_md.progress) | 333 | if (ppc_md.progress) |
106 | ppc_md.progress("mpc8544_ds_setup_arch()", 0); | 334 | ppc_md.progress("mpc8544_ds_setup_arch()", 0); |
107 | 335 | ||
336 | #ifdef CONFIG_PCI | ||
337 | for (np = NULL; (np = of_find_node_by_type(np, "pci")) != NULL;) { | ||
338 | struct resource rsrc; | ||
339 | of_address_to_resource(np, 0, &rsrc); | ||
340 | if ((rsrc.start & 0xfffff) == 0xb000) | ||
341 | fsl_add_bridge(np, 1); | ||
342 | else | ||
343 | fsl_add_bridge(np, 0); | ||
344 | } | ||
345 | #endif | ||
346 | |||
108 | printk("MPC8544 DS board from Freescale Semiconductor\n"); | 347 | printk("MPC8544 DS board from Freescale Semiconductor\n"); |
109 | } | 348 | } |
110 | 349 | ||
111 | |||
112 | /* | 350 | /* |
113 | * Called very early, device-tree isn't unflattened | 351 | * Called very early, device-tree isn't unflattened |
114 | */ | 352 | */ |
@@ -124,6 +362,7 @@ define_machine(mpc8544_ds) { | |||
124 | .probe = mpc8544_ds_probe, | 362 | .probe = mpc8544_ds_probe, |
125 | .setup_arch = mpc8544_ds_setup_arch, | 363 | .setup_arch = mpc8544_ds_setup_arch, |
126 | .init_IRQ = mpc8544_ds_pic_init, | 364 | .init_IRQ = mpc8544_ds_pic_init, |
365 | .pcibios_fixup_bus = fsl_pcibios_fixup_bus, | ||
127 | .get_irq = mpic_get_irq, | 366 | .get_irq = mpic_get_irq, |
128 | .restart = mpc85xx_restart, | 367 | .restart = mpc85xx_restart, |
129 | .calibrate_decr = generic_calibrate_decr, | 368 | .calibrate_decr = generic_calibrate_decr, |
diff --git a/arch/powerpc/platforms/85xx/mpc85xx.h b/arch/powerpc/platforms/85xx/mpc85xx.h index 7286ffac2c1d..5b34deef12b5 100644 --- a/arch/powerpc/platforms/85xx/mpc85xx.h +++ b/arch/powerpc/platforms/85xx/mpc85xx.h | |||
@@ -15,4 +15,3 @@ | |||
15 | */ | 15 | */ |
16 | 16 | ||
17 | extern void mpc85xx_restart(char *); | 17 | extern void mpc85xx_restart(char *); |
18 | extern int mpc85xx_add_bridge(struct device_node *dev); | ||
diff --git a/arch/powerpc/platforms/85xx/mpc85xx_ads.c b/arch/powerpc/platforms/85xx/mpc85xx_ads.c index 7235f702394c..40a828675c7b 100644 --- a/arch/powerpc/platforms/85xx/mpc85xx_ads.c +++ b/arch/powerpc/platforms/85xx/mpc85xx_ads.c | |||
@@ -29,6 +29,7 @@ | |||
29 | #include <asm/udbg.h> | 29 | #include <asm/udbg.h> |
30 | 30 | ||
31 | #include <sysdev/fsl_soc.h> | 31 | #include <sysdev/fsl_soc.h> |
32 | #include <sysdev/fsl_pci.h> | ||
32 | #include "mpc85xx.h" | 33 | #include "mpc85xx.h" |
33 | 34 | ||
34 | #ifdef CONFIG_CPM2 | 35 | #ifdef CONFIG_CPM2 |
@@ -217,7 +218,7 @@ static void __init mpc85xx_ads_setup_arch(void) | |||
217 | 218 | ||
218 | #ifdef CONFIG_PCI | 219 | #ifdef CONFIG_PCI |
219 | for (np = NULL; (np = of_find_node_by_type(np, "pci")) != NULL;) | 220 | for (np = NULL; (np = of_find_node_by_type(np, "pci")) != NULL;) |
220 | mpc85xx_add_bridge(np); | 221 | fsl_add_bridge(np, 1); |
221 | ppc_md.pci_exclude_device = mpc85xx_exclude_device; | 222 | ppc_md.pci_exclude_device = mpc85xx_exclude_device; |
222 | #endif | 223 | #endif |
223 | } | 224 | } |
diff --git a/arch/powerpc/platforms/85xx/mpc85xx_cds.c b/arch/powerpc/platforms/85xx/mpc85xx_cds.c index 50c8d6458362..6a171e9abf7d 100644 --- a/arch/powerpc/platforms/85xx/mpc85xx_cds.c +++ b/arch/powerpc/platforms/85xx/mpc85xx_cds.c | |||
@@ -24,6 +24,7 @@ | |||
24 | #include <linux/seq_file.h> | 24 | #include <linux/seq_file.h> |
25 | #include <linux/initrd.h> | 25 | #include <linux/initrd.h> |
26 | #include <linux/module.h> | 26 | #include <linux/module.h> |
27 | #include <linux/interrupt.h> | ||
27 | #include <linux/fsl_devices.h> | 28 | #include <linux/fsl_devices.h> |
28 | 29 | ||
29 | #include <asm/system.h> | 30 | #include <asm/system.h> |
@@ -45,6 +46,7 @@ | |||
45 | #include <asm/i8259.h> | 46 | #include <asm/i8259.h> |
46 | 47 | ||
47 | #include <sysdev/fsl_soc.h> | 48 | #include <sysdev/fsl_soc.h> |
49 | #include <sysdev/fsl_pci.h> | ||
48 | #include "mpc85xx.h" | 50 | #include "mpc85xx.h" |
49 | 51 | ||
50 | static int cds_pci_slot = 2; | 52 | static int cds_pci_slot = 2; |
@@ -58,8 +60,6 @@ static volatile u8 *cadmus; | |||
58 | static int mpc85xx_exclude_device(struct pci_controller *hose, | 60 | static int mpc85xx_exclude_device(struct pci_controller *hose, |
59 | u_char bus, u_char devfn) | 61 | u_char bus, u_char devfn) |
60 | { | 62 | { |
61 | if ((bus == hose->first_busno) && PCI_SLOT(devfn) == 0) | ||
62 | return PCIBIOS_DEVICE_NOT_FOUND; | ||
63 | /* We explicitly do not go past the Tundra 320 Bridge */ | 63 | /* We explicitly do not go past the Tundra 320 Bridge */ |
64 | if ((bus == 1) && (PCI_SLOT(devfn) == ARCADIA_2ND_BRIDGE_IDSEL)) | 64 | if ((bus == 1) && (PCI_SLOT(devfn) == ARCADIA_2ND_BRIDGE_IDSEL)) |
65 | return PCIBIOS_DEVICE_NOT_FOUND; | 65 | return PCIBIOS_DEVICE_NOT_FOUND; |
@@ -69,6 +69,37 @@ static int mpc85xx_exclude_device(struct pci_controller *hose, | |||
69 | return PCIBIOS_SUCCESSFUL; | 69 | return PCIBIOS_SUCCESSFUL; |
70 | } | 70 | } |
71 | 71 | ||
72 | static void mpc85xx_cds_restart(char *cmd) | ||
73 | { | ||
74 | struct pci_dev *dev; | ||
75 | u_char tmp; | ||
76 | |||
77 | if ((dev = pci_get_device(PCI_VENDOR_ID_VIA, PCI_DEVICE_ID_VIA_82C686, | ||
78 | NULL))) { | ||
79 | |||
80 | /* Use the VIA Super Southbridge to force a PCI reset */ | ||
81 | pci_read_config_byte(dev, 0x47, &tmp); | ||
82 | pci_write_config_byte(dev, 0x47, tmp | 1); | ||
83 | |||
84 | /* Flush the outbound PCI write queues */ | ||
85 | pci_read_config_byte(dev, 0x47, &tmp); | ||
86 | |||
87 | /* | ||
88 | * At this point, the harware reset should have triggered. | ||
89 | * However, if it doesn't work for some mysterious reason, | ||
90 | * just fall through to the default reset below. | ||
91 | */ | ||
92 | |||
93 | pci_dev_put(dev); | ||
94 | } | ||
95 | |||
96 | /* | ||
97 | * If we can't find the VIA chip (maybe the P2P bridge is disabled) | ||
98 | * or the VIA chip reset didn't work, just use the default reset. | ||
99 | */ | ||
100 | mpc85xx_restart(NULL); | ||
101 | } | ||
102 | |||
72 | static void __init mpc85xx_cds_pci_irq_fixup(struct pci_dev *dev) | 103 | static void __init mpc85xx_cds_pci_irq_fixup(struct pci_dev *dev) |
73 | { | 104 | { |
74 | u_char c; | 105 | u_char c; |
@@ -98,7 +129,7 @@ static void __init mpc85xx_cds_pci_irq_fixup(struct pci_dev *dev) | |||
98 | /* There are two USB controllers. | 129 | /* There are two USB controllers. |
99 | * Identify them by functon number | 130 | * Identify them by functon number |
100 | */ | 131 | */ |
101 | if (PCI_FUNC(dev->devfn)) | 132 | if (PCI_FUNC(dev->devfn) == 3) |
102 | dev->irq = 11; | 133 | dev->irq = 11; |
103 | else | 134 | else |
104 | dev->irq = 10; | 135 | dev->irq = 10; |
@@ -109,17 +140,41 @@ static void __init mpc85xx_cds_pci_irq_fixup(struct pci_dev *dev) | |||
109 | } | 140 | } |
110 | } | 141 | } |
111 | 142 | ||
143 | static void __devinit skip_fake_bridge(struct pci_dev *dev) | ||
144 | { | ||
145 | /* Make it an error to skip the fake bridge | ||
146 | * in pci_setup_device() in probe.c */ | ||
147 | dev->hdr_type = 0x7f; | ||
148 | } | ||
149 | DECLARE_PCI_FIXUP_EARLY(0x1957, 0x3fff, skip_fake_bridge); | ||
150 | DECLARE_PCI_FIXUP_EARLY(0x3fff, 0x1957, skip_fake_bridge); | ||
151 | DECLARE_PCI_FIXUP_EARLY(0xff3f, 0x5719, skip_fake_bridge); | ||
152 | |||
112 | #ifdef CONFIG_PPC_I8259 | 153 | #ifdef CONFIG_PPC_I8259 |
113 | #warning The i8259 PIC support is currently broken | 154 | static void mpc85xx_8259_cascade_handler(unsigned int irq, |
114 | static void mpc85xx_8259_cascade(unsigned int irq, struct irq_desc *desc) | 155 | struct irq_desc *desc) |
115 | { | 156 | { |
116 | unsigned int cascade_irq = i8259_irq(); | 157 | unsigned int cascade_irq = i8259_irq(); |
117 | 158 | ||
118 | if (cascade_irq != NO_IRQ) | 159 | if (cascade_irq != NO_IRQ) |
160 | /* handle an interrupt from the 8259 */ | ||
119 | generic_handle_irq(cascade_irq); | 161 | generic_handle_irq(cascade_irq); |
120 | 162 | ||
121 | desc->chip->eoi(irq); | 163 | /* check for any interrupts from the shared IRQ line */ |
164 | handle_fasteoi_irq(irq, desc); | ||
165 | } | ||
166 | |||
167 | static irqreturn_t mpc85xx_8259_cascade_action(int irq, void *dev_id) | ||
168 | { | ||
169 | return IRQ_HANDLED; | ||
122 | } | 170 | } |
171 | |||
172 | static struct irqaction mpc85xxcds_8259_irqaction = { | ||
173 | .handler = mpc85xx_8259_cascade_action, | ||
174 | .flags = IRQF_SHARED, | ||
175 | .mask = CPU_MASK_NONE, | ||
176 | .name = "8259 cascade", | ||
177 | }; | ||
123 | #endif /* PPC_I8259 */ | 178 | #endif /* PPC_I8259 */ |
124 | #endif /* CONFIG_PCI */ | 179 | #endif /* CONFIG_PCI */ |
125 | 180 | ||
@@ -128,10 +183,6 @@ static void __init mpc85xx_cds_pic_init(void) | |||
128 | struct mpic *mpic; | 183 | struct mpic *mpic; |
129 | struct resource r; | 184 | struct resource r; |
130 | struct device_node *np = NULL; | 185 | struct device_node *np = NULL; |
131 | #ifdef CONFIG_PPC_I8259 | ||
132 | struct device_node *cascade_node = NULL; | ||
133 | int cascade_irq; | ||
134 | #endif | ||
135 | 186 | ||
136 | np = of_find_node_by_type(np, "open-pic"); | 187 | np = of_find_node_by_type(np, "open-pic"); |
137 | 188 | ||
@@ -155,8 +206,19 @@ static void __init mpc85xx_cds_pic_init(void) | |||
155 | of_node_put(np); | 206 | of_node_put(np); |
156 | 207 | ||
157 | mpic_init(mpic); | 208 | mpic_init(mpic); |
209 | } | ||
210 | |||
211 | #if defined(CONFIG_PPC_I8259) && defined(CONFIG_PCI) | ||
212 | static int mpc85xx_cds_8259_attach(void) | ||
213 | { | ||
214 | int ret; | ||
215 | struct device_node *np = NULL; | ||
216 | struct device_node *cascade_node = NULL; | ||
217 | int cascade_irq; | ||
218 | |||
219 | if (!machine_is(mpc85xx_cds)) | ||
220 | return 0; | ||
158 | 221 | ||
159 | #ifdef CONFIG_PPC_I8259 | ||
160 | /* Initialize the i8259 controller */ | 222 | /* Initialize the i8259 controller */ |
161 | for_each_node_by_type(np, "interrupt-controller") | 223 | for_each_node_by_type(np, "interrupt-controller") |
162 | if (of_device_is_compatible(np, "chrp,iic")) { | 224 | if (of_device_is_compatible(np, "chrp,iic")) { |
@@ -166,22 +228,39 @@ static void __init mpc85xx_cds_pic_init(void) | |||
166 | 228 | ||
167 | if (cascade_node == NULL) { | 229 | if (cascade_node == NULL) { |
168 | printk(KERN_DEBUG "Could not find i8259 PIC\n"); | 230 | printk(KERN_DEBUG "Could not find i8259 PIC\n"); |
169 | return; | 231 | return -ENODEV; |
170 | } | 232 | } |
171 | 233 | ||
172 | cascade_irq = irq_of_parse_and_map(cascade_node, 0); | 234 | cascade_irq = irq_of_parse_and_map(cascade_node, 0); |
173 | if (cascade_irq == NO_IRQ) { | 235 | if (cascade_irq == NO_IRQ) { |
174 | printk(KERN_ERR "Failed to map cascade interrupt\n"); | 236 | printk(KERN_ERR "Failed to map cascade interrupt\n"); |
175 | return; | 237 | return -ENXIO; |
176 | } | 238 | } |
177 | 239 | ||
178 | i8259_init(cascade_node, 0); | 240 | i8259_init(cascade_node, 0); |
179 | of_node_put(cascade_node); | 241 | of_node_put(cascade_node); |
180 | 242 | ||
181 | set_irq_chained_handler(cascade_irq, mpc85xx_8259_cascade); | 243 | /* |
182 | #endif /* CONFIG_PPC_I8259 */ | 244 | * Hook the interrupt to make sure desc->action is never NULL. |
245 | * This is required to ensure that the interrupt does not get | ||
246 | * disabled when the last user of the shared IRQ line frees their | ||
247 | * interrupt. | ||
248 | */ | ||
249 | if ((ret = setup_irq(cascade_irq, &mpc85xxcds_8259_irqaction))) { | ||
250 | printk(KERN_ERR "Failed to setup cascade interrupt\n"); | ||
251 | return ret; | ||
252 | } | ||
253 | |||
254 | /* Success. Connect our low-level cascade handler. */ | ||
255 | set_irq_handler(cascade_irq, mpc85xx_8259_cascade_handler); | ||
256 | |||
257 | return 0; | ||
183 | } | 258 | } |
184 | 259 | ||
260 | device_initcall(mpc85xx_cds_8259_attach); | ||
261 | |||
262 | #endif /* CONFIG_PPC_I8259 */ | ||
263 | |||
185 | /* | 264 | /* |
186 | * Setup the architecture | 265 | * Setup the architecture |
187 | */ | 266 | */ |
@@ -218,9 +297,14 @@ static void __init mpc85xx_cds_setup_arch(void) | |||
218 | } | 297 | } |
219 | 298 | ||
220 | #ifdef CONFIG_PCI | 299 | #ifdef CONFIG_PCI |
221 | for (np = NULL; (np = of_find_node_by_type(np, "pci")) != NULL;) | 300 | for (np = NULL; (np = of_find_node_by_type(np, "pci")) != NULL;) { |
222 | mpc85xx_add_bridge(np); | 301 | struct resource rsrc; |
223 | 302 | of_address_to_resource(np, 0, &rsrc); | |
303 | if ((rsrc.start & 0xfffff) == 0x8000) | ||
304 | fsl_add_bridge(np, 1); | ||
305 | else | ||
306 | fsl_add_bridge(np, 0); | ||
307 | } | ||
224 | ppc_md.pci_irq_fixup = mpc85xx_cds_pci_irq_fixup; | 308 | ppc_md.pci_irq_fixup = mpc85xx_cds_pci_irq_fixup; |
225 | ppc_md.pci_exclude_device = mpc85xx_exclude_device; | 309 | ppc_md.pci_exclude_device = mpc85xx_exclude_device; |
226 | #endif | 310 | #endif |
@@ -265,7 +349,12 @@ define_machine(mpc85xx_cds) { | |||
265 | .init_IRQ = mpc85xx_cds_pic_init, | 349 | .init_IRQ = mpc85xx_cds_pic_init, |
266 | .show_cpuinfo = mpc85xx_cds_show_cpuinfo, | 350 | .show_cpuinfo = mpc85xx_cds_show_cpuinfo, |
267 | .get_irq = mpic_get_irq, | 351 | .get_irq = mpic_get_irq, |
352 | #ifdef CONFIG_PCI | ||
353 | .restart = mpc85xx_cds_restart, | ||
354 | #else | ||
268 | .restart = mpc85xx_restart, | 355 | .restart = mpc85xx_restart, |
356 | #endif | ||
269 | .calibrate_decr = generic_calibrate_decr, | 357 | .calibrate_decr = generic_calibrate_decr, |
270 | .progress = udbg_progress, | 358 | .progress = udbg_progress, |
359 | .pcibios_fixup_bus = fsl_pcibios_fixup_bus, | ||
271 | }; | 360 | }; |
diff --git a/arch/powerpc/platforms/85xx/mpc85xx_mds.c b/arch/powerpc/platforms/85xx/mpc85xx_mds.c index 004b80bd0b84..e8003bf00c9a 100644 --- a/arch/powerpc/platforms/85xx/mpc85xx_mds.c +++ b/arch/powerpc/platforms/85xx/mpc85xx_mds.c | |||
@@ -46,6 +46,7 @@ | |||
46 | #include <asm/prom.h> | 46 | #include <asm/prom.h> |
47 | #include <asm/udbg.h> | 47 | #include <asm/udbg.h> |
48 | #include <sysdev/fsl_soc.h> | 48 | #include <sysdev/fsl_soc.h> |
49 | #include <sysdev/fsl_pci.h> | ||
49 | #include <asm/qe.h> | 50 | #include <asm/qe.h> |
50 | #include <asm/qe_ic.h> | 51 | #include <asm/qe_ic.h> |
51 | #include <asm/mpic.h> | 52 | #include <asm/mpic.h> |
@@ -94,9 +95,8 @@ static void __init mpc85xx_mds_setup_arch(void) | |||
94 | } | 95 | } |
95 | 96 | ||
96 | #ifdef CONFIG_PCI | 97 | #ifdef CONFIG_PCI |
97 | for (np = NULL; (np = of_find_node_by_type(np, "pci")) != NULL;) { | 98 | for (np = NULL; (np = of_find_node_by_type(np, "pci")) != NULL;) |
98 | mpc85xx_add_bridge(np); | 99 | fsl_add_bridge(np, 1); |
99 | } | ||
100 | of_node_put(np); | 100 | of_node_put(np); |
101 | #endif | 101 | #endif |
102 | 102 | ||
@@ -208,4 +208,5 @@ define_machine(mpc85xx_mds) { | |||
208 | .restart = mpc85xx_restart, | 208 | .restart = mpc85xx_restart, |
209 | .calibrate_decr = generic_calibrate_decr, | 209 | .calibrate_decr = generic_calibrate_decr, |
210 | .progress = udbg_progress, | 210 | .progress = udbg_progress, |
211 | .pcibios_fixup_bus = fsl_pcibios_fixup_bus, | ||
211 | }; | 212 | }; |
diff --git a/arch/powerpc/platforms/85xx/pci.c b/arch/powerpc/platforms/85xx/pci.c deleted file mode 100644 index 8118417b7364..000000000000 --- a/arch/powerpc/platforms/85xx/pci.c +++ /dev/null | |||
@@ -1,91 +0,0 @@ | |||
1 | /* | ||
2 | * FSL SoC setup code | ||
3 | * | ||
4 | * Maintained by Kumar Gala (see MAINTAINERS for contact information) | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify it | ||
7 | * under the terms of the GNU General Public License as published by the | ||
8 | * Free Software Foundation; either version 2 of the License, or (at your | ||
9 | * option) any later version. | ||
10 | */ | ||
11 | |||
12 | #include <linux/stddef.h> | ||
13 | #include <linux/kernel.h> | ||
14 | #include <linux/init.h> | ||
15 | #include <linux/errno.h> | ||
16 | #include <linux/pci.h> | ||
17 | #include <linux/delay.h> | ||
18 | #include <linux/irq.h> | ||
19 | #include <linux/module.h> | ||
20 | |||
21 | #include <asm/system.h> | ||
22 | #include <asm/atomic.h> | ||
23 | #include <asm/io.h> | ||
24 | #include <asm/pci-bridge.h> | ||
25 | #include <asm/prom.h> | ||
26 | #include <sysdev/fsl_soc.h> | ||
27 | |||
28 | #undef DEBUG | ||
29 | |||
30 | #ifdef DEBUG | ||
31 | #define DBG(x...) printk(x) | ||
32 | #else | ||
33 | #define DBG(x...) | ||
34 | #endif | ||
35 | |||
36 | #ifdef CONFIG_PCI | ||
37 | int __init mpc85xx_add_bridge(struct device_node *dev) | ||
38 | { | ||
39 | int len; | ||
40 | struct pci_controller *hose; | ||
41 | struct resource rsrc; | ||
42 | const int *bus_range; | ||
43 | int primary = 1, has_address = 0; | ||
44 | phys_addr_t immr = get_immrbase(); | ||
45 | |||
46 | DBG("Adding PCI host bridge %s\n", dev->full_name); | ||
47 | |||
48 | /* Fetch host bridge registers address */ | ||
49 | has_address = (of_address_to_resource(dev, 0, &rsrc) == 0); | ||
50 | |||
51 | /* Get bus range if any */ | ||
52 | bus_range = of_get_property(dev, "bus-range", &len); | ||
53 | if (bus_range == NULL || len < 2 * sizeof(int)) { | ||
54 | printk(KERN_WARNING "Can't get bus-range for %s, assume" | ||
55 | " bus 0\n", dev->full_name); | ||
56 | } | ||
57 | |||
58 | pci_assign_all_buses = 1; | ||
59 | hose = pcibios_alloc_controller(dev); | ||
60 | if (!hose) | ||
61 | return -ENOMEM; | ||
62 | |||
63 | hose->first_busno = bus_range ? bus_range[0] : 0; | ||
64 | hose->last_busno = bus_range ? bus_range[1] : 0xff; | ||
65 | |||
66 | /* PCI 1 */ | ||
67 | if ((rsrc.start & 0xfffff) == 0x8000) { | ||
68 | setup_indirect_pci(hose, immr + 0x8000, immr + 0x8004); | ||
69 | } | ||
70 | /* PCI 2 */ | ||
71 | if ((rsrc.start & 0xfffff) == 0x9000) { | ||
72 | setup_indirect_pci(hose, immr + 0x9000, immr + 0x9004); | ||
73 | primary = 0; | ||
74 | } | ||
75 | |||
76 | printk(KERN_INFO "Found MPC85xx PCI host bridge at 0x%016llx. " | ||
77 | "Firmware bus number: %d->%d\n", | ||
78 | (unsigned long long)rsrc.start, hose->first_busno, | ||
79 | hose->last_busno); | ||
80 | |||
81 | DBG(" ->Hose at 0x%p, cfg_addr=0x%p,cfg_data=0x%p\n", | ||
82 | hose, hose->cfg_addr, hose->cfg_data); | ||
83 | |||
84 | /* Interpret the "ranges" property */ | ||
85 | /* This also maps the I/O region and sets isa_io/mem_base */ | ||
86 | pci_process_bridge_OF_ranges(hose, dev, primary); | ||
87 | |||
88 | return 0; | ||
89 | } | ||
90 | |||
91 | #endif | ||
diff --git a/arch/powerpc/platforms/86xx/Kconfig b/arch/powerpc/platforms/86xx/Kconfig index 0faebfdc1596..343b76d0d793 100644 --- a/arch/powerpc/platforms/86xx/Kconfig +++ b/arch/powerpc/platforms/86xx/Kconfig | |||
@@ -14,8 +14,7 @@ endchoice | |||
14 | 14 | ||
15 | config MPC8641 | 15 | config MPC8641 |
16 | bool | 16 | bool |
17 | select PPC_INDIRECT_PCI | 17 | select FSL_PCI if PCI |
18 | select PPC_INDIRECT_PCI_BE | ||
19 | select PPC_UDBG_16550 | 18 | select PPC_UDBG_16550 |
20 | select MPIC | 19 | select MPIC |
21 | default y if MPC8641_HPCN | 20 | default y if MPC8641_HPCN |
diff --git a/arch/powerpc/platforms/86xx/Makefile b/arch/powerpc/platforms/86xx/Makefile index 418fd8f4d268..3376c7767f2d 100644 --- a/arch/powerpc/platforms/86xx/Makefile +++ b/arch/powerpc/platforms/86xx/Makefile | |||
@@ -4,4 +4,3 @@ | |||
4 | 4 | ||
5 | obj-$(CONFIG_SMP) += mpc86xx_smp.o | 5 | obj-$(CONFIG_SMP) += mpc86xx_smp.o |
6 | obj-$(CONFIG_MPC8641_HPCN) += mpc86xx_hpcn.o | 6 | obj-$(CONFIG_MPC8641_HPCN) += mpc86xx_hpcn.o |
7 | obj-$(CONFIG_PCI) += pci.o | ||
diff --git a/arch/powerpc/platforms/86xx/mpc86xx.h b/arch/powerpc/platforms/86xx/mpc86xx.h index 23f7ed2a7f88..525ffa1904f9 100644 --- a/arch/powerpc/platforms/86xx/mpc86xx.h +++ b/arch/powerpc/platforms/86xx/mpc86xx.h | |||
@@ -15,11 +15,6 @@ | |||
15 | * mpc86xx_* files. Mostly for use by mpc86xx_setup(). | 15 | * mpc86xx_* files. Mostly for use by mpc86xx_setup(). |
16 | */ | 16 | */ |
17 | 17 | ||
18 | extern int mpc86xx_add_bridge(struct device_node *dev); | ||
19 | |||
20 | extern int mpc86xx_exclude_device(struct pci_controller *hose, | ||
21 | u_char bus, u_char devfn); | ||
22 | |||
23 | extern void __init mpc86xx_smp_init(void); | 18 | extern void __init mpc86xx_smp_init(void); |
24 | 19 | ||
25 | #endif /* __MPC86XX_H__ */ | 20 | #endif /* __MPC86XX_H__ */ |
diff --git a/arch/powerpc/platforms/86xx/mpc86xx_hpcn.c b/arch/powerpc/platforms/86xx/mpc86xx_hpcn.c index 5b01ec7c13dc..e9eaa0749ae6 100644 --- a/arch/powerpc/platforms/86xx/mpc86xx_hpcn.c +++ b/arch/powerpc/platforms/86xx/mpc86xx_hpcn.c | |||
@@ -31,6 +31,7 @@ | |||
31 | 31 | ||
32 | #include <asm/mpic.h> | 32 | #include <asm/mpic.h> |
33 | 33 | ||
34 | #include <sysdev/fsl_pci.h> | ||
34 | #include <sysdev/fsl_soc.h> | 35 | #include <sysdev/fsl_soc.h> |
35 | 36 | ||
36 | #include "mpc86xx.h" | 37 | #include "mpc86xx.h" |
@@ -344,8 +345,14 @@ mpc86xx_hpcn_setup_arch(void) | |||
344 | } | 345 | } |
345 | 346 | ||
346 | #ifdef CONFIG_PCI | 347 | #ifdef CONFIG_PCI |
347 | for (np = NULL; (np = of_find_node_by_type(np, "pci")) != NULL;) | 348 | for (np = NULL; (np = of_find_node_by_type(np, "pci")) != NULL;) { |
348 | mpc86xx_add_bridge(np); | 349 | struct resource rsrc; |
350 | of_address_to_resource(np, 0, &rsrc); | ||
351 | if ((rsrc.start & 0xfffff) == 0x8000) | ||
352 | fsl_add_bridge(np, 1); | ||
353 | else | ||
354 | fsl_add_bridge(np, 0); | ||
355 | } | ||
349 | #endif | 356 | #endif |
350 | 357 | ||
351 | printk("MPC86xx HPCN board from Freescale Semiconductor\n"); | 358 | printk("MPC86xx HPCN board from Freescale Semiconductor\n"); |
@@ -424,7 +431,6 @@ mpc86xx_time_init(void) | |||
424 | return 0; | 431 | return 0; |
425 | } | 432 | } |
426 | 433 | ||
427 | |||
428 | define_machine(mpc86xx_hpcn) { | 434 | define_machine(mpc86xx_hpcn) { |
429 | .name = "MPC86xx HPCN", | 435 | .name = "MPC86xx HPCN", |
430 | .probe = mpc86xx_hpcn_probe, | 436 | .probe = mpc86xx_hpcn_probe, |
@@ -436,4 +442,5 @@ define_machine(mpc86xx_hpcn) { | |||
436 | .time_init = mpc86xx_time_init, | 442 | .time_init = mpc86xx_time_init, |
437 | .calibrate_decr = generic_calibrate_decr, | 443 | .calibrate_decr = generic_calibrate_decr, |
438 | .progress = udbg_progress, | 444 | .progress = udbg_progress, |
445 | .pcibios_fixup_bus = fsl_pcibios_fixup_bus, | ||
439 | }; | 446 | }; |
diff --git a/arch/powerpc/platforms/86xx/pci.c b/arch/powerpc/platforms/86xx/pci.c deleted file mode 100644 index 73cd5b05a84e..000000000000 --- a/arch/powerpc/platforms/86xx/pci.c +++ /dev/null | |||
@@ -1,238 +0,0 @@ | |||
1 | /* | ||
2 | * MPC86XX pci setup code | ||
3 | * | ||
4 | * Recode: ZHANG WEI <wei.zhang@freescale.com> | ||
5 | * Initial author: Xianghua Xiao <x.xiao@freescale.com> | ||
6 | * | ||
7 | * Copyright 2006 Freescale Semiconductor Inc. | ||
8 | * | ||
9 | * This program is free software; you can redistribute it and/or modify it | ||
10 | * under the terms of the GNU General Public License as published by the | ||
11 | * Free Software Foundation; either version 2 of the License, or (at your | ||
12 | * option) any later version. | ||
13 | */ | ||
14 | |||
15 | #include <linux/types.h> | ||
16 | #include <linux/module.h> | ||
17 | #include <linux/init.h> | ||
18 | #include <linux/pci.h> | ||
19 | #include <linux/serial.h> | ||
20 | |||
21 | #include <asm/system.h> | ||
22 | #include <asm/atomic.h> | ||
23 | #include <asm/io.h> | ||
24 | #include <asm/prom.h> | ||
25 | #include <asm/pci-bridge.h> | ||
26 | #include <sysdev/fsl_soc.h> | ||
27 | #include <sysdev/fsl_pcie.h> | ||
28 | |||
29 | #include "mpc86xx.h" | ||
30 | |||
31 | #undef DEBUG | ||
32 | |||
33 | #ifdef DEBUG | ||
34 | #define DBG(fmt, args...) printk(KERN_ERR "%s: " fmt, __FUNCTION__, ## args) | ||
35 | #else | ||
36 | #define DBG(fmt, args...) | ||
37 | #endif | ||
38 | |||
39 | struct pcie_outbound_window_regs { | ||
40 | uint pexotar; /* 0x.0 - PCI Express outbound translation address register */ | ||
41 | uint pexotear; /* 0x.4 - PCI Express outbound translation extended address register */ | ||
42 | uint pexowbar; /* 0x.8 - PCI Express outbound window base address register */ | ||
43 | char res1[4]; | ||
44 | uint pexowar; /* 0x.10 - PCI Express outbound window attributes register */ | ||
45 | char res2[12]; | ||
46 | }; | ||
47 | |||
48 | struct pcie_inbound_window_regs { | ||
49 | uint pexitar; /* 0x.0 - PCI Express inbound translation address register */ | ||
50 | char res1[4]; | ||
51 | uint pexiwbar; /* 0x.8 - PCI Express inbound window base address register */ | ||
52 | uint pexiwbear; /* 0x.c - PCI Express inbound window base extended address register */ | ||
53 | uint pexiwar; /* 0x.10 - PCI Express inbound window attributes register */ | ||
54 | char res2[12]; | ||
55 | }; | ||
56 | |||
57 | static void __init setup_pcie_atmu(struct pci_controller *hose, struct resource *rsrc) | ||
58 | { | ||
59 | volatile struct ccsr_pex *pcie; | ||
60 | volatile struct pcie_outbound_window_regs *pcieow; | ||
61 | volatile struct pcie_inbound_window_regs *pcieiw; | ||
62 | int i = 0; | ||
63 | |||
64 | DBG("PCIE memory map start 0x%x, size 0x%x\n", rsrc->start, | ||
65 | rsrc->end - rsrc->start + 1); | ||
66 | pcie = ioremap(rsrc->start, rsrc->end - rsrc->start + 1); | ||
67 | |||
68 | /* Disable all windows (except pexowar0 since its ignored) */ | ||
69 | pcie->pexowar1 = 0; | ||
70 | pcie->pexowar2 = 0; | ||
71 | pcie->pexowar3 = 0; | ||
72 | pcie->pexowar4 = 0; | ||
73 | pcie->pexiwar1 = 0; | ||
74 | pcie->pexiwar2 = 0; | ||
75 | pcie->pexiwar3 = 0; | ||
76 | |||
77 | pcieow = (struct pcie_outbound_window_regs *)&pcie->pexotar1; | ||
78 | pcieiw = (struct pcie_inbound_window_regs *)&pcie->pexitar1; | ||
79 | |||
80 | /* Setup outbound MEM window */ | ||
81 | for(i = 0; i < 3; i++) | ||
82 | if (hose->mem_resources[i].flags & IORESOURCE_MEM){ | ||
83 | DBG("PCIE MEM resource start 0x%08x, size 0x%08x.\n", | ||
84 | hose->mem_resources[i].start, | ||
85 | hose->mem_resources[i].end | ||
86 | - hose->mem_resources[i].start + 1); | ||
87 | pcieow->pexotar = (hose->mem_resources[i].start) >> 12 | ||
88 | & 0x000fffff; | ||
89 | pcieow->pexotear = 0; | ||
90 | pcieow->pexowbar = (hose->mem_resources[i].start) >> 12 | ||
91 | & 0x000fffff; | ||
92 | /* Enable, Mem R/W */ | ||
93 | pcieow->pexowar = 0x80044000 | | ||
94 | (__ilog2(hose->mem_resources[i].end | ||
95 | - hose->mem_resources[i].start + 1) | ||
96 | - 1); | ||
97 | pcieow++; | ||
98 | } | ||
99 | |||
100 | /* Setup outbound IO window */ | ||
101 | if (hose->io_resource.flags & IORESOURCE_IO){ | ||
102 | DBG("PCIE IO resource start 0x%08x, size 0x%08x, phy base 0x%08x.\n", | ||
103 | hose->io_resource.start, | ||
104 | hose->io_resource.end - hose->io_resource.start + 1, | ||
105 | hose->io_base_phys); | ||
106 | pcieow->pexotar = (hose->io_resource.start) >> 12 & 0x000fffff; | ||
107 | pcieow->pexotear = 0; | ||
108 | pcieow->pexowbar = (hose->io_base_phys) >> 12 & 0x000fffff; | ||
109 | /* Enable, IO R/W */ | ||
110 | pcieow->pexowar = 0x80088000 | (__ilog2(hose->io_resource.end | ||
111 | - hose->io_resource.start + 1) - 1); | ||
112 | } | ||
113 | |||
114 | /* Setup 2G inbound Memory Window @ 0 */ | ||
115 | pcieiw->pexitar = 0x00000000; | ||
116 | pcieiw->pexiwbar = 0x00000000; | ||
117 | /* Enable, Prefetch, Local Mem, Snoop R/W, 2G */ | ||
118 | pcieiw->pexiwar = 0xa0f5501e; | ||
119 | } | ||
120 | |||
121 | static void __init | ||
122 | mpc86xx_setup_pcie(struct pci_controller *hose, u32 pcie_offset, u32 pcie_size) | ||
123 | { | ||
124 | u16 cmd; | ||
125 | |||
126 | DBG("PCIE host controller register offset 0x%08x, size 0x%08x.\n", | ||
127 | pcie_offset, pcie_size); | ||
128 | |||
129 | early_read_config_word(hose, 0, 0, PCI_COMMAND, &cmd); | ||
130 | cmd |= PCI_COMMAND_SERR | PCI_COMMAND_MASTER | PCI_COMMAND_MEMORY | ||
131 | | PCI_COMMAND_IO; | ||
132 | early_write_config_word(hose, 0, 0, PCI_COMMAND, cmd); | ||
133 | |||
134 | early_write_config_byte(hose, 0, 0, PCI_LATENCY_TIMER, 0x80); | ||
135 | } | ||
136 | |||
137 | static void __devinit quirk_fsl_pcie_transparent(struct pci_dev *dev) | ||
138 | { | ||
139 | struct resource *res; | ||
140 | int i, res_idx = PCI_BRIDGE_RESOURCES; | ||
141 | struct pci_controller *hose; | ||
142 | |||
143 | /* | ||
144 | * Make the bridge be transparent. | ||
145 | */ | ||
146 | dev->transparent = 1; | ||
147 | |||
148 | hose = pci_bus_to_host(dev->bus); | ||
149 | if (!hose) { | ||
150 | printk(KERN_ERR "Can't find hose for bus %d\n", | ||
151 | dev->bus->number); | ||
152 | return; | ||
153 | } | ||
154 | |||
155 | if (hose->io_resource.flags) { | ||
156 | res = &dev->resource[res_idx++]; | ||
157 | res->start = hose->io_resource.start; | ||
158 | res->end = hose->io_resource.end; | ||
159 | res->flags = hose->io_resource.flags; | ||
160 | } | ||
161 | |||
162 | for (i = 0; i < 3; i++) { | ||
163 | res = &dev->resource[res_idx + i]; | ||
164 | res->start = hose->mem_resources[i].start; | ||
165 | res->end = hose->mem_resources[i].end; | ||
166 | res->flags = hose->mem_resources[i].flags; | ||
167 | } | ||
168 | } | ||
169 | |||
170 | |||
171 | DECLARE_PCI_FIXUP_EARLY(0x1957, 0x7010, quirk_fsl_pcie_transparent); | ||
172 | DECLARE_PCI_FIXUP_EARLY(0x1957, 0x7011, quirk_fsl_pcie_transparent); | ||
173 | |||
174 | #define PCIE_LTSSM 0x404 /* PCIe Link Training and Status */ | ||
175 | #define PCIE_LTSSM_L0 0x16 /* L0 state */ | ||
176 | |||
177 | int __init mpc86xx_add_bridge(struct device_node *dev) | ||
178 | { | ||
179 | int len; | ||
180 | struct pci_controller *hose; | ||
181 | struct resource rsrc; | ||
182 | const int *bus_range; | ||
183 | int has_address = 0; | ||
184 | int primary = 0; | ||
185 | u16 val; | ||
186 | |||
187 | DBG("Adding PCIE host bridge %s\n", dev->full_name); | ||
188 | |||
189 | /* Fetch host bridge registers address */ | ||
190 | has_address = (of_address_to_resource(dev, 0, &rsrc) == 0); | ||
191 | |||
192 | /* Get bus range if any */ | ||
193 | bus_range = of_get_property(dev, "bus-range", &len); | ||
194 | if (bus_range == NULL || len < 2 * sizeof(int)) | ||
195 | printk(KERN_WARNING "Can't get bus-range for %s, assume" | ||
196 | " bus 0\n", dev->full_name); | ||
197 | |||
198 | pci_assign_all_buses = 1; | ||
199 | hose = pcibios_alloc_controller(dev); | ||
200 | if (!hose) | ||
201 | return -ENOMEM; | ||
202 | |||
203 | hose->indirect_type = PPC_INDIRECT_TYPE_EXT_REG | | ||
204 | PPC_INDIRECT_TYPE_SURPRESS_PRIMARY_BUS; | ||
205 | |||
206 | hose->first_busno = bus_range ? bus_range[0] : 0x0; | ||
207 | hose->last_busno = bus_range ? bus_range[1] : 0xff; | ||
208 | |||
209 | setup_indirect_pci(hose, rsrc.start, rsrc.start + 0x4); | ||
210 | |||
211 | /* Probe the hose link training status */ | ||
212 | early_read_config_word(hose, 0, 0, PCIE_LTSSM, &val); | ||
213 | if (val < PCIE_LTSSM_L0) | ||
214 | return -ENXIO; | ||
215 | |||
216 | /* Setup the PCIE host controller. */ | ||
217 | mpc86xx_setup_pcie(hose, rsrc.start, rsrc.end - rsrc.start + 1); | ||
218 | |||
219 | if ((rsrc.start & 0xfffff) == 0x8000) | ||
220 | primary = 1; | ||
221 | |||
222 | printk(KERN_INFO "Found MPC86xx PCIE host bridge at 0x%08lx. " | ||
223 | "Firmware bus number: %d->%d\n", | ||
224 | (unsigned long) rsrc.start, | ||
225 | hose->first_busno, hose->last_busno); | ||
226 | |||
227 | DBG(" ->Hose at 0x%p, cfg_addr=0x%p,cfg_data=0x%p\n", | ||
228 | hose, hose->cfg_addr, hose->cfg_data); | ||
229 | |||
230 | /* Interpret the "ranges" property */ | ||
231 | /* This also maps the I/O region and sets isa_io/mem_base */ | ||
232 | pci_process_bridge_OF_ranges(hose, dev, primary); | ||
233 | |||
234 | /* Setup PEX window registers */ | ||
235 | setup_pcie_atmu(hose, &rsrc); | ||
236 | |||
237 | return 0; | ||
238 | } | ||
diff --git a/arch/powerpc/platforms/cell/spufs/context.c b/arch/powerpc/platforms/cell/spufs/context.c index 6694f86d7000..9cb081c26e71 100644 --- a/arch/powerpc/platforms/cell/spufs/context.c +++ b/arch/powerpc/platforms/cell/spufs/context.c | |||
@@ -59,7 +59,8 @@ struct spu_context *alloc_spu_context(struct spu_gang *gang) | |||
59 | INIT_LIST_HEAD(&ctx->aff_list); | 59 | INIT_LIST_HEAD(&ctx->aff_list); |
60 | if (gang) | 60 | if (gang) |
61 | spu_gang_add_ctx(gang, ctx); | 61 | spu_gang_add_ctx(gang, ctx); |
62 | ctx->cpus_allowed = current->cpus_allowed; | 62 | |
63 | __spu_update_sched_info(ctx); | ||
63 | spu_set_timeslice(ctx); | 64 | spu_set_timeslice(ctx); |
64 | ctx->stats.util_state = SPU_UTIL_IDLE_LOADED; | 65 | ctx->stats.util_state = SPU_UTIL_IDLE_LOADED; |
65 | 66 | ||
diff --git a/arch/powerpc/platforms/cell/spufs/run.c b/arch/powerpc/platforms/cell/spufs/run.c index 0b50fa5cb39d..6abdd8fe2098 100644 --- a/arch/powerpc/platforms/cell/spufs/run.c +++ b/arch/powerpc/platforms/cell/spufs/run.c | |||
@@ -312,6 +312,7 @@ long spufs_run_spu(struct spu_context *ctx, u32 *npc, u32 *event) | |||
312 | spu_acquire(ctx); | 312 | spu_acquire(ctx); |
313 | if (ctx->state == SPU_STATE_SAVED) { | 313 | if (ctx->state == SPU_STATE_SAVED) { |
314 | __spu_update_sched_info(ctx); | 314 | __spu_update_sched_info(ctx); |
315 | spu_set_timeslice(ctx); | ||
315 | 316 | ||
316 | ret = spu_activate(ctx, 0); | 317 | ret = spu_activate(ctx, 0); |
317 | if (ret) { | 318 | if (ret) { |
@@ -322,6 +323,9 @@ long spufs_run_spu(struct spu_context *ctx, u32 *npc, u32 *event) | |||
322 | /* | 323 | /* |
323 | * We have to update the scheduling priority under active_mutex | 324 | * We have to update the scheduling priority under active_mutex |
324 | * to protect against find_victim(). | 325 | * to protect against find_victim(). |
326 | * | ||
327 | * No need to update the timeslice ASAP, it will get updated | ||
328 | * once the current one has expired. | ||
325 | */ | 329 | */ |
326 | spu_update_sched_info(ctx); | 330 | spu_update_sched_info(ctx); |
327 | } | 331 | } |
diff --git a/arch/powerpc/platforms/cell/spufs/sched.c b/arch/powerpc/platforms/cell/spufs/sched.c index 227968b4779d..758a80ac080a 100644 --- a/arch/powerpc/platforms/cell/spufs/sched.c +++ b/arch/powerpc/platforms/cell/spufs/sched.c | |||
@@ -927,10 +927,6 @@ int __init spu_sched_init(void) | |||
927 | INIT_LIST_HEAD(&spu_prio->runq[i]); | 927 | INIT_LIST_HEAD(&spu_prio->runq[i]); |
928 | __clear_bit(i, spu_prio->bitmap); | 928 | __clear_bit(i, spu_prio->bitmap); |
929 | } | 929 | } |
930 | for (i = 0; i < MAX_NUMNODES; i++) { | ||
931 | mutex_init(&cbe_spu_info[i].list_mutex); | ||
932 | INIT_LIST_HEAD(&cbe_spu_info[i].spus); | ||
933 | } | ||
934 | spin_lock_init(&spu_prio->runq_lock); | 930 | spin_lock_init(&spu_prio->runq_lock); |
935 | 931 | ||
936 | setup_timer(&spusched_timer, spusched_wake, 0); | 932 | setup_timer(&spusched_timer, spusched_wake, 0); |
diff --git a/arch/powerpc/platforms/chrp/pci.c b/arch/powerpc/platforms/chrp/pci.c index 3690624e49d4..28d1647b204e 100644 --- a/arch/powerpc/platforms/chrp/pci.c +++ b/arch/powerpc/platforms/chrp/pci.c | |||
@@ -181,7 +181,7 @@ setup_python(struct pci_controller *hose, struct device_node *dev) | |||
181 | } | 181 | } |
182 | iounmap(reg); | 182 | iounmap(reg); |
183 | 183 | ||
184 | setup_indirect_pci(hose, r.start + 0xf8000, r.start + 0xf8010); | 184 | setup_indirect_pci(hose, r.start + 0xf8000, r.start + 0xf8010, 0); |
185 | } | 185 | } |
186 | 186 | ||
187 | /* Marvell Discovery II based Pegasos 2 */ | 187 | /* Marvell Discovery II based Pegasos 2 */ |
@@ -277,13 +277,14 @@ chrp_find_bridges(void) | |||
277 | hose->cfg_data = p; | 277 | hose->cfg_data = p; |
278 | gg2_pci_config_base = p; | 278 | gg2_pci_config_base = p; |
279 | } else if (is_pegasos == 1) { | 279 | } else if (is_pegasos == 1) { |
280 | setup_indirect_pci(hose, 0xfec00cf8, 0xfee00cfc); | 280 | setup_indirect_pci(hose, 0xfec00cf8, 0xfee00cfc, 0); |
281 | } else if (is_pegasos == 2) { | 281 | } else if (is_pegasos == 2) { |
282 | setup_peg2(hose, dev); | 282 | setup_peg2(hose, dev); |
283 | } else if (!strncmp(model, "IBM,CPC710", 10)) { | 283 | } else if (!strncmp(model, "IBM,CPC710", 10)) { |
284 | setup_indirect_pci(hose, | 284 | setup_indirect_pci(hose, |
285 | r.start + 0x000f8000, | 285 | r.start + 0x000f8000, |
286 | r.start + 0x000f8010); | 286 | r.start + 0x000f8010, |
287 | 0); | ||
287 | if (index == 0) { | 288 | if (index == 0) { |
288 | dma = of_get_property(dev, "system-dma-base", | 289 | dma = of_get_property(dev, "system-dma-base", |
289 | &len); | 290 | &len); |
diff --git a/arch/powerpc/platforms/embedded6xx/linkstation.c b/arch/powerpc/platforms/embedded6xx/linkstation.c index f4d0a7a603f5..bd5ca58345a1 100644 --- a/arch/powerpc/platforms/embedded6xx/linkstation.c +++ b/arch/powerpc/platforms/embedded6xx/linkstation.c | |||
@@ -73,7 +73,7 @@ static int __init linkstation_add_bridge(struct device_node *dev) | |||
73 | return -ENOMEM; | 73 | return -ENOMEM; |
74 | hose->first_busno = bus_range ? bus_range[0] : 0; | 74 | hose->first_busno = bus_range ? bus_range[0] : 0; |
75 | hose->last_busno = bus_range ? bus_range[1] : 0xff; | 75 | hose->last_busno = bus_range ? bus_range[1] : 0xff; |
76 | setup_indirect_pci(hose, 0xfec00000, 0xfee00000); | 76 | setup_indirect_pci(hose, 0xfec00000, 0xfee00000, 0); |
77 | 77 | ||
78 | /* Interpret the "ranges" property */ | 78 | /* Interpret the "ranges" property */ |
79 | /* This also maps the I/O region and sets isa_io/mem_base */ | 79 | /* This also maps the I/O region and sets isa_io/mem_base */ |
diff --git a/arch/powerpc/platforms/iseries/lpevents.c b/arch/powerpc/platforms/iseries/lpevents.c index 91df52a1899a..34bdbbe3ce59 100644 --- a/arch/powerpc/platforms/iseries/lpevents.c +++ b/arch/powerpc/platforms/iseries/lpevents.c | |||
@@ -182,7 +182,7 @@ static int set_spread_lpevents(char *str) | |||
182 | } | 182 | } |
183 | __setup("spread_lpevents=", set_spread_lpevents); | 183 | __setup("spread_lpevents=", set_spread_lpevents); |
184 | 184 | ||
185 | void setup_hvlpevent_queue(void) | 185 | void __init setup_hvlpevent_queue(void) |
186 | { | 186 | { |
187 | void *eventStack; | 187 | void *eventStack; |
188 | 188 | ||
diff --git a/arch/powerpc/platforms/maple/pci.c b/arch/powerpc/platforms/maple/pci.c index fceaae40fe70..2542403288f9 100644 --- a/arch/powerpc/platforms/maple/pci.c +++ b/arch/powerpc/platforms/maple/pci.c | |||
@@ -490,6 +490,9 @@ static int __init maple_add_bridge(struct device_node *dev) | |||
490 | /* Fixup "bus-range" OF property */ | 490 | /* Fixup "bus-range" OF property */ |
491 | fixup_bus_range(dev); | 491 | fixup_bus_range(dev); |
492 | 492 | ||
493 | /* Check for legacy IOs */ | ||
494 | isa_bridge_find_early(hose); | ||
495 | |||
493 | return 0; | 496 | return 0; |
494 | } | 497 | } |
495 | 498 | ||
diff --git a/arch/powerpc/sysdev/Makefile b/arch/powerpc/sysdev/Makefile index 484eb4e0e9db..08ce31e612c2 100644 --- a/arch/powerpc/sysdev/Makefile +++ b/arch/powerpc/sysdev/Makefile | |||
@@ -12,6 +12,7 @@ obj-$(CONFIG_PPC_PMI) += pmi.o | |||
12 | obj-$(CONFIG_U3_DART) += dart_iommu.o | 12 | obj-$(CONFIG_U3_DART) += dart_iommu.o |
13 | obj-$(CONFIG_MMIO_NVRAM) += mmio_nvram.o | 13 | obj-$(CONFIG_MMIO_NVRAM) += mmio_nvram.o |
14 | obj-$(CONFIG_FSL_SOC) += fsl_soc.o | 14 | obj-$(CONFIG_FSL_SOC) += fsl_soc.o |
15 | obj-$(CONFIG_FSL_PCI) += fsl_pci.o | ||
15 | obj-$(CONFIG_TSI108_BRIDGE) += tsi108_pci.o tsi108_dev.o | 16 | obj-$(CONFIG_TSI108_BRIDGE) += tsi108_pci.o tsi108_dev.o |
16 | obj-$(CONFIG_QUICC_ENGINE) += qe_lib/ | 17 | obj-$(CONFIG_QUICC_ENGINE) += qe_lib/ |
17 | mv64x60-$(CONFIG_PCI) += mv64x60_pci.o | 18 | mv64x60-$(CONFIG_PCI) += mv64x60_pci.o |
diff --git a/arch/powerpc/sysdev/fsl_pci.c b/arch/powerpc/sysdev/fsl_pci.c new file mode 100644 index 000000000000..9fb0ce5c7176 --- /dev/null +++ b/arch/powerpc/sysdev/fsl_pci.c | |||
@@ -0,0 +1,257 @@ | |||
1 | /* | ||
2 | * MPC85xx/86xx PCI/PCIE support routing. | ||
3 | * | ||
4 | * Copyright 2007 Freescale Semiconductor, Inc | ||
5 | * | ||
6 | * Initial author: Xianghua Xiao <x.xiao@freescale.com> | ||
7 | * Recode: ZHANG WEI <wei.zhang@freescale.com> | ||
8 | * Rewrite the routing for Frescale PCI and PCI Express | ||
9 | * Roy Zang <tie-fei.zang@freescale.com> | ||
10 | * | ||
11 | * This program is free software; you can redistribute it and/or modify it | ||
12 | * under the terms of the GNU General Public License as published by the | ||
13 | * Free Software Foundation; either version 2 of the License, or (at your | ||
14 | * option) any later version. | ||
15 | */ | ||
16 | #include <linux/kernel.h> | ||
17 | #include <linux/pci.h> | ||
18 | #include <linux/delay.h> | ||
19 | #include <linux/string.h> | ||
20 | #include <linux/init.h> | ||
21 | #include <linux/bootmem.h> | ||
22 | |||
23 | #include <asm/io.h> | ||
24 | #include <asm/prom.h> | ||
25 | #include <asm/pci-bridge.h> | ||
26 | #include <asm/machdep.h> | ||
27 | #include <sysdev/fsl_soc.h> | ||
28 | #include <sysdev/fsl_pci.h> | ||
29 | |||
30 | /* atmu setup for fsl pci/pcie controller */ | ||
31 | void __init setup_pci_atmu(struct pci_controller *hose, struct resource *rsrc) | ||
32 | { | ||
33 | struct ccsr_pci __iomem *pci; | ||
34 | int i; | ||
35 | |||
36 | pr_debug("PCI memory map start 0x%x, size 0x%x\n", rsrc->start, | ||
37 | rsrc->end - rsrc->start + 1); | ||
38 | pci = ioremap(rsrc->start, rsrc->end - rsrc->start + 1); | ||
39 | |||
40 | /* Disable all windows (except powar0 since its ignored) */ | ||
41 | for(i = 1; i < 5; i++) | ||
42 | out_be32(&pci->pow[i].powar, 0); | ||
43 | for(i = 0; i < 3; i++) | ||
44 | out_be32(&pci->piw[i].piwar, 0); | ||
45 | |||
46 | /* Setup outbound MEM window */ | ||
47 | for(i = 0; i < 3; i++) | ||
48 | if (hose->mem_resources[i].flags & IORESOURCE_MEM){ | ||
49 | pr_debug("PCI MEM resource start 0x%08x, size 0x%08x.\n", | ||
50 | hose->mem_resources[i].start, | ||
51 | hose->mem_resources[i].end | ||
52 | - hose->mem_resources[i].start + 1); | ||
53 | out_be32(&pci->pow[i+1].potar, | ||
54 | (hose->mem_resources[i].start >> 12) | ||
55 | & 0x000fffff); | ||
56 | out_be32(&pci->pow[i+1].potear, 0); | ||
57 | out_be32(&pci->pow[i+1].powbar, | ||
58 | (hose->mem_resources[i].start >> 12) | ||
59 | & 0x000fffff); | ||
60 | /* Enable, Mem R/W */ | ||
61 | out_be32(&pci->pow[i+1].powar, 0x80044000 | ||
62 | | (__ilog2(hose->mem_resources[i].end | ||
63 | - hose->mem_resources[i].start + 1) - 1)); | ||
64 | } | ||
65 | |||
66 | /* Setup outbound IO window */ | ||
67 | if (hose->io_resource.flags & IORESOURCE_IO){ | ||
68 | pr_debug("PCI IO resource start 0x%08x, size 0x%08x, phy base 0x%08x.\n", | ||
69 | hose->io_resource.start, | ||
70 | hose->io_resource.end - hose->io_resource.start + 1, | ||
71 | hose->io_base_phys); | ||
72 | out_be32(&pci->pow[i+1].potar, (hose->io_resource.start >> 12) | ||
73 | & 0x000fffff); | ||
74 | out_be32(&pci->pow[i+1].potear, 0); | ||
75 | out_be32(&pci->pow[i+1].powbar, (hose->io_base_phys >> 12) | ||
76 | & 0x000fffff); | ||
77 | /* Enable, IO R/W */ | ||
78 | out_be32(&pci->pow[i+1].powar, 0x80088000 | ||
79 | | (__ilog2(hose->io_resource.end | ||
80 | - hose->io_resource.start + 1) - 1)); | ||
81 | } | ||
82 | |||
83 | /* Setup 2G inbound Memory Window @ 1 */ | ||
84 | out_be32(&pci->piw[2].pitar, 0x00000000); | ||
85 | out_be32(&pci->piw[2].piwbar,0x00000000); | ||
86 | out_be32(&pci->piw[2].piwar, PIWAR_2G); | ||
87 | } | ||
88 | |||
89 | void __init setup_pci_cmd(struct pci_controller *hose) | ||
90 | { | ||
91 | u16 cmd; | ||
92 | int cap_x; | ||
93 | |||
94 | early_read_config_word(hose, 0, 0, PCI_COMMAND, &cmd); | ||
95 | cmd |= PCI_COMMAND_SERR | PCI_COMMAND_MASTER | PCI_COMMAND_MEMORY | ||
96 | | PCI_COMMAND_IO; | ||
97 | early_write_config_word(hose, 0, 0, PCI_COMMAND, cmd); | ||
98 | |||
99 | cap_x = early_find_capability(hose, 0, 0, PCI_CAP_ID_PCIX); | ||
100 | if (cap_x) { | ||
101 | int pci_x_cmd = cap_x + PCI_X_CMD; | ||
102 | cmd = PCI_X_CMD_MAX_SPLIT | PCI_X_CMD_MAX_READ | ||
103 | | PCI_X_CMD_ERO | PCI_X_CMD_DPERR_E; | ||
104 | early_write_config_word(hose, 0, 0, pci_x_cmd, cmd); | ||
105 | } else { | ||
106 | early_write_config_byte(hose, 0, 0, PCI_LATENCY_TIMER, 0x80); | ||
107 | } | ||
108 | } | ||
109 | |||
110 | static void __init quirk_fsl_pcie_transparent(struct pci_dev *dev) | ||
111 | { | ||
112 | struct resource *res; | ||
113 | int i, res_idx = PCI_BRIDGE_RESOURCES; | ||
114 | struct pci_controller *hose; | ||
115 | |||
116 | /* if we aren't a PCIe don't bother */ | ||
117 | if (!pci_find_capability(dev, PCI_CAP_ID_EXP)) | ||
118 | return ; | ||
119 | |||
120 | /* | ||
121 | * Make the bridge be transparent. | ||
122 | */ | ||
123 | dev->transparent = 1; | ||
124 | |||
125 | hose = pci_bus_to_host(dev->bus); | ||
126 | if (!hose) { | ||
127 | printk(KERN_ERR "Can't find hose for bus %d\n", | ||
128 | dev->bus->number); | ||
129 | return; | ||
130 | } | ||
131 | |||
132 | /* Clear out any of the virtual P2P bridge registers */ | ||
133 | pci_write_config_word(dev, PCI_IO_BASE_UPPER16, 0); | ||
134 | pci_write_config_word(dev, PCI_IO_LIMIT_UPPER16, 0); | ||
135 | pci_write_config_byte(dev, PCI_IO_BASE, 0x10); | ||
136 | pci_write_config_byte(dev, PCI_IO_LIMIT, 0); | ||
137 | pci_write_config_word(dev, PCI_MEMORY_BASE, 0x10); | ||
138 | pci_write_config_word(dev, PCI_MEMORY_LIMIT, 0); | ||
139 | pci_write_config_word(dev, PCI_PREF_BASE_UPPER32, 0x0); | ||
140 | pci_write_config_word(dev, PCI_PREF_LIMIT_UPPER32, 0x0); | ||
141 | pci_write_config_word(dev, PCI_PREF_MEMORY_BASE, 0x10); | ||
142 | pci_write_config_word(dev, PCI_PREF_MEMORY_LIMIT, 0); | ||
143 | |||
144 | if (hose->io_resource.flags) { | ||
145 | res = &dev->resource[res_idx++]; | ||
146 | res->start = hose->io_resource.start; | ||
147 | res->end = hose->io_resource.end; | ||
148 | res->flags = hose->io_resource.flags; | ||
149 | update_bridge_resource(dev, res); | ||
150 | } | ||
151 | |||
152 | for (i = 0; i < 3; i++) { | ||
153 | res = &dev->resource[res_idx + i]; | ||
154 | res->start = hose->mem_resources[i].start; | ||
155 | res->end = hose->mem_resources[i].end; | ||
156 | res->flags = hose->mem_resources[i].flags; | ||
157 | update_bridge_resource(dev, res); | ||
158 | } | ||
159 | } | ||
160 | |||
161 | int __init fsl_pcie_check_link(struct pci_controller *hose) | ||
162 | { | ||
163 | u16 val; | ||
164 | early_read_config_word(hose, 0, 0, PCIE_LTSSM, &val); | ||
165 | if (val < PCIE_LTSSM_L0) | ||
166 | return 1; | ||
167 | return 0; | ||
168 | } | ||
169 | |||
170 | void fsl_pcibios_fixup_bus(struct pci_bus *bus) | ||
171 | { | ||
172 | struct pci_controller *hose = (struct pci_controller *) bus->sysdata; | ||
173 | int i; | ||
174 | |||
175 | /* deal with bogus pci_bus when we don't have anything connected on PCIe */ | ||
176 | if (hose->indirect_type & PPC_INDIRECT_TYPE_NO_PCIE_LINK) { | ||
177 | if (bus->parent) { | ||
178 | for (i = 0; i < 4; ++i) | ||
179 | bus->resource[i] = bus->parent->resource[i]; | ||
180 | } | ||
181 | } | ||
182 | } | ||
183 | |||
184 | int __init fsl_add_bridge(struct device_node *dev, int is_primary) | ||
185 | { | ||
186 | int len; | ||
187 | struct pci_controller *hose; | ||
188 | struct resource rsrc; | ||
189 | const int *bus_range; | ||
190 | |||
191 | pr_debug("Adding PCI host bridge %s\n", dev->full_name); | ||
192 | |||
193 | /* Fetch host bridge registers address */ | ||
194 | if (of_address_to_resource(dev, 0, &rsrc)) { | ||
195 | printk(KERN_WARNING "Can't get pci register base!"); | ||
196 | return -ENOMEM; | ||
197 | } | ||
198 | |||
199 | /* Get bus range if any */ | ||
200 | bus_range = of_get_property(dev, "bus-range", &len); | ||
201 | if (bus_range == NULL || len < 2 * sizeof(int)) | ||
202 | printk(KERN_WARNING "Can't get bus-range for %s, assume" | ||
203 | " bus 0\n", dev->full_name); | ||
204 | |||
205 | pci_assign_all_buses = 1; | ||
206 | hose = pcibios_alloc_controller(dev); | ||
207 | if (!hose) | ||
208 | return -ENOMEM; | ||
209 | |||
210 | hose->first_busno = bus_range ? bus_range[0] : 0x0; | ||
211 | hose->last_busno = bus_range ? bus_range[1] : 0xff; | ||
212 | |||
213 | setup_indirect_pci(hose, rsrc.start, rsrc.start + 0x4, | ||
214 | PPC_INDIRECT_TYPE_BIG_ENDIAN); | ||
215 | setup_pci_cmd(hose); | ||
216 | |||
217 | /* check PCI express link status */ | ||
218 | if (early_find_capability(hose, 0, 0, PCI_CAP_ID_EXP)) { | ||
219 | hose->indirect_type |= PPC_INDIRECT_TYPE_EXT_REG | | ||
220 | PPC_INDIRECT_TYPE_SURPRESS_PRIMARY_BUS; | ||
221 | if (fsl_pcie_check_link(hose)) | ||
222 | hose->indirect_type |= PPC_INDIRECT_TYPE_NO_PCIE_LINK; | ||
223 | } | ||
224 | |||
225 | printk(KERN_INFO "Found FSL PCI host bridge at 0x%016llx." | ||
226 | "Firmware bus number: %d->%d\n", | ||
227 | (unsigned long long)rsrc.start, hose->first_busno, | ||
228 | hose->last_busno); | ||
229 | |||
230 | pr_debug(" ->Hose at 0x%p, cfg_addr=0x%p,cfg_data=0x%p\n", | ||
231 | hose, hose->cfg_addr, hose->cfg_data); | ||
232 | |||
233 | /* Interpret the "ranges" property */ | ||
234 | /* This also maps the I/O region and sets isa_io/mem_base */ | ||
235 | pci_process_bridge_OF_ranges(hose, dev, is_primary); | ||
236 | |||
237 | /* Setup PEX window registers */ | ||
238 | setup_pci_atmu(hose, &rsrc); | ||
239 | |||
240 | return 0; | ||
241 | } | ||
242 | |||
243 | DECLARE_PCI_FIXUP_EARLY(0x1957, PCI_DEVICE_ID_MPC8548E, quirk_fsl_pcie_transparent); | ||
244 | DECLARE_PCI_FIXUP_EARLY(0x1957, PCI_DEVICE_ID_MPC8548, quirk_fsl_pcie_transparent); | ||
245 | DECLARE_PCI_FIXUP_EARLY(0x1957, PCI_DEVICE_ID_MPC8543E, quirk_fsl_pcie_transparent); | ||
246 | DECLARE_PCI_FIXUP_EARLY(0x1957, PCI_DEVICE_ID_MPC8543, quirk_fsl_pcie_transparent); | ||
247 | DECLARE_PCI_FIXUP_EARLY(0x1957, PCI_DEVICE_ID_MPC8547E, quirk_fsl_pcie_transparent); | ||
248 | DECLARE_PCI_FIXUP_EARLY(0x1957, PCI_DEVICE_ID_MPC8545E, quirk_fsl_pcie_transparent); | ||
249 | DECLARE_PCI_FIXUP_EARLY(0x1957, PCI_DEVICE_ID_MPC8545, quirk_fsl_pcie_transparent); | ||
250 | DECLARE_PCI_FIXUP_EARLY(0x1957, PCI_DEVICE_ID_MPC8568E, quirk_fsl_pcie_transparent); | ||
251 | DECLARE_PCI_FIXUP_EARLY(0x1957, PCI_DEVICE_ID_MPC8568, quirk_fsl_pcie_transparent); | ||
252 | DECLARE_PCI_FIXUP_EARLY(0x1957, PCI_DEVICE_ID_MPC8567E, quirk_fsl_pcie_transparent); | ||
253 | DECLARE_PCI_FIXUP_EARLY(0x1957, PCI_DEVICE_ID_MPC8567, quirk_fsl_pcie_transparent); | ||
254 | DECLARE_PCI_FIXUP_EARLY(0x1957, PCI_DEVICE_ID_MPC8544E, quirk_fsl_pcie_transparent); | ||
255 | DECLARE_PCI_FIXUP_EARLY(0x1957, PCI_DEVICE_ID_MPC8544, quirk_fsl_pcie_transparent); | ||
256 | DECLARE_PCI_FIXUP_EARLY(0x1957, PCI_DEVICE_ID_MPC8641, quirk_fsl_pcie_transparent); | ||
257 | DECLARE_PCI_FIXUP_EARLY(0x1957, PCI_DEVICE_ID_MPC8641D, quirk_fsl_pcie_transparent); | ||
diff --git a/arch/powerpc/sysdev/fsl_pci.h b/arch/powerpc/sysdev/fsl_pci.h new file mode 100644 index 000000000000..37b04ad26571 --- /dev/null +++ b/arch/powerpc/sysdev/fsl_pci.h | |||
@@ -0,0 +1,88 @@ | |||
1 | /* | ||
2 | * MPC85xx/86xx PCI Express structure define | ||
3 | * | ||
4 | * Copyright 2007 Freescale Semiconductor, Inc | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify it | ||
7 | * under the terms of the GNU General Public License as published by the | ||
8 | * Free Software Foundation; either version 2 of the License, or (at your | ||
9 | * option) any later version. | ||
10 | * | ||
11 | */ | ||
12 | |||
13 | #ifdef __KERNEL__ | ||
14 | #ifndef __POWERPC_FSL_PCI_H | ||
15 | #define __POWERPC_FSL_PCI_H | ||
16 | |||
17 | #define PCIE_LTSSM 0x0404 /* PCIE Link Training and Status */ | ||
18 | #define PCIE_LTSSM_L0 0x16 /* L0 state */ | ||
19 | #define PIWAR_2G 0xa0f5501e /* Enable, Prefetch, Local Mem, Snoop R/W, 2G */ | ||
20 | |||
21 | /* PCI/PCI Express outbound window reg */ | ||
22 | struct pci_outbound_window_regs { | ||
23 | __be32 potar; /* 0x.0 - Outbound translation address register */ | ||
24 | __be32 potear; /* 0x.4 - Outbound translation extended address register */ | ||
25 | __be32 powbar; /* 0x.8 - Outbound window base address register */ | ||
26 | u8 res1[4]; | ||
27 | __be32 powar; /* 0x.10 - Outbound window attributes register */ | ||
28 | u8 res2[12]; | ||
29 | }; | ||
30 | |||
31 | /* PCI/PCI Express inbound window reg */ | ||
32 | struct pci_inbound_window_regs { | ||
33 | __be32 pitar; /* 0x.0 - Inbound translation address register */ | ||
34 | u8 res1[4]; | ||
35 | __be32 piwbar; /* 0x.8 - Inbound window base address register */ | ||
36 | __be32 piwbear; /* 0x.c - Inbound window base extended address register */ | ||
37 | __be32 piwar; /* 0x.10 - Inbound window attributes register */ | ||
38 | u8 res2[12]; | ||
39 | }; | ||
40 | |||
41 | /* PCI/PCI Express IO block registers for 85xx/86xx */ | ||
42 | struct ccsr_pci { | ||
43 | __be32 config_addr; /* 0x.000 - PCI/PCIE Configuration Address Register */ | ||
44 | __be32 config_data; /* 0x.004 - PCI/PCIE Configuration Data Register */ | ||
45 | __be32 int_ack; /* 0x.008 - PCI Interrupt Acknowledge Register */ | ||
46 | __be32 pex_otb_cpl_tor; /* 0x.00c - PCIE Outbound completion timeout register */ | ||
47 | __be32 pex_conf_tor; /* 0x.010 - PCIE configuration timeout register */ | ||
48 | u8 res2[12]; | ||
49 | __be32 pex_pme_mes_dr; /* 0x.020 - PCIE PME and message detect register */ | ||
50 | __be32 pex_pme_mes_disr; /* 0x.024 - PCIE PME and message disable register */ | ||
51 | __be32 pex_pme_mes_ier; /* 0x.028 - PCIE PME and message interrupt enable register */ | ||
52 | __be32 pex_pmcr; /* 0x.02c - PCIE power management command register */ | ||
53 | u8 res3[3024]; | ||
54 | |||
55 | /* PCI/PCI Express outbound window 0-4 | ||
56 | * Window 0 is the default window and is the only window enabled upon reset. | ||
57 | * The default outbound register set is used when a transaction misses | ||
58 | * in all of the other outbound windows. | ||
59 | */ | ||
60 | struct pci_outbound_window_regs pow[5]; | ||
61 | |||
62 | u8 res14[256]; | ||
63 | |||
64 | /* PCI/PCI Express inbound window 3-1 | ||
65 | * inbound window 1 supports only a 32-bit base address and does not | ||
66 | * define an inbound window base extended address register. | ||
67 | */ | ||
68 | struct pci_inbound_window_regs piw[3]; | ||
69 | |||
70 | __be32 pex_err_dr; /* 0x.e00 - PCI/PCIE error detect register */ | ||
71 | u8 res21[4]; | ||
72 | __be32 pex_err_en; /* 0x.e08 - PCI/PCIE error interrupt enable register */ | ||
73 | u8 res22[4]; | ||
74 | __be32 pex_err_disr; /* 0x.e10 - PCI/PCIE error disable register */ | ||
75 | u8 res23[12]; | ||
76 | __be32 pex_err_cap_stat; /* 0x.e20 - PCI/PCIE error capture status register */ | ||
77 | u8 res24[4]; | ||
78 | __be32 pex_err_cap_r0; /* 0x.e28 - PCIE error capture register 0 */ | ||
79 | __be32 pex_err_cap_r1; /* 0x.e2c - PCIE error capture register 0 */ | ||
80 | __be32 pex_err_cap_r2; /* 0x.e30 - PCIE error capture register 0 */ | ||
81 | __be32 pex_err_cap_r3; /* 0x.e34 - PCIE error capture register 0 */ | ||
82 | }; | ||
83 | |||
84 | extern int fsl_add_bridge(struct device_node *dev, int is_primary); | ||
85 | extern void fsl_pcibios_fixup_bus(struct pci_bus *bus); | ||
86 | |||
87 | #endif /* __POWERPC_FSL_PCI_H */ | ||
88 | #endif /* __KERNEL__ */ | ||
diff --git a/arch/powerpc/sysdev/fsl_pcie.h b/arch/powerpc/sysdev/fsl_pcie.h deleted file mode 100644 index 8d9779c84bea..000000000000 --- a/arch/powerpc/sysdev/fsl_pcie.h +++ /dev/null | |||
@@ -1,94 +0,0 @@ | |||
1 | /* | ||
2 | * MPC85xx/86xx PCI Express structure define | ||
3 | * | ||
4 | * Copyright 2007 Freescale Semiconductor, Inc | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify it | ||
7 | * under the terms of the GNU General Public License as published by the | ||
8 | * Free Software Foundation; either version 2 of the License, or (at your | ||
9 | * option) any later version. | ||
10 | * | ||
11 | */ | ||
12 | |||
13 | #ifdef __KERNEL__ | ||
14 | #ifndef __POWERPC_FSL_PCIE_H | ||
15 | #define __POWERPC_FSL_PCIE_H | ||
16 | |||
17 | /* PCIE Express IO block registers in 85xx/86xx */ | ||
18 | |||
19 | struct ccsr_pex { | ||
20 | __be32 __iomem pex_config_addr; /* 0x.000 - PCI Express Configuration Address Register */ | ||
21 | __be32 __iomem pex_config_data; /* 0x.004 - PCI Express Configuration Data Register */ | ||
22 | u8 __iomem res1[4]; | ||
23 | __be32 __iomem pex_otb_cpl_tor; /* 0x.00c - PCI Express Outbound completion timeout register */ | ||
24 | __be32 __iomem pex_conf_tor; /* 0x.010 - PCI Express configuration timeout register */ | ||
25 | u8 __iomem res2[12]; | ||
26 | __be32 __iomem pex_pme_mes_dr; /* 0x.020 - PCI Express PME and message detect register */ | ||
27 | __be32 __iomem pex_pme_mes_disr; /* 0x.024 - PCI Express PME and message disable register */ | ||
28 | __be32 __iomem pex_pme_mes_ier; /* 0x.028 - PCI Express PME and message interrupt enable register */ | ||
29 | __be32 __iomem pex_pmcr; /* 0x.02c - PCI Express power management command register */ | ||
30 | u8 __iomem res3[3024]; | ||
31 | __be32 __iomem pexotar0; /* 0x.c00 - PCI Express outbound translation address register 0 */ | ||
32 | __be32 __iomem pexotear0; /* 0x.c04 - PCI Express outbound translation extended address register 0*/ | ||
33 | u8 __iomem res4[8]; | ||
34 | __be32 __iomem pexowar0; /* 0x.c10 - PCI Express outbound window attributes register 0*/ | ||
35 | u8 __iomem res5[12]; | ||
36 | __be32 __iomem pexotar1; /* 0x.c20 - PCI Express outbound translation address register 1 */ | ||
37 | __be32 __iomem pexotear1; /* 0x.c24 - PCI Express outbound translation extended address register 1*/ | ||
38 | __be32 __iomem pexowbar1; /* 0x.c28 - PCI Express outbound window base address register 1*/ | ||
39 | u8 __iomem res6[4]; | ||
40 | __be32 __iomem pexowar1; /* 0x.c30 - PCI Express outbound window attributes register 1*/ | ||
41 | u8 __iomem res7[12]; | ||
42 | __be32 __iomem pexotar2; /* 0x.c40 - PCI Express outbound translation address register 2 */ | ||
43 | __be32 __iomem pexotear2; /* 0x.c44 - PCI Express outbound translation extended address register 2*/ | ||
44 | __be32 __iomem pexowbar2; /* 0x.c48 - PCI Express outbound window base address register 2*/ | ||
45 | u8 __iomem res8[4]; | ||
46 | __be32 __iomem pexowar2; /* 0x.c50 - PCI Express outbound window attributes register 2*/ | ||
47 | u8 __iomem res9[12]; | ||
48 | __be32 __iomem pexotar3; /* 0x.c60 - PCI Express outbound translation address register 3 */ | ||
49 | __be32 __iomem pexotear3; /* 0x.c64 - PCI Express outbound translation extended address register 3*/ | ||
50 | __be32 __iomem pexowbar3; /* 0x.c68 - PCI Express outbound window base address register 3*/ | ||
51 | u8 __iomem res10[4]; | ||
52 | __be32 __iomem pexowar3; /* 0x.c70 - PCI Express outbound window attributes register 3*/ | ||
53 | u8 __iomem res11[12]; | ||
54 | __be32 __iomem pexotar4; /* 0x.c80 - PCI Express outbound translation address register 4 */ | ||
55 | __be32 __iomem pexotear4; /* 0x.c84 - PCI Express outbound translation extended address register 4*/ | ||
56 | __be32 __iomem pexowbar4; /* 0x.c88 - PCI Express outbound window base address register 4*/ | ||
57 | u8 __iomem res12[4]; | ||
58 | __be32 __iomem pexowar4; /* 0x.c90 - PCI Express outbound window attributes register 4*/ | ||
59 | u8 __iomem res13[12]; | ||
60 | u8 __iomem res14[256]; | ||
61 | __be32 __iomem pexitar3; /* 0x.da0 - PCI Express inbound translation address register 3 */ | ||
62 | u8 __iomem res15[4]; | ||
63 | __be32 __iomem pexiwbar3; /* 0x.da8 - PCI Express inbound window base address register 3 */ | ||
64 | __be32 __iomem pexiwbear3; /* 0x.dac - PCI Express inbound window base extended address register 3 */ | ||
65 | __be32 __iomem pexiwar3; /* 0x.db0 - PCI Express inbound window attributes register 3 */ | ||
66 | u8 __iomem res16[12]; | ||
67 | __be32 __iomem pexitar2; /* 0x.dc0 - PCI Express inbound translation address register 2 */ | ||
68 | u8 __iomem res17[4]; | ||
69 | __be32 __iomem pexiwbar2; /* 0x.dc8 - PCI Express inbound window base address register 2 */ | ||
70 | __be32 __iomem pexiwbear2; /* 0x.dcc - PCI Express inbound window base extended address register 2 */ | ||
71 | __be32 __iomem pexiwar2; /* 0x.dd0 - PCI Express inbound window attributes register 2 */ | ||
72 | u8 __iomem res18[12]; | ||
73 | __be32 __iomem pexitar1; /* 0x.de0 - PCI Express inbound translation address register 2 */ | ||
74 | u8 __iomem res19[4]; | ||
75 | __be32 __iomem pexiwbar1; /* 0x.de8 - PCI Express inbound window base address register 2 */ | ||
76 | __be32 __iomem pexiwbear1; /* 0x.dec - PCI Express inbound window base extended address register 2 */ | ||
77 | __be32 __iomem pexiwar1; /* 0x.df0 - PCI Express inbound window attributes register 2 */ | ||
78 | u8 __iomem res20[12]; | ||
79 | __be32 __iomem pex_err_dr; /* 0x.e00 - PCI Express error detect register */ | ||
80 | u8 __iomem res21[4]; | ||
81 | __be32 __iomem pex_err_en; /* 0x.e08 - PCI Express error interrupt enable register */ | ||
82 | u8 __iomem res22[4]; | ||
83 | __be32 __iomem pex_err_disr; /* 0x.e10 - PCI Express error disable register */ | ||
84 | u8 __iomem res23[12]; | ||
85 | __be32 __iomem pex_err_cap_stat; /* 0x.e20 - PCI Express error capture status register */ | ||
86 | u8 __iomem res24[4]; | ||
87 | __be32 __iomem pex_err_cap_r0; /* 0x.e28 - PCI Express error capture register 0 */ | ||
88 | __be32 __iomem pex_err_cap_r1; /* 0x.e2c - PCI Express error capture register 0 */ | ||
89 | __be32 __iomem pex_err_cap_r2; /* 0x.e30 - PCI Express error capture register 0 */ | ||
90 | __be32 __iomem pex_err_cap_r3; /* 0x.e34 - PCI Express error capture register 0 */ | ||
91 | }; | ||
92 | |||
93 | #endif /* __POWERPC_FSL_PCIE_H */ | ||
94 | #endif /* __KERNEL__ */ | ||
diff --git a/arch/powerpc/sysdev/fsl_soc.c b/arch/powerpc/sysdev/fsl_soc.c index 3289fab01e92..727453d3e8b9 100644 --- a/arch/powerpc/sysdev/fsl_soc.c +++ b/arch/powerpc/sysdev/fsl_soc.c | |||
@@ -305,6 +305,64 @@ err: | |||
305 | 305 | ||
306 | arch_initcall(gfar_of_init); | 306 | arch_initcall(gfar_of_init); |
307 | 307 | ||
308 | #ifdef CONFIG_I2C_BOARDINFO | ||
309 | #include <linux/i2c.h> | ||
310 | struct i2c_driver_device { | ||
311 | char *of_device; | ||
312 | char *i2c_driver; | ||
313 | char *i2c_type; | ||
314 | }; | ||
315 | |||
316 | static struct i2c_driver_device i2c_devices[] __initdata = { | ||
317 | {"ricoh,rs5c372a", "rtc-rs5c372", "rs5c372a",}, | ||
318 | {"ricoh,rs5c372b", "rtc-rs5c372", "rs5c372b",}, | ||
319 | {"ricoh,rv5c386", "rtc-rs5c372", "rv5c386",}, | ||
320 | {"ricoh,rv5c387a", "rtc-rs5c372", "rv5c387a",}, | ||
321 | }; | ||
322 | |||
323 | static int __init of_find_i2c_driver(struct device_node *node, struct i2c_board_info *info) | ||
324 | { | ||
325 | int i; | ||
326 | |||
327 | for (i = 0; i < ARRAY_SIZE(i2c_devices); i++) { | ||
328 | if (!of_device_is_compatible(node, i2c_devices[i].of_device)) | ||
329 | continue; | ||
330 | strncpy(info->driver_name, i2c_devices[i].i2c_driver, KOBJ_NAME_LEN); | ||
331 | strncpy(info->type, i2c_devices[i].i2c_type, I2C_NAME_SIZE); | ||
332 | return 0; | ||
333 | } | ||
334 | return -ENODEV; | ||
335 | } | ||
336 | |||
337 | static void __init of_register_i2c_devices(struct device_node *adap_node, int bus_num) | ||
338 | { | ||
339 | struct device_node *node = NULL; | ||
340 | |||
341 | while ((node = of_get_next_child(adap_node, node))) { | ||
342 | struct i2c_board_info info; | ||
343 | const u32 *addr; | ||
344 | int len; | ||
345 | |||
346 | addr = of_get_property(node, "reg", &len); | ||
347 | if (!addr || len < sizeof(int) || *addr > (1 << 10) - 1) { | ||
348 | printk(KERN_WARNING "fsl_ioc.c: invalid i2c device entry\n"); | ||
349 | continue; | ||
350 | } | ||
351 | |||
352 | info.irq = irq_of_parse_and_map(node, 0); | ||
353 | if (info.irq == NO_IRQ) | ||
354 | info.irq = -1; | ||
355 | |||
356 | if (of_find_i2c_driver(node, &info) < 0) | ||
357 | continue; | ||
358 | |||
359 | info.platform_data = NULL; | ||
360 | info.addr = *addr; | ||
361 | |||
362 | i2c_register_board_info(bus_num, &info, 1); | ||
363 | } | ||
364 | } | ||
365 | |||
308 | static int __init fsl_i2c_of_init(void) | 366 | static int __init fsl_i2c_of_init(void) |
309 | { | 367 | { |
310 | struct device_node *np; | 368 | struct device_node *np; |
@@ -349,6 +407,8 @@ static int __init fsl_i2c_of_init(void) | |||
349 | fsl_i2c_platform_data)); | 407 | fsl_i2c_platform_data)); |
350 | if (ret) | 408 | if (ret) |
351 | goto unreg; | 409 | goto unreg; |
410 | |||
411 | of_register_i2c_devices(np, i); | ||
352 | } | 412 | } |
353 | 413 | ||
354 | return 0; | 414 | return 0; |
@@ -360,6 +420,7 @@ err: | |||
360 | } | 420 | } |
361 | 421 | ||
362 | arch_initcall(fsl_i2c_of_init); | 422 | arch_initcall(fsl_i2c_of_init); |
423 | #endif | ||
363 | 424 | ||
364 | #ifdef CONFIG_PPC_83xx | 425 | #ifdef CONFIG_PPC_83xx |
365 | static int __init mpc83xx_wdt_init(void) | 426 | static int __init mpc83xx_wdt_init(void) |
diff --git a/arch/powerpc/sysdev/grackle.c b/arch/powerpc/sysdev/grackle.c index 42053625f498..11ad5622eb76 100644 --- a/arch/powerpc/sysdev/grackle.c +++ b/arch/powerpc/sysdev/grackle.c | |||
@@ -55,7 +55,7 @@ static inline void grackle_set_loop_snoop(struct pci_controller *bp, int enable) | |||
55 | 55 | ||
56 | void __init setup_grackle(struct pci_controller *hose) | 56 | void __init setup_grackle(struct pci_controller *hose) |
57 | { | 57 | { |
58 | setup_indirect_pci(hose, 0xfec00000, 0xfee00000); | 58 | setup_indirect_pci(hose, 0xfec00000, 0xfee00000, 0); |
59 | if (machine_is_compatible("PowerMac1,1")) | 59 | if (machine_is_compatible("PowerMac1,1")) |
60 | pci_assign_all_buses = 1; | 60 | pci_assign_all_buses = 1; |
61 | if (machine_is_compatible("AAPL,PowerBook1998")) | 61 | if (machine_is_compatible("AAPL,PowerBook1998")) |
diff --git a/arch/powerpc/sysdev/indirect_pci.c b/arch/powerpc/sysdev/indirect_pci.c index c7e6e859b393..5294560c7b00 100644 --- a/arch/powerpc/sysdev/indirect_pci.c +++ b/arch/powerpc/sysdev/indirect_pci.c | |||
@@ -20,12 +20,6 @@ | |||
20 | #include <asm/pci-bridge.h> | 20 | #include <asm/pci-bridge.h> |
21 | #include <asm/machdep.h> | 21 | #include <asm/machdep.h> |
22 | 22 | ||
23 | #ifdef CONFIG_PPC_INDIRECT_PCI_BE | ||
24 | #define PCI_CFG_OUT out_be32 | ||
25 | #else | ||
26 | #define PCI_CFG_OUT out_le32 | ||
27 | #endif | ||
28 | |||
29 | static int | 23 | static int |
30 | indirect_read_config(struct pci_bus *bus, unsigned int devfn, int offset, | 24 | indirect_read_config(struct pci_bus *bus, unsigned int devfn, int offset, |
31 | int len, u32 *val) | 25 | int len, u32 *val) |
@@ -35,10 +29,17 @@ indirect_read_config(struct pci_bus *bus, unsigned int devfn, int offset, | |||
35 | u8 cfg_type = 0; | 29 | u8 cfg_type = 0; |
36 | u32 bus_no, reg; | 30 | u32 bus_no, reg; |
37 | 31 | ||
32 | if (hose->indirect_type & PPC_INDIRECT_TYPE_NO_PCIE_LINK) { | ||
33 | if (bus->number != hose->first_busno) | ||
34 | return PCIBIOS_DEVICE_NOT_FOUND; | ||
35 | if (devfn != 0) | ||
36 | return PCIBIOS_DEVICE_NOT_FOUND; | ||
37 | } | ||
38 | |||
38 | if (ppc_md.pci_exclude_device) | 39 | if (ppc_md.pci_exclude_device) |
39 | if (ppc_md.pci_exclude_device(hose, bus->number, devfn)) | 40 | if (ppc_md.pci_exclude_device(hose, bus->number, devfn)) |
40 | return PCIBIOS_DEVICE_NOT_FOUND; | 41 | return PCIBIOS_DEVICE_NOT_FOUND; |
41 | 42 | ||
42 | if (hose->indirect_type & PPC_INDIRECT_TYPE_SET_CFG_TYPE) | 43 | if (hose->indirect_type & PPC_INDIRECT_TYPE_SET_CFG_TYPE) |
43 | if (bus->number != hose->first_busno) | 44 | if (bus->number != hose->first_busno) |
44 | cfg_type = 1; | 45 | cfg_type = 1; |
@@ -51,9 +52,12 @@ indirect_read_config(struct pci_bus *bus, unsigned int devfn, int offset, | |||
51 | else | 52 | else |
52 | reg = offset & 0xfc; | 53 | reg = offset & 0xfc; |
53 | 54 | ||
54 | PCI_CFG_OUT(hose->cfg_addr, | 55 | if (hose->indirect_type & PPC_INDIRECT_TYPE_BIG_ENDIAN) |
55 | (0x80000000 | (bus_no << 16) | 56 | out_be32(hose->cfg_addr, (0x80000000 | (bus_no << 16) | |
56 | | (devfn << 8) | reg | cfg_type)); | 57 | (devfn << 8) | reg | cfg_type)); |
58 | else | ||
59 | out_le32(hose->cfg_addr, (0x80000000 | (bus_no << 16) | | ||
60 | (devfn << 8) | reg | cfg_type)); | ||
57 | 61 | ||
58 | /* | 62 | /* |
59 | * Note: the caller has already checked that offset is | 63 | * Note: the caller has already checked that offset is |
@@ -83,6 +87,13 @@ indirect_write_config(struct pci_bus *bus, unsigned int devfn, int offset, | |||
83 | u8 cfg_type = 0; | 87 | u8 cfg_type = 0; |
84 | u32 bus_no, reg; | 88 | u32 bus_no, reg; |
85 | 89 | ||
90 | if (hose->indirect_type & PPC_INDIRECT_TYPE_NO_PCIE_LINK) { | ||
91 | if (bus->number != hose->first_busno) | ||
92 | return PCIBIOS_DEVICE_NOT_FOUND; | ||
93 | if (devfn != 0) | ||
94 | return PCIBIOS_DEVICE_NOT_FOUND; | ||
95 | } | ||
96 | |||
86 | if (ppc_md.pci_exclude_device) | 97 | if (ppc_md.pci_exclude_device) |
87 | if (ppc_md.pci_exclude_device(hose, bus->number, devfn)) | 98 | if (ppc_md.pci_exclude_device(hose, bus->number, devfn)) |
88 | return PCIBIOS_DEVICE_NOT_FOUND; | 99 | return PCIBIOS_DEVICE_NOT_FOUND; |
@@ -99,9 +110,12 @@ indirect_write_config(struct pci_bus *bus, unsigned int devfn, int offset, | |||
99 | else | 110 | else |
100 | reg = offset & 0xfc; | 111 | reg = offset & 0xfc; |
101 | 112 | ||
102 | PCI_CFG_OUT(hose->cfg_addr, | 113 | if (hose->indirect_type & PPC_INDIRECT_TYPE_BIG_ENDIAN) |
103 | (0x80000000 | (bus_no << 16) | 114 | out_be32(hose->cfg_addr, (0x80000000 | (bus_no << 16) | |
104 | | (devfn << 8) | reg | cfg_type)); | 115 | (devfn << 8) | reg | cfg_type)); |
116 | else | ||
117 | out_le32(hose->cfg_addr, (0x80000000 | (bus_no << 16) | | ||
118 | (devfn << 8) | reg | cfg_type)); | ||
105 | 119 | ||
106 | /* surpress setting of PCI_PRIMARY_BUS */ | 120 | /* surpress setting of PCI_PRIMARY_BUS */ |
107 | if (hose->indirect_type & PPC_INDIRECT_TYPE_SURPRESS_PRIMARY_BUS) | 121 | if (hose->indirect_type & PPC_INDIRECT_TYPE_SURPRESS_PRIMARY_BUS) |
@@ -135,24 +149,16 @@ static struct pci_ops indirect_pci_ops = | |||
135 | }; | 149 | }; |
136 | 150 | ||
137 | void __init | 151 | void __init |
138 | setup_indirect_pci_nomap(struct pci_controller* hose, void __iomem * cfg_addr, | 152 | setup_indirect_pci(struct pci_controller* hose, u32 cfg_addr, u32 cfg_data, u32 flags) |
139 | void __iomem * cfg_data) | ||
140 | { | ||
141 | hose->cfg_addr = cfg_addr; | ||
142 | hose->cfg_data = cfg_data; | ||
143 | hose->ops = &indirect_pci_ops; | ||
144 | } | ||
145 | |||
146 | void __init | ||
147 | setup_indirect_pci(struct pci_controller* hose, u32 cfg_addr, u32 cfg_data) | ||
148 | { | 153 | { |
149 | unsigned long base = cfg_addr & PAGE_MASK; | 154 | unsigned long base = cfg_addr & PAGE_MASK; |
150 | void __iomem *mbase, *addr, *data; | 155 | void __iomem *mbase; |
151 | 156 | ||
152 | mbase = ioremap(base, PAGE_SIZE); | 157 | mbase = ioremap(base, PAGE_SIZE); |
153 | addr = mbase + (cfg_addr & ~PAGE_MASK); | 158 | hose->cfg_addr = mbase + (cfg_addr & ~PAGE_MASK); |
154 | if ((cfg_data & PAGE_MASK) != base) | 159 | if ((cfg_data & PAGE_MASK) != base) |
155 | mbase = ioremap(cfg_data & PAGE_MASK, PAGE_SIZE); | 160 | mbase = ioremap(cfg_data & PAGE_MASK, PAGE_SIZE); |
156 | data = mbase + (cfg_data & ~PAGE_MASK); | 161 | hose->cfg_data = mbase + (cfg_data & ~PAGE_MASK); |
157 | setup_indirect_pci_nomap(hose, addr, data); | 162 | hose->ops = &indirect_pci_ops; |
163 | hose->indirect_type = flags; | ||
158 | } | 164 | } |
diff --git a/arch/powerpc/sysdev/mv64x60_pci.c b/arch/powerpc/sysdev/mv64x60_pci.c index 45db86c2363c..9b3baa7317d7 100644 --- a/arch/powerpc/sysdev/mv64x60_pci.c +++ b/arch/powerpc/sysdev/mv64x60_pci.c | |||
@@ -144,7 +144,7 @@ static int __init mv64x60_add_bridge(struct device_node *dev) | |||
144 | hose->first_busno = bus_range ? bus_range[0] : 0; | 144 | hose->first_busno = bus_range ? bus_range[0] : 0; |
145 | hose->last_busno = bus_range ? bus_range[1] : 0xff; | 145 | hose->last_busno = bus_range ? bus_range[1] : 0xff; |
146 | 146 | ||
147 | setup_indirect_pci(hose, rsrc.start, rsrc.start + 4); | 147 | setup_indirect_pci(hose, rsrc.start, rsrc.start + 4, 0); |
148 | hose->self_busno = hose->first_busno; | 148 | hose->self_busno = hose->first_busno; |
149 | 149 | ||
150 | printk(KERN_INFO "Found MV64x60 PCI host bridge at 0x%016llx. " | 150 | printk(KERN_INFO "Found MV64x60 PCI host bridge at 0x%016llx. " |
diff --git a/arch/sparc/defconfig b/arch/sparc/defconfig index 38bd79fe6e75..fdc67238408a 100644 --- a/arch/sparc/defconfig +++ b/arch/sparc/defconfig | |||
@@ -600,7 +600,7 @@ CONFIG_LEGACY_PTY_COUNT=256 | |||
600 | # CONFIG_IPMI_HANDLER is not set | 600 | # CONFIG_IPMI_HANDLER is not set |
601 | # CONFIG_WATCHDOG is not set | 601 | # CONFIG_WATCHDOG is not set |
602 | CONFIG_HW_RANDOM=m | 602 | CONFIG_HW_RANDOM=m |
603 | CONFIG_RTC=m | 603 | CONFIG_JS_RTC=m |
604 | # CONFIG_R3964 is not set | 604 | # CONFIG_R3964 is not set |
605 | # CONFIG_APPLICOM is not set | 605 | # CONFIG_APPLICOM is not set |
606 | # CONFIG_DRM is not set | 606 | # CONFIG_DRM is not set |
diff --git a/arch/sparc/kernel/sparc_ksyms.c b/arch/sparc/kernel/sparc_ksyms.c index 55bac516dfe2..7b4abde43028 100644 --- a/arch/sparc/kernel/sparc_ksyms.c +++ b/arch/sparc/kernel/sparc_ksyms.c | |||
@@ -161,6 +161,8 @@ EXPORT_SYMBOL(BTFIXUP_CALL(mmu_get_scsi_one)); | |||
161 | EXPORT_SYMBOL(BTFIXUP_CALL(mmu_release_scsi_sgl)); | 161 | EXPORT_SYMBOL(BTFIXUP_CALL(mmu_release_scsi_sgl)); |
162 | EXPORT_SYMBOL(BTFIXUP_CALL(mmu_release_scsi_one)); | 162 | EXPORT_SYMBOL(BTFIXUP_CALL(mmu_release_scsi_one)); |
163 | 163 | ||
164 | EXPORT_SYMBOL(BTFIXUP_CALL(pgprot_noncached)); | ||
165 | |||
164 | #ifdef CONFIG_SBUS | 166 | #ifdef CONFIG_SBUS |
165 | EXPORT_SYMBOL(sbus_root); | 167 | EXPORT_SYMBOL(sbus_root); |
166 | EXPORT_SYMBOL(dma_chain); | 168 | EXPORT_SYMBOL(dma_chain); |
@@ -260,6 +262,7 @@ EXPORT_SYMBOL(__memmove); | |||
260 | /* Moving data to/from userspace. */ | 262 | /* Moving data to/from userspace. */ |
261 | EXPORT_SYMBOL(__copy_user); | 263 | EXPORT_SYMBOL(__copy_user); |
262 | EXPORT_SYMBOL(__strncpy_from_user); | 264 | EXPORT_SYMBOL(__strncpy_from_user); |
265 | EXPORT_SYMBOL(__strnlen_user); | ||
263 | 266 | ||
264 | /* Networking helper routines. */ | 267 | /* Networking helper routines. */ |
265 | EXPORT_SYMBOL(__csum_partial_copy_sparc_generic); | 268 | EXPORT_SYMBOL(__csum_partial_copy_sparc_generic); |
diff --git a/arch/sparc/kernel/vmlinux.lds.S b/arch/sparc/kernel/vmlinux.lds.S index 47583887abc6..15109c156e83 100644 --- a/arch/sparc/kernel/vmlinux.lds.S +++ b/arch/sparc/kernel/vmlinux.lds.S | |||
@@ -35,6 +35,8 @@ SECTIONS | |||
35 | __ex_table : { *(__ex_table) } | 35 | __ex_table : { *(__ex_table) } |
36 | __stop___ex_table = .; | 36 | __stop___ex_table = .; |
37 | 37 | ||
38 | NOTES | ||
39 | |||
38 | . = ALIGN(4096); | 40 | . = ALIGN(4096); |
39 | __init_begin = .; | 41 | __init_begin = .; |
40 | _sinittext = .; | 42 | _sinittext = .; |
diff --git a/arch/sparc/lib/memset.S b/arch/sparc/lib/memset.S index a65eba41097c..1c37ea892deb 100644 --- a/arch/sparc/lib/memset.S +++ b/arch/sparc/lib/memset.S | |||
@@ -162,7 +162,7 @@ __bzero: | |||
162 | 8: | 162 | 8: |
163 | add %o0, 1, %o0 | 163 | add %o0, 1, %o0 |
164 | subcc %o1, 1, %o1 | 164 | subcc %o1, 1, %o1 |
165 | bne,a 8b | 165 | bne 8b |
166 | EX(stb %g3, [%o0 - 1], add %o1, 1) | 166 | EX(stb %g3, [%o0 - 1], add %o1, 1) |
167 | 0: | 167 | 0: |
168 | retl | 168 | retl |
diff --git a/arch/sparc/prom/printf.c b/arch/sparc/prom/printf.c index 27fdac99f790..a36ab9c5ee08 100644 --- a/arch/sparc/prom/printf.c +++ b/arch/sparc/prom/printf.c | |||
@@ -13,6 +13,7 @@ | |||
13 | */ | 13 | */ |
14 | 14 | ||
15 | #include <linux/kernel.h> | 15 | #include <linux/kernel.h> |
16 | #include <linux/module.h> | ||
16 | 17 | ||
17 | #include <asm/openprom.h> | 18 | #include <asm/openprom.h> |
18 | #include <asm/oplib.h> | 19 | #include <asm/oplib.h> |
@@ -44,3 +45,4 @@ prom_printf(char *fmt, ...) | |||
44 | 45 | ||
45 | prom_write(ppbuf, i); | 46 | prom_write(ppbuf, i); |
46 | } | 47 | } |
48 | EXPORT_SYMBOL(prom_printf); | ||
diff --git a/arch/sparc64/defconfig b/arch/sparc64/defconfig index 10e301970a44..68338a601f70 100644 --- a/arch/sparc64/defconfig +++ b/arch/sparc64/defconfig | |||
@@ -1,11 +1,12 @@ | |||
1 | # | 1 | # |
2 | # Automatically generated make config: don't edit | 2 | # Automatically generated make config: don't edit |
3 | # Linux kernel version: 2.6.22 | 3 | # Linux kernel version: 2.6.23-rc1 |
4 | # Thu Jul 19 21:30:37 2007 | 4 | # Sun Jul 22 19:24:37 2007 |
5 | # | 5 | # |
6 | CONFIG_SPARC=y | 6 | CONFIG_SPARC=y |
7 | CONFIG_SPARC64=y | 7 | CONFIG_SPARC64=y |
8 | CONFIG_GENERIC_TIME=y | 8 | CONFIG_GENERIC_TIME=y |
9 | CONFIG_GENERIC_CMOS_UPDATE=y | ||
9 | CONFIG_GENERIC_CLOCKEVENTS=y | 10 | CONFIG_GENERIC_CLOCKEVENTS=y |
10 | CONFIG_64BIT=y | 11 | CONFIG_64BIT=y |
11 | CONFIG_MMU=y | 12 | CONFIG_MMU=y |
@@ -17,6 +18,7 @@ CONFIG_ARCH_MAY_HAVE_PC_FDC=y | |||
17 | # CONFIG_ARCH_HAS_ILOG2_U64 is not set | 18 | # CONFIG_ARCH_HAS_ILOG2_U64 is not set |
18 | CONFIG_AUDIT_ARCH=y | 19 | CONFIG_AUDIT_ARCH=y |
19 | CONFIG_ARCH_NO_VIRT_TO_BUS=y | 20 | CONFIG_ARCH_NO_VIRT_TO_BUS=y |
21 | CONFIG_OF=y | ||
20 | CONFIG_SPARC64_PAGE_SIZE_8KB=y | 22 | CONFIG_SPARC64_PAGE_SIZE_8KB=y |
21 | # CONFIG_SPARC64_PAGE_SIZE_64KB is not set | 23 | # CONFIG_SPARC64_PAGE_SIZE_64KB is not set |
22 | # CONFIG_SPARC64_PAGE_SIZE_512KB is not set | 24 | # CONFIG_SPARC64_PAGE_SIZE_512KB is not set |
@@ -314,6 +316,7 @@ CONFIG_FW_LOADER=y | |||
314 | # CONFIG_SYS_HYPERVISOR is not set | 316 | # CONFIG_SYS_HYPERVISOR is not set |
315 | CONFIG_CONNECTOR=m | 317 | CONFIG_CONNECTOR=m |
316 | # CONFIG_MTD is not set | 318 | # CONFIG_MTD is not set |
319 | CONFIG_OF_DEVICE=y | ||
317 | # CONFIG_PARPORT is not set | 320 | # CONFIG_PARPORT is not set |
318 | CONFIG_BLK_DEV=y | 321 | CONFIG_BLK_DEV=y |
319 | # CONFIG_BLK_DEV_FD is not set | 322 | # CONFIG_BLK_DEV_FD is not set |
@@ -433,10 +436,7 @@ CONFIG_SCSI_FC_ATTRS=y | |||
433 | CONFIG_SCSI_ISCSI_ATTRS=m | 436 | CONFIG_SCSI_ISCSI_ATTRS=m |
434 | # CONFIG_SCSI_SAS_ATTRS is not set | 437 | # CONFIG_SCSI_SAS_ATTRS is not set |
435 | # CONFIG_SCSI_SAS_LIBSAS is not set | 438 | # CONFIG_SCSI_SAS_LIBSAS is not set |
436 | 439 | CONFIG_SCSI_LOWLEVEL=y | |
437 | # | ||
438 | # SCSI low-level drivers | ||
439 | # | ||
440 | CONFIG_ISCSI_TCP=m | 440 | CONFIG_ISCSI_TCP=m |
441 | # CONFIG_BLK_DEV_3W_XXXX_RAID is not set | 441 | # CONFIG_BLK_DEV_3W_XXXX_RAID is not set |
442 | # CONFIG_SCSI_3W_9XXX is not set | 442 | # CONFIG_SCSI_3W_9XXX is not set |
@@ -701,7 +701,6 @@ CONFIG_UNIX98_PTYS=y | |||
701 | # CONFIG_IPMI_HANDLER is not set | 701 | # CONFIG_IPMI_HANDLER is not set |
702 | # CONFIG_WATCHDOG is not set | 702 | # CONFIG_WATCHDOG is not set |
703 | # CONFIG_HW_RANDOM is not set | 703 | # CONFIG_HW_RANDOM is not set |
704 | CONFIG_RTC=y | ||
705 | # CONFIG_R3964 is not set | 704 | # CONFIG_R3964 is not set |
706 | # CONFIG_APPLICOM is not set | 705 | # CONFIG_APPLICOM is not set |
707 | # CONFIG_DRM is not set | 706 | # CONFIG_DRM is not set |
@@ -844,6 +843,7 @@ CONFIG_HWMON=y | |||
844 | # | 843 | # |
845 | # CONFIG_DISPLAY_SUPPORT is not set | 844 | # CONFIG_DISPLAY_SUPPORT is not set |
846 | # CONFIG_VGASTATE is not set | 845 | # CONFIG_VGASTATE is not set |
846 | # CONFIG_VIDEO_OUTPUT_CONTROL is not set | ||
847 | CONFIG_FB=y | 847 | CONFIG_FB=y |
848 | # CONFIG_FIRMWARE_EDID is not set | 848 | # CONFIG_FIRMWARE_EDID is not set |
849 | CONFIG_FB_DDC=y | 849 | CONFIG_FB_DDC=y |
@@ -937,7 +937,6 @@ CONFIG_SND_MIXER_OSS=m | |||
937 | CONFIG_SND_PCM_OSS=m | 937 | CONFIG_SND_PCM_OSS=m |
938 | CONFIG_SND_PCM_OSS_PLUGINS=y | 938 | CONFIG_SND_PCM_OSS_PLUGINS=y |
939 | CONFIG_SND_SEQUENCER_OSS=y | 939 | CONFIG_SND_SEQUENCER_OSS=y |
940 | # CONFIG_SND_RTCTIMER is not set | ||
941 | # CONFIG_SND_DYNAMIC_MINORS is not set | 940 | # CONFIG_SND_DYNAMIC_MINORS is not set |
942 | CONFIG_SND_SUPPORT_OLD_API=y | 941 | CONFIG_SND_SUPPORT_OLD_API=y |
943 | CONFIG_SND_VERBOSE_PROCFS=y | 942 | CONFIG_SND_VERBOSE_PROCFS=y |
@@ -1034,6 +1033,10 @@ CONFIG_SND_SUN_CS4231=m | |||
1034 | # CONFIG_SND_SOC is not set | 1033 | # CONFIG_SND_SOC is not set |
1035 | 1034 | ||
1036 | # | 1035 | # |
1036 | # SoC Audio support for SuperH | ||
1037 | # | ||
1038 | |||
1039 | # | ||
1037 | # Open Sound System | 1040 | # Open Sound System |
1038 | # | 1041 | # |
1039 | # CONFIG_SOUND_PRIME is not set | 1042 | # CONFIG_SOUND_PRIME is not set |
@@ -1157,19 +1160,7 @@ CONFIG_USB_STORAGE=m | |||
1157 | # | 1160 | # |
1158 | # CONFIG_USB_GADGET is not set | 1161 | # CONFIG_USB_GADGET is not set |
1159 | # CONFIG_MMC is not set | 1162 | # CONFIG_MMC is not set |
1160 | |||
1161 | # | ||
1162 | # LED devices | ||
1163 | # | ||
1164 | # CONFIG_NEW_LEDS is not set | 1163 | # CONFIG_NEW_LEDS is not set |
1165 | |||
1166 | # | ||
1167 | # LED drivers | ||
1168 | # | ||
1169 | |||
1170 | # | ||
1171 | # LED Triggers | ||
1172 | # | ||
1173 | # CONFIG_INFINIBAND is not set | 1164 | # CONFIG_INFINIBAND is not set |
1174 | 1165 | ||
1175 | # | 1166 | # |
@@ -1199,7 +1190,6 @@ CONFIG_USB_STORAGE=m | |||
1199 | # Misc Linux/SPARC drivers | 1190 | # Misc Linux/SPARC drivers |
1200 | # | 1191 | # |
1201 | CONFIG_SUN_OPENPROMIO=m | 1192 | CONFIG_SUN_OPENPROMIO=m |
1202 | CONFIG_SUN_MOSTEK_RTC=y | ||
1203 | # CONFIG_OBP_FLASH is not set | 1193 | # CONFIG_OBP_FLASH is not set |
1204 | # CONFIG_SUN_BPP is not set | 1194 | # CONFIG_SUN_BPP is not set |
1205 | # CONFIG_BBC_I2C is not set | 1195 | # CONFIG_BBC_I2C is not set |
diff --git a/arch/sparc64/kernel/head.S b/arch/sparc64/kernel/head.S index 35feacb6b8ec..9dbd833d79d6 100644 --- a/arch/sparc64/kernel/head.S +++ b/arch/sparc64/kernel/head.S | |||
@@ -1,15 +1,15 @@ | |||
1 | /* $Id: head.S,v 1.87 2002/02/09 19:49:31 davem Exp $ | 1 | /* head.S: Initial boot code for the Sparc64 port of Linux. |
2 | * head.S: Initial boot code for the Sparc64 port of Linux. | ||
3 | * | 2 | * |
4 | * Copyright (C) 1996,1997 David S. Miller (davem@caip.rutgers.edu) | 3 | * Copyright (C) 1996, 1997, 2007 David S. Miller (davem@davemloft.net) |
5 | * Copyright (C) 1996 David Sitsky (David.Sitsky@anu.edu.au) | 4 | * Copyright (C) 1996 David Sitsky (David.Sitsky@anu.edu.au) |
6 | * Copyright (C) 1997,1998 Jakub Jelinek (jj@sunsite.mff.cuni.cz) | 5 | * Copyright (C) 1997, 1998 Jakub Jelinek (jj@sunsite.mff.cuni.cz) |
7 | * Copyright (C) 1997 Miguel de Icaza (miguel@nuclecu.unam.mx) | 6 | * Copyright (C) 1997 Miguel de Icaza (miguel@nuclecu.unam.mx) |
8 | */ | 7 | */ |
9 | 8 | ||
10 | #include <linux/version.h> | 9 | #include <linux/version.h> |
11 | #include <linux/errno.h> | 10 | #include <linux/errno.h> |
12 | #include <linux/threads.h> | 11 | #include <linux/threads.h> |
12 | #include <linux/init.h> | ||
13 | #include <asm/thread_info.h> | 13 | #include <asm/thread_info.h> |
14 | #include <asm/asi.h> | 14 | #include <asm/asi.h> |
15 | #include <asm/pstate.h> | 15 | #include <asm/pstate.h> |
@@ -374,6 +374,7 @@ jump_to_sun4u_init: | |||
374 | jmpl %g2 + %g0, %g0 | 374 | jmpl %g2 + %g0, %g0 |
375 | nop | 375 | nop |
376 | 376 | ||
377 | .section .text.init.refok | ||
377 | sun4u_init: | 378 | sun4u_init: |
378 | BRANCH_IF_SUN4V(g1, sun4v_init) | 379 | BRANCH_IF_SUN4V(g1, sun4v_init) |
379 | 380 | ||
@@ -529,6 +530,8 @@ tlb_fixup_done: | |||
529 | nop | 530 | nop |
530 | /* Not reached... */ | 531 | /* Not reached... */ |
531 | 532 | ||
533 | .previous | ||
534 | |||
532 | /* This is meant to allow the sharing of this code between | 535 | /* This is meant to allow the sharing of this code between |
533 | * boot processor invocation (via setup_tba() below) and | 536 | * boot processor invocation (via setup_tba() below) and |
534 | * secondary processor startup (via trampoline.S). The | 537 | * secondary processor startup (via trampoline.S). The |
diff --git a/arch/sparc64/kernel/viohs.c b/arch/sparc64/kernel/viohs.c index 09126fc338ba..708fa1705fbe 100644 --- a/arch/sparc64/kernel/viohs.c +++ b/arch/sparc64/kernel/viohs.c | |||
@@ -702,7 +702,7 @@ u32 vio_send_sid(struct vio_driver_state *vio) | |||
702 | } | 702 | } |
703 | EXPORT_SYMBOL(vio_send_sid); | 703 | EXPORT_SYMBOL(vio_send_sid); |
704 | 704 | ||
705 | extern int vio_ldc_alloc(struct vio_driver_state *vio, | 705 | int vio_ldc_alloc(struct vio_driver_state *vio, |
706 | struct ldc_channel_config *base_cfg, | 706 | struct ldc_channel_config *base_cfg, |
707 | void *event_arg) | 707 | void *event_arg) |
708 | { | 708 | { |
diff --git a/arch/sparc64/kernel/vmlinux.lds.S b/arch/sparc64/kernel/vmlinux.lds.S index 481861764deb..b982fa3dd748 100644 --- a/arch/sparc64/kernel/vmlinux.lds.S +++ b/arch/sparc64/kernel/vmlinux.lds.S | |||
@@ -45,6 +45,8 @@ SECTIONS | |||
45 | __ex_table : { *(__ex_table) } | 45 | __ex_table : { *(__ex_table) } |
46 | __stop___ex_table = .; | 46 | __stop___ex_table = .; |
47 | 47 | ||
48 | NOTES | ||
49 | |||
48 | . = ALIGN(PAGE_SIZE); | 50 | . = ALIGN(PAGE_SIZE); |
49 | __init_begin = .; | 51 | __init_begin = .; |
50 | .init.text : { | 52 | .init.text : { |
diff --git a/arch/um/drivers/ubd_kern.c b/arch/um/drivers/ubd_kern.c index fc27f6c72b41..aff661fe2ee1 100644 --- a/arch/um/drivers/ubd_kern.c +++ b/arch/um/drivers/ubd_kern.c | |||
@@ -469,7 +469,7 @@ __uml_help(fakehd, | |||
469 | " Change the ubd device name to \"hd\".\n\n" | 469 | " Change the ubd device name to \"hd\".\n\n" |
470 | ); | 470 | ); |
471 | 471 | ||
472 | static void do_ubd_request(request_queue_t * q); | 472 | static void do_ubd_request(struct request_queue * q); |
473 | 473 | ||
474 | /* Only changed by ubd_init, which is an initcall. */ | 474 | /* Only changed by ubd_init, which is an initcall. */ |
475 | int thread_fd = -1; | 475 | int thread_fd = -1; |
@@ -1081,7 +1081,7 @@ static void prepare_request(struct request *req, struct io_thread_req *io_req, | |||
1081 | } | 1081 | } |
1082 | 1082 | ||
1083 | /* Called with dev->lock held */ | 1083 | /* Called with dev->lock held */ |
1084 | static void do_ubd_request(request_queue_t *q) | 1084 | static void do_ubd_request(struct request_queue *q) |
1085 | { | 1085 | { |
1086 | struct io_thread_req *io_req; | 1086 | struct io_thread_req *io_req; |
1087 | struct request *req; | 1087 | struct request *req; |
diff --git a/arch/x86_64/ia32/ia32_binfmt.c b/arch/x86_64/ia32/ia32_binfmt.c index b70f3e7cf06c..dffd2ac72747 100644 --- a/arch/x86_64/ia32/ia32_binfmt.c +++ b/arch/x86_64/ia32/ia32_binfmt.c | |||
@@ -41,8 +41,9 @@ int sysctl_vsyscall32 = 1; | |||
41 | #undef ARCH_DLINFO | 41 | #undef ARCH_DLINFO |
42 | #define ARCH_DLINFO do { \ | 42 | #define ARCH_DLINFO do { \ |
43 | if (sysctl_vsyscall32) { \ | 43 | if (sysctl_vsyscall32) { \ |
44 | NEW_AUX_ENT(AT_SYSINFO, (u32)(u64)VSYSCALL32_VSYSCALL); \ | 44 | current->mm->context.vdso = (void *)VSYSCALL32_BASE; \ |
45 | NEW_AUX_ENT(AT_SYSINFO_EHDR, VSYSCALL32_BASE); \ | 45 | NEW_AUX_ENT(AT_SYSINFO, (u32)(u64)VSYSCALL32_VSYSCALL); \ |
46 | NEW_AUX_ENT(AT_SYSINFO_EHDR, VSYSCALL32_BASE); \ | ||
46 | } \ | 47 | } \ |
47 | } while(0) | 48 | } while(0) |
48 | 49 | ||
diff --git a/arch/x86_64/ia32/syscall32.c b/arch/x86_64/ia32/syscall32.c index fc4419ff0355..15013bac181c 100644 --- a/arch/x86_64/ia32/syscall32.c +++ b/arch/x86_64/ia32/syscall32.c | |||
@@ -49,14 +49,6 @@ int syscall32_setup_pages(struct linux_binprm *bprm, int exstack) | |||
49 | return ret; | 49 | return ret; |
50 | } | 50 | } |
51 | 51 | ||
52 | const char *arch_vma_name(struct vm_area_struct *vma) | ||
53 | { | ||
54 | if (vma->vm_start == VSYSCALL32_BASE && | ||
55 | vma->vm_mm && vma->vm_mm->task_size == IA32_PAGE_OFFSET) | ||
56 | return "[vdso]"; | ||
57 | return NULL; | ||
58 | } | ||
59 | |||
60 | static int __init init_syscall32(void) | 52 | static int __init init_syscall32(void) |
61 | { | 53 | { |
62 | char *syscall32_page = (void *)get_zeroed_page(GFP_KERNEL); | 54 | char *syscall32_page = (void *)get_zeroed_page(GFP_KERNEL); |
diff --git a/arch/x86_64/kernel/acpi/Makefile b/arch/x86_64/kernel/acpi/Makefile index 080b9963f1bc..17595d23fee7 100644 --- a/arch/x86_64/kernel/acpi/Makefile +++ b/arch/x86_64/kernel/acpi/Makefile | |||
@@ -1,6 +1,6 @@ | |||
1 | obj-y := boot.o | 1 | obj-y := boot.o |
2 | boot-y := ../../../i386/kernel/acpi/boot.o | 2 | boot-y := ../../../i386/kernel/acpi/boot.o |
3 | obj-$(CONFIG_ACPI_SLEEP) += sleep.o wakeup.o | 3 | obj-y += sleep.o wakeup.o |
4 | 4 | ||
5 | ifneq ($(CONFIG_ACPI_PROCESSOR),) | 5 | ifneq ($(CONFIG_ACPI_PROCESSOR),) |
6 | obj-y += processor.o | 6 | obj-y += processor.o |
diff --git a/arch/x86_64/kernel/acpi/sleep.c b/arch/x86_64/kernel/acpi/sleep.c index 4277f2b27e6d..79475d237071 100644 --- a/arch/x86_64/kernel/acpi/sleep.c +++ b/arch/x86_64/kernel/acpi/sleep.c | |||
@@ -51,8 +51,6 @@ | |||
51 | Low-Level Sleep Support | 51 | Low-Level Sleep Support |
52 | -------------------------------------------------------------------------- */ | 52 | -------------------------------------------------------------------------- */ |
53 | 53 | ||
54 | #ifdef CONFIG_ACPI_SLEEP | ||
55 | |||
56 | /* address in low memory of the wakeup routine. */ | 54 | /* address in low memory of the wakeup routine. */ |
57 | unsigned long acpi_wakeup_address = 0; | 55 | unsigned long acpi_wakeup_address = 0; |
58 | unsigned long acpi_realmode_flags; | 56 | unsigned long acpi_realmode_flags; |
@@ -117,8 +115,6 @@ static int __init acpi_sleep_setup(char *str) | |||
117 | 115 | ||
118 | __setup("acpi_sleep=", acpi_sleep_setup); | 116 | __setup("acpi_sleep=", acpi_sleep_setup); |
119 | 117 | ||
120 | #endif /*CONFIG_ACPI_SLEEP */ | ||
121 | |||
122 | void acpi_pci_link_exit(void) | 118 | void acpi_pci_link_exit(void) |
123 | { | 119 | { |
124 | } | 120 | } |
diff --git a/arch/x86_64/kernel/head.S b/arch/x86_64/kernel/head.S index e89abcdbdde8..3a16e417dd8d 100644 --- a/arch/x86_64/kernel/head.S +++ b/arch/x86_64/kernel/head.S | |||
@@ -120,7 +120,7 @@ ident_complete: | |||
120 | addq %rbp, trampoline_level4_pgt + 0(%rip) | 120 | addq %rbp, trampoline_level4_pgt + 0(%rip) |
121 | addq %rbp, trampoline_level4_pgt + (511*8)(%rip) | 121 | addq %rbp, trampoline_level4_pgt + (511*8)(%rip) |
122 | #endif | 122 | #endif |
123 | #ifdef CONFIG_ACPI_SLEEP | 123 | #ifdef CONFIG_ACPI |
124 | addq %rbp, wakeup_level4_pgt + 0(%rip) | 124 | addq %rbp, wakeup_level4_pgt + 0(%rip) |
125 | addq %rbp, wakeup_level4_pgt + (511*8)(%rip) | 125 | addq %rbp, wakeup_level4_pgt + (511*8)(%rip) |
126 | #endif | 126 | #endif |
diff --git a/arch/x86_64/kernel/setup.c b/arch/x86_64/kernel/setup.c index af838f6b0b7f..0f400f3c4694 100644 --- a/arch/x86_64/kernel/setup.c +++ b/arch/x86_64/kernel/setup.c | |||
@@ -333,7 +333,7 @@ void __init setup_arch(char **cmdline_p) | |||
333 | reserve_bootmem_generic(SMP_TRAMPOLINE_BASE, 2*PAGE_SIZE); | 333 | reserve_bootmem_generic(SMP_TRAMPOLINE_BASE, 2*PAGE_SIZE); |
334 | #endif | 334 | #endif |
335 | 335 | ||
336 | #ifdef CONFIG_ACPI_SLEEP | 336 | #ifdef CONFIG_ACPI |
337 | /* | 337 | /* |
338 | * Reserve low memory region for sleep support. | 338 | * Reserve low memory region for sleep support. |
339 | */ | 339 | */ |
diff --git a/arch/x86_64/kernel/tce.c b/arch/x86_64/kernel/tce.c index 3aeae2fa2e24..821527e7faa3 100644 --- a/arch/x86_64/kernel/tce.c +++ b/arch/x86_64/kernel/tce.c | |||
@@ -165,7 +165,7 @@ done: | |||
165 | return ret; | 165 | return ret; |
166 | } | 166 | } |
167 | 167 | ||
168 | void* alloc_tce_table(void) | 168 | void * __init alloc_tce_table(void) |
169 | { | 169 | { |
170 | unsigned int size; | 170 | unsigned int size; |
171 | 171 | ||
@@ -175,7 +175,7 @@ void* alloc_tce_table(void) | |||
175 | return __alloc_bootmem_low(size, size, 0); | 175 | return __alloc_bootmem_low(size, size, 0); |
176 | } | 176 | } |
177 | 177 | ||
178 | void free_tce_table(void *tbl) | 178 | void __init free_tce_table(void *tbl) |
179 | { | 179 | { |
180 | unsigned int size; | 180 | unsigned int size; |
181 | 181 | ||
diff --git a/arch/x86_64/kernel/tsc.c b/arch/x86_64/kernel/tsc.c index 9b76b03d0600..2a59bde663f2 100644 --- a/arch/x86_64/kernel/tsc.c +++ b/arch/x86_64/kernel/tsc.c | |||
@@ -118,8 +118,6 @@ core_initcall(cpufreq_tsc); | |||
118 | 118 | ||
119 | #endif | 119 | #endif |
120 | 120 | ||
121 | static int tsc_unstable = 0; | ||
122 | |||
123 | /* | 121 | /* |
124 | * Make an educated guess if the TSC is trustworthy and synchronized | 122 | * Make an educated guess if the TSC is trustworthy and synchronized |
125 | * over all CPUs. | 123 | * over all CPUs. |
diff --git a/arch/x86_64/mm/init.c b/arch/x86_64/mm/init.c index 38f5d6368006..458893b376f8 100644 --- a/arch/x86_64/mm/init.c +++ b/arch/x86_64/mm/init.c | |||
@@ -600,6 +600,16 @@ void mark_rodata_ro(void) | |||
600 | { | 600 | { |
601 | unsigned long start = (unsigned long)_stext, end; | 601 | unsigned long start = (unsigned long)_stext, end; |
602 | 602 | ||
603 | #ifdef CONFIG_HOTPLUG_CPU | ||
604 | /* It must still be possible to apply SMP alternatives. */ | ||
605 | if (num_possible_cpus() > 1) | ||
606 | start = (unsigned long)_etext; | ||
607 | #endif | ||
608 | |||
609 | #ifdef CONFIG_KPROBES | ||
610 | start = (unsigned long)__start_rodata; | ||
611 | #endif | ||
612 | |||
603 | end = (unsigned long)__end_rodata; | 613 | end = (unsigned long)__end_rodata; |
604 | start = (start + PAGE_SIZE - 1) & PAGE_MASK; | 614 | start = (start + PAGE_SIZE - 1) & PAGE_MASK; |
605 | end &= PAGE_MASK; | 615 | end &= PAGE_MASK; |
diff --git a/block/as-iosched.c b/block/as-iosched.c index 3e316dd72529..dc715a562e14 100644 --- a/block/as-iosched.c +++ b/block/as-iosched.c | |||
@@ -796,7 +796,7 @@ static void update_write_batch(struct as_data *ad) | |||
796 | * as_completed_request is to be called when a request has completed and | 796 | * as_completed_request is to be called when a request has completed and |
797 | * returned something to the requesting process, be it an error or data. | 797 | * returned something to the requesting process, be it an error or data. |
798 | */ | 798 | */ |
799 | static void as_completed_request(request_queue_t *q, struct request *rq) | 799 | static void as_completed_request(struct request_queue *q, struct request *rq) |
800 | { | 800 | { |
801 | struct as_data *ad = q->elevator->elevator_data; | 801 | struct as_data *ad = q->elevator->elevator_data; |
802 | 802 | ||
@@ -853,7 +853,8 @@ out: | |||
853 | * reference unless it replaces the request at somepart of the elevator | 853 | * reference unless it replaces the request at somepart of the elevator |
854 | * (ie. the dispatch queue) | 854 | * (ie. the dispatch queue) |
855 | */ | 855 | */ |
856 | static void as_remove_queued_request(request_queue_t *q, struct request *rq) | 856 | static void as_remove_queued_request(struct request_queue *q, |
857 | struct request *rq) | ||
857 | { | 858 | { |
858 | const int data_dir = rq_is_sync(rq); | 859 | const int data_dir = rq_is_sync(rq); |
859 | struct as_data *ad = q->elevator->elevator_data; | 860 | struct as_data *ad = q->elevator->elevator_data; |
@@ -978,7 +979,7 @@ static void as_move_to_dispatch(struct as_data *ad, struct request *rq) | |||
978 | * read/write expire, batch expire, etc, and moves it to the dispatch | 979 | * read/write expire, batch expire, etc, and moves it to the dispatch |
979 | * queue. Returns 1 if a request was found, 0 otherwise. | 980 | * queue. Returns 1 if a request was found, 0 otherwise. |
980 | */ | 981 | */ |
981 | static int as_dispatch_request(request_queue_t *q, int force) | 982 | static int as_dispatch_request(struct request_queue *q, int force) |
982 | { | 983 | { |
983 | struct as_data *ad = q->elevator->elevator_data; | 984 | struct as_data *ad = q->elevator->elevator_data; |
984 | const int reads = !list_empty(&ad->fifo_list[REQ_SYNC]); | 985 | const int reads = !list_empty(&ad->fifo_list[REQ_SYNC]); |
@@ -1139,7 +1140,7 @@ fifo_expired: | |||
1139 | /* | 1140 | /* |
1140 | * add rq to rbtree and fifo | 1141 | * add rq to rbtree and fifo |
1141 | */ | 1142 | */ |
1142 | static void as_add_request(request_queue_t *q, struct request *rq) | 1143 | static void as_add_request(struct request_queue *q, struct request *rq) |
1143 | { | 1144 | { |
1144 | struct as_data *ad = q->elevator->elevator_data; | 1145 | struct as_data *ad = q->elevator->elevator_data; |
1145 | int data_dir; | 1146 | int data_dir; |
@@ -1167,7 +1168,7 @@ static void as_add_request(request_queue_t *q, struct request *rq) | |||
1167 | RQ_SET_STATE(rq, AS_RQ_QUEUED); | 1168 | RQ_SET_STATE(rq, AS_RQ_QUEUED); |
1168 | } | 1169 | } |
1169 | 1170 | ||
1170 | static void as_activate_request(request_queue_t *q, struct request *rq) | 1171 | static void as_activate_request(struct request_queue *q, struct request *rq) |
1171 | { | 1172 | { |
1172 | WARN_ON(RQ_STATE(rq) != AS_RQ_DISPATCHED); | 1173 | WARN_ON(RQ_STATE(rq) != AS_RQ_DISPATCHED); |
1173 | RQ_SET_STATE(rq, AS_RQ_REMOVED); | 1174 | RQ_SET_STATE(rq, AS_RQ_REMOVED); |
@@ -1175,7 +1176,7 @@ static void as_activate_request(request_queue_t *q, struct request *rq) | |||
1175 | atomic_dec(&RQ_IOC(rq)->aic->nr_dispatched); | 1176 | atomic_dec(&RQ_IOC(rq)->aic->nr_dispatched); |
1176 | } | 1177 | } |
1177 | 1178 | ||
1178 | static void as_deactivate_request(request_queue_t *q, struct request *rq) | 1179 | static void as_deactivate_request(struct request_queue *q, struct request *rq) |
1179 | { | 1180 | { |
1180 | WARN_ON(RQ_STATE(rq) != AS_RQ_REMOVED); | 1181 | WARN_ON(RQ_STATE(rq) != AS_RQ_REMOVED); |
1181 | RQ_SET_STATE(rq, AS_RQ_DISPATCHED); | 1182 | RQ_SET_STATE(rq, AS_RQ_DISPATCHED); |
@@ -1189,7 +1190,7 @@ static void as_deactivate_request(request_queue_t *q, struct request *rq) | |||
1189 | * is not empty - it is used in the block layer to check for plugging and | 1190 | * is not empty - it is used in the block layer to check for plugging and |
1190 | * merging opportunities | 1191 | * merging opportunities |
1191 | */ | 1192 | */ |
1192 | static int as_queue_empty(request_queue_t *q) | 1193 | static int as_queue_empty(struct request_queue *q) |
1193 | { | 1194 | { |
1194 | struct as_data *ad = q->elevator->elevator_data; | 1195 | struct as_data *ad = q->elevator->elevator_data; |
1195 | 1196 | ||
@@ -1198,7 +1199,7 @@ static int as_queue_empty(request_queue_t *q) | |||
1198 | } | 1199 | } |
1199 | 1200 | ||
1200 | static int | 1201 | static int |
1201 | as_merge(request_queue_t *q, struct request **req, struct bio *bio) | 1202 | as_merge(struct request_queue *q, struct request **req, struct bio *bio) |
1202 | { | 1203 | { |
1203 | struct as_data *ad = q->elevator->elevator_data; | 1204 | struct as_data *ad = q->elevator->elevator_data; |
1204 | sector_t rb_key = bio->bi_sector + bio_sectors(bio); | 1205 | sector_t rb_key = bio->bi_sector + bio_sectors(bio); |
@@ -1216,7 +1217,8 @@ as_merge(request_queue_t *q, struct request **req, struct bio *bio) | |||
1216 | return ELEVATOR_NO_MERGE; | 1217 | return ELEVATOR_NO_MERGE; |
1217 | } | 1218 | } |
1218 | 1219 | ||
1219 | static void as_merged_request(request_queue_t *q, struct request *req, int type) | 1220 | static void as_merged_request(struct request_queue *q, struct request *req, |
1221 | int type) | ||
1220 | { | 1222 | { |
1221 | struct as_data *ad = q->elevator->elevator_data; | 1223 | struct as_data *ad = q->elevator->elevator_data; |
1222 | 1224 | ||
@@ -1234,7 +1236,7 @@ static void as_merged_request(request_queue_t *q, struct request *req, int type) | |||
1234 | } | 1236 | } |
1235 | } | 1237 | } |
1236 | 1238 | ||
1237 | static void as_merged_requests(request_queue_t *q, struct request *req, | 1239 | static void as_merged_requests(struct request_queue *q, struct request *req, |
1238 | struct request *next) | 1240 | struct request *next) |
1239 | { | 1241 | { |
1240 | /* | 1242 | /* |
@@ -1285,7 +1287,7 @@ static void as_work_handler(struct work_struct *work) | |||
1285 | spin_unlock_irqrestore(q->queue_lock, flags); | 1287 | spin_unlock_irqrestore(q->queue_lock, flags); |
1286 | } | 1288 | } |
1287 | 1289 | ||
1288 | static int as_may_queue(request_queue_t *q, int rw) | 1290 | static int as_may_queue(struct request_queue *q, int rw) |
1289 | { | 1291 | { |
1290 | int ret = ELV_MQUEUE_MAY; | 1292 | int ret = ELV_MQUEUE_MAY; |
1291 | struct as_data *ad = q->elevator->elevator_data; | 1293 | struct as_data *ad = q->elevator->elevator_data; |
@@ -1318,7 +1320,7 @@ static void as_exit_queue(elevator_t *e) | |||
1318 | /* | 1320 | /* |
1319 | * initialize elevator private data (as_data). | 1321 | * initialize elevator private data (as_data). |
1320 | */ | 1322 | */ |
1321 | static void *as_init_queue(request_queue_t *q) | 1323 | static void *as_init_queue(struct request_queue *q) |
1322 | { | 1324 | { |
1323 | struct as_data *ad; | 1325 | struct as_data *ad; |
1324 | 1326 | ||
diff --git a/block/blktrace.c b/block/blktrace.c index 3f0e7c37c059..20c3e22587b5 100644 --- a/block/blktrace.c +++ b/block/blktrace.c | |||
@@ -231,7 +231,7 @@ static void blk_trace_cleanup(struct blk_trace *bt) | |||
231 | kfree(bt); | 231 | kfree(bt); |
232 | } | 232 | } |
233 | 233 | ||
234 | static int blk_trace_remove(request_queue_t *q) | 234 | static int blk_trace_remove(struct request_queue *q) |
235 | { | 235 | { |
236 | struct blk_trace *bt; | 236 | struct blk_trace *bt; |
237 | 237 | ||
@@ -312,7 +312,7 @@ static struct rchan_callbacks blk_relay_callbacks = { | |||
312 | /* | 312 | /* |
313 | * Setup everything required to start tracing | 313 | * Setup everything required to start tracing |
314 | */ | 314 | */ |
315 | static int blk_trace_setup(request_queue_t *q, struct block_device *bdev, | 315 | static int blk_trace_setup(struct request_queue *q, struct block_device *bdev, |
316 | char __user *arg) | 316 | char __user *arg) |
317 | { | 317 | { |
318 | struct blk_user_trace_setup buts; | 318 | struct blk_user_trace_setup buts; |
@@ -401,7 +401,7 @@ err: | |||
401 | return ret; | 401 | return ret; |
402 | } | 402 | } |
403 | 403 | ||
404 | static int blk_trace_startstop(request_queue_t *q, int start) | 404 | static int blk_trace_startstop(struct request_queue *q, int start) |
405 | { | 405 | { |
406 | struct blk_trace *bt; | 406 | struct blk_trace *bt; |
407 | int ret; | 407 | int ret; |
@@ -444,7 +444,7 @@ static int blk_trace_startstop(request_queue_t *q, int start) | |||
444 | **/ | 444 | **/ |
445 | int blk_trace_ioctl(struct block_device *bdev, unsigned cmd, char __user *arg) | 445 | int blk_trace_ioctl(struct block_device *bdev, unsigned cmd, char __user *arg) |
446 | { | 446 | { |
447 | request_queue_t *q; | 447 | struct request_queue *q; |
448 | int ret, start = 0; | 448 | int ret, start = 0; |
449 | 449 | ||
450 | q = bdev_get_queue(bdev); | 450 | q = bdev_get_queue(bdev); |
@@ -479,7 +479,7 @@ int blk_trace_ioctl(struct block_device *bdev, unsigned cmd, char __user *arg) | |||
479 | * @q: the request queue associated with the device | 479 | * @q: the request queue associated with the device |
480 | * | 480 | * |
481 | **/ | 481 | **/ |
482 | void blk_trace_shutdown(request_queue_t *q) | 482 | void blk_trace_shutdown(struct request_queue *q) |
483 | { | 483 | { |
484 | if (q->blk_trace) { | 484 | if (q->blk_trace) { |
485 | blk_trace_startstop(q, 0); | 485 | blk_trace_startstop(q, 0); |
diff --git a/block/bsg.c b/block/bsg.c index b571869928a8..3b2f05258a92 100644 --- a/block/bsg.c +++ b/block/bsg.c | |||
@@ -37,7 +37,7 @@ | |||
37 | #define BSG_VERSION "0.4" | 37 | #define BSG_VERSION "0.4" |
38 | 38 | ||
39 | struct bsg_device { | 39 | struct bsg_device { |
40 | request_queue_t *queue; | 40 | struct request_queue *queue; |
41 | spinlock_t lock; | 41 | spinlock_t lock; |
42 | struct list_head busy_list; | 42 | struct list_head busy_list; |
43 | struct list_head done_list; | 43 | struct list_head done_list; |
@@ -180,7 +180,7 @@ unlock: | |||
180 | return ret; | 180 | return ret; |
181 | } | 181 | } |
182 | 182 | ||
183 | static int blk_fill_sgv4_hdr_rq(request_queue_t *q, struct request *rq, | 183 | static int blk_fill_sgv4_hdr_rq(struct request_queue *q, struct request *rq, |
184 | struct sg_io_v4 *hdr, int has_write_perm) | 184 | struct sg_io_v4 *hdr, int has_write_perm) |
185 | { | 185 | { |
186 | memset(rq->cmd, 0, BLK_MAX_CDB); /* ATAPI hates garbage after CDB */ | 186 | memset(rq->cmd, 0, BLK_MAX_CDB); /* ATAPI hates garbage after CDB */ |
@@ -214,7 +214,7 @@ static int blk_fill_sgv4_hdr_rq(request_queue_t *q, struct request *rq, | |||
214 | * Check if sg_io_v4 from user is allowed and valid | 214 | * Check if sg_io_v4 from user is allowed and valid |
215 | */ | 215 | */ |
216 | static int | 216 | static int |
217 | bsg_validate_sgv4_hdr(request_queue_t *q, struct sg_io_v4 *hdr, int *rw) | 217 | bsg_validate_sgv4_hdr(struct request_queue *q, struct sg_io_v4 *hdr, int *rw) |
218 | { | 218 | { |
219 | int ret = 0; | 219 | int ret = 0; |
220 | 220 | ||
@@ -250,7 +250,7 @@ bsg_validate_sgv4_hdr(request_queue_t *q, struct sg_io_v4 *hdr, int *rw) | |||
250 | static struct request * | 250 | static struct request * |
251 | bsg_map_hdr(struct bsg_device *bd, struct sg_io_v4 *hdr) | 251 | bsg_map_hdr(struct bsg_device *bd, struct sg_io_v4 *hdr) |
252 | { | 252 | { |
253 | request_queue_t *q = bd->queue; | 253 | struct request_queue *q = bd->queue; |
254 | struct request *rq, *next_rq = NULL; | 254 | struct request *rq, *next_rq = NULL; |
255 | int ret, rw; | 255 | int ret, rw; |
256 | unsigned int dxfer_len; | 256 | unsigned int dxfer_len; |
@@ -345,7 +345,7 @@ static void bsg_rq_end_io(struct request *rq, int uptodate) | |||
345 | * do final setup of a 'bc' and submit the matching 'rq' to the block | 345 | * do final setup of a 'bc' and submit the matching 'rq' to the block |
346 | * layer for io | 346 | * layer for io |
347 | */ | 347 | */ |
348 | static void bsg_add_command(struct bsg_device *bd, request_queue_t *q, | 348 | static void bsg_add_command(struct bsg_device *bd, struct request_queue *q, |
349 | struct bsg_command *bc, struct request *rq) | 349 | struct bsg_command *bc, struct request *rq) |
350 | { | 350 | { |
351 | rq->sense = bc->sense; | 351 | rq->sense = bc->sense; |
@@ -611,7 +611,7 @@ static int __bsg_write(struct bsg_device *bd, const char __user *buf, | |||
611 | bc = NULL; | 611 | bc = NULL; |
612 | ret = 0; | 612 | ret = 0; |
613 | while (nr_commands) { | 613 | while (nr_commands) { |
614 | request_queue_t *q = bd->queue; | 614 | struct request_queue *q = bd->queue; |
615 | 615 | ||
616 | bc = bsg_alloc_command(bd); | 616 | bc = bsg_alloc_command(bd); |
617 | if (IS_ERR(bc)) { | 617 | if (IS_ERR(bc)) { |
diff --git a/block/cfq-iosched.c b/block/cfq-iosched.c index d148ccbc36d1..54dc05439009 100644 --- a/block/cfq-iosched.c +++ b/block/cfq-iosched.c | |||
@@ -71,7 +71,7 @@ struct cfq_rb_root { | |||
71 | * Per block device queue structure | 71 | * Per block device queue structure |
72 | */ | 72 | */ |
73 | struct cfq_data { | 73 | struct cfq_data { |
74 | request_queue_t *queue; | 74 | struct request_queue *queue; |
75 | 75 | ||
76 | /* | 76 | /* |
77 | * rr list of queues with requests and the count of them | 77 | * rr list of queues with requests and the count of them |
@@ -197,7 +197,7 @@ CFQ_CFQQ_FNS(slice_new); | |||
197 | CFQ_CFQQ_FNS(sync); | 197 | CFQ_CFQQ_FNS(sync); |
198 | #undef CFQ_CFQQ_FNS | 198 | #undef CFQ_CFQQ_FNS |
199 | 199 | ||
200 | static void cfq_dispatch_insert(request_queue_t *, struct request *); | 200 | static void cfq_dispatch_insert(struct request_queue *, struct request *); |
201 | static struct cfq_queue *cfq_get_queue(struct cfq_data *, int, | 201 | static struct cfq_queue *cfq_get_queue(struct cfq_data *, int, |
202 | struct task_struct *, gfp_t); | 202 | struct task_struct *, gfp_t); |
203 | static struct cfq_io_context *cfq_cic_rb_lookup(struct cfq_data *, | 203 | static struct cfq_io_context *cfq_cic_rb_lookup(struct cfq_data *, |
@@ -237,7 +237,7 @@ static inline void cfq_schedule_dispatch(struct cfq_data *cfqd) | |||
237 | kblockd_schedule_work(&cfqd->unplug_work); | 237 | kblockd_schedule_work(&cfqd->unplug_work); |
238 | } | 238 | } |
239 | 239 | ||
240 | static int cfq_queue_empty(request_queue_t *q) | 240 | static int cfq_queue_empty(struct request_queue *q) |
241 | { | 241 | { |
242 | struct cfq_data *cfqd = q->elevator->elevator_data; | 242 | struct cfq_data *cfqd = q->elevator->elevator_data; |
243 | 243 | ||
@@ -623,7 +623,7 @@ cfq_find_rq_fmerge(struct cfq_data *cfqd, struct bio *bio) | |||
623 | return NULL; | 623 | return NULL; |
624 | } | 624 | } |
625 | 625 | ||
626 | static void cfq_activate_request(request_queue_t *q, struct request *rq) | 626 | static void cfq_activate_request(struct request_queue *q, struct request *rq) |
627 | { | 627 | { |
628 | struct cfq_data *cfqd = q->elevator->elevator_data; | 628 | struct cfq_data *cfqd = q->elevator->elevator_data; |
629 | 629 | ||
@@ -641,7 +641,7 @@ static void cfq_activate_request(request_queue_t *q, struct request *rq) | |||
641 | cfqd->last_position = rq->hard_sector + rq->hard_nr_sectors; | 641 | cfqd->last_position = rq->hard_sector + rq->hard_nr_sectors; |
642 | } | 642 | } |
643 | 643 | ||
644 | static void cfq_deactivate_request(request_queue_t *q, struct request *rq) | 644 | static void cfq_deactivate_request(struct request_queue *q, struct request *rq) |
645 | { | 645 | { |
646 | struct cfq_data *cfqd = q->elevator->elevator_data; | 646 | struct cfq_data *cfqd = q->elevator->elevator_data; |
647 | 647 | ||
@@ -665,7 +665,8 @@ static void cfq_remove_request(struct request *rq) | |||
665 | } | 665 | } |
666 | } | 666 | } |
667 | 667 | ||
668 | static int cfq_merge(request_queue_t *q, struct request **req, struct bio *bio) | 668 | static int cfq_merge(struct request_queue *q, struct request **req, |
669 | struct bio *bio) | ||
669 | { | 670 | { |
670 | struct cfq_data *cfqd = q->elevator->elevator_data; | 671 | struct cfq_data *cfqd = q->elevator->elevator_data; |
671 | struct request *__rq; | 672 | struct request *__rq; |
@@ -679,7 +680,7 @@ static int cfq_merge(request_queue_t *q, struct request **req, struct bio *bio) | |||
679 | return ELEVATOR_NO_MERGE; | 680 | return ELEVATOR_NO_MERGE; |
680 | } | 681 | } |
681 | 682 | ||
682 | static void cfq_merged_request(request_queue_t *q, struct request *req, | 683 | static void cfq_merged_request(struct request_queue *q, struct request *req, |
683 | int type) | 684 | int type) |
684 | { | 685 | { |
685 | if (type == ELEVATOR_FRONT_MERGE) { | 686 | if (type == ELEVATOR_FRONT_MERGE) { |
@@ -690,7 +691,7 @@ static void cfq_merged_request(request_queue_t *q, struct request *req, | |||
690 | } | 691 | } |
691 | 692 | ||
692 | static void | 693 | static void |
693 | cfq_merged_requests(request_queue_t *q, struct request *rq, | 694 | cfq_merged_requests(struct request_queue *q, struct request *rq, |
694 | struct request *next) | 695 | struct request *next) |
695 | { | 696 | { |
696 | /* | 697 | /* |
@@ -703,7 +704,7 @@ cfq_merged_requests(request_queue_t *q, struct request *rq, | |||
703 | cfq_remove_request(next); | 704 | cfq_remove_request(next); |
704 | } | 705 | } |
705 | 706 | ||
706 | static int cfq_allow_merge(request_queue_t *q, struct request *rq, | 707 | static int cfq_allow_merge(struct request_queue *q, struct request *rq, |
707 | struct bio *bio) | 708 | struct bio *bio) |
708 | { | 709 | { |
709 | struct cfq_data *cfqd = q->elevator->elevator_data; | 710 | struct cfq_data *cfqd = q->elevator->elevator_data; |
@@ -913,7 +914,7 @@ static void cfq_arm_slice_timer(struct cfq_data *cfqd) | |||
913 | /* | 914 | /* |
914 | * Move request from internal lists to the request queue dispatch list. | 915 | * Move request from internal lists to the request queue dispatch list. |
915 | */ | 916 | */ |
916 | static void cfq_dispatch_insert(request_queue_t *q, struct request *rq) | 917 | static void cfq_dispatch_insert(struct request_queue *q, struct request *rq) |
917 | { | 918 | { |
918 | struct cfq_data *cfqd = q->elevator->elevator_data; | 919 | struct cfq_data *cfqd = q->elevator->elevator_data; |
919 | struct cfq_queue *cfqq = RQ_CFQQ(rq); | 920 | struct cfq_queue *cfqq = RQ_CFQQ(rq); |
@@ -1093,7 +1094,7 @@ static int cfq_forced_dispatch(struct cfq_data *cfqd) | |||
1093 | return dispatched; | 1094 | return dispatched; |
1094 | } | 1095 | } |
1095 | 1096 | ||
1096 | static int cfq_dispatch_requests(request_queue_t *q, int force) | 1097 | static int cfq_dispatch_requests(struct request_queue *q, int force) |
1097 | { | 1098 | { |
1098 | struct cfq_data *cfqd = q->elevator->elevator_data; | 1099 | struct cfq_data *cfqd = q->elevator->elevator_data; |
1099 | struct cfq_queue *cfqq; | 1100 | struct cfq_queue *cfqq; |
@@ -1214,7 +1215,7 @@ static void cfq_exit_single_io_context(struct cfq_io_context *cic) | |||
1214 | struct cfq_data *cfqd = cic->key; | 1215 | struct cfq_data *cfqd = cic->key; |
1215 | 1216 | ||
1216 | if (cfqd) { | 1217 | if (cfqd) { |
1217 | request_queue_t *q = cfqd->queue; | 1218 | struct request_queue *q = cfqd->queue; |
1218 | 1219 | ||
1219 | spin_lock_irq(q->queue_lock); | 1220 | spin_lock_irq(q->queue_lock); |
1220 | __cfq_exit_single_io_context(cfqd, cic); | 1221 | __cfq_exit_single_io_context(cfqd, cic); |
@@ -1775,7 +1776,7 @@ cfq_rq_enqueued(struct cfq_data *cfqd, struct cfq_queue *cfqq, | |||
1775 | } | 1776 | } |
1776 | } | 1777 | } |
1777 | 1778 | ||
1778 | static void cfq_insert_request(request_queue_t *q, struct request *rq) | 1779 | static void cfq_insert_request(struct request_queue *q, struct request *rq) |
1779 | { | 1780 | { |
1780 | struct cfq_data *cfqd = q->elevator->elevator_data; | 1781 | struct cfq_data *cfqd = q->elevator->elevator_data; |
1781 | struct cfq_queue *cfqq = RQ_CFQQ(rq); | 1782 | struct cfq_queue *cfqq = RQ_CFQQ(rq); |
@@ -1789,7 +1790,7 @@ static void cfq_insert_request(request_queue_t *q, struct request *rq) | |||
1789 | cfq_rq_enqueued(cfqd, cfqq, rq); | 1790 | cfq_rq_enqueued(cfqd, cfqq, rq); |
1790 | } | 1791 | } |
1791 | 1792 | ||
1792 | static void cfq_completed_request(request_queue_t *q, struct request *rq) | 1793 | static void cfq_completed_request(struct request_queue *q, struct request *rq) |
1793 | { | 1794 | { |
1794 | struct cfq_queue *cfqq = RQ_CFQQ(rq); | 1795 | struct cfq_queue *cfqq = RQ_CFQQ(rq); |
1795 | struct cfq_data *cfqd = cfqq->cfqd; | 1796 | struct cfq_data *cfqd = cfqq->cfqd; |
@@ -1868,7 +1869,7 @@ static inline int __cfq_may_queue(struct cfq_queue *cfqq) | |||
1868 | return ELV_MQUEUE_MAY; | 1869 | return ELV_MQUEUE_MAY; |
1869 | } | 1870 | } |
1870 | 1871 | ||
1871 | static int cfq_may_queue(request_queue_t *q, int rw) | 1872 | static int cfq_may_queue(struct request_queue *q, int rw) |
1872 | { | 1873 | { |
1873 | struct cfq_data *cfqd = q->elevator->elevator_data; | 1874 | struct cfq_data *cfqd = q->elevator->elevator_data; |
1874 | struct task_struct *tsk = current; | 1875 | struct task_struct *tsk = current; |
@@ -1922,7 +1923,7 @@ static void cfq_put_request(struct request *rq) | |||
1922 | * Allocate cfq data structures associated with this request. | 1923 | * Allocate cfq data structures associated with this request. |
1923 | */ | 1924 | */ |
1924 | static int | 1925 | static int |
1925 | cfq_set_request(request_queue_t *q, struct request *rq, gfp_t gfp_mask) | 1926 | cfq_set_request(struct request_queue *q, struct request *rq, gfp_t gfp_mask) |
1926 | { | 1927 | { |
1927 | struct cfq_data *cfqd = q->elevator->elevator_data; | 1928 | struct cfq_data *cfqd = q->elevator->elevator_data; |
1928 | struct task_struct *tsk = current; | 1929 | struct task_struct *tsk = current; |
@@ -1974,7 +1975,7 @@ static void cfq_kick_queue(struct work_struct *work) | |||
1974 | { | 1975 | { |
1975 | struct cfq_data *cfqd = | 1976 | struct cfq_data *cfqd = |
1976 | container_of(work, struct cfq_data, unplug_work); | 1977 | container_of(work, struct cfq_data, unplug_work); |
1977 | request_queue_t *q = cfqd->queue; | 1978 | struct request_queue *q = cfqd->queue; |
1978 | unsigned long flags; | 1979 | unsigned long flags; |
1979 | 1980 | ||
1980 | spin_lock_irqsave(q->queue_lock, flags); | 1981 | spin_lock_irqsave(q->queue_lock, flags); |
@@ -2072,7 +2073,7 @@ static void cfq_put_async_queues(struct cfq_data *cfqd) | |||
2072 | static void cfq_exit_queue(elevator_t *e) | 2073 | static void cfq_exit_queue(elevator_t *e) |
2073 | { | 2074 | { |
2074 | struct cfq_data *cfqd = e->elevator_data; | 2075 | struct cfq_data *cfqd = e->elevator_data; |
2075 | request_queue_t *q = cfqd->queue; | 2076 | struct request_queue *q = cfqd->queue; |
2076 | 2077 | ||
2077 | cfq_shutdown_timer_wq(cfqd); | 2078 | cfq_shutdown_timer_wq(cfqd); |
2078 | 2079 | ||
@@ -2098,7 +2099,7 @@ static void cfq_exit_queue(elevator_t *e) | |||
2098 | kfree(cfqd); | 2099 | kfree(cfqd); |
2099 | } | 2100 | } |
2100 | 2101 | ||
2101 | static void *cfq_init_queue(request_queue_t *q) | 2102 | static void *cfq_init_queue(struct request_queue *q) |
2102 | { | 2103 | { |
2103 | struct cfq_data *cfqd; | 2104 | struct cfq_data *cfqd; |
2104 | 2105 | ||
diff --git a/block/deadline-iosched.c b/block/deadline-iosched.c index 87ca02ac84cb..1a511ffaf8a4 100644 --- a/block/deadline-iosched.c +++ b/block/deadline-iosched.c | |||
@@ -106,7 +106,7 @@ deadline_add_request(struct request_queue *q, struct request *rq) | |||
106 | /* | 106 | /* |
107 | * remove rq from rbtree and fifo. | 107 | * remove rq from rbtree and fifo. |
108 | */ | 108 | */ |
109 | static void deadline_remove_request(request_queue_t *q, struct request *rq) | 109 | static void deadline_remove_request(struct request_queue *q, struct request *rq) |
110 | { | 110 | { |
111 | struct deadline_data *dd = q->elevator->elevator_data; | 111 | struct deadline_data *dd = q->elevator->elevator_data; |
112 | 112 | ||
@@ -115,7 +115,7 @@ static void deadline_remove_request(request_queue_t *q, struct request *rq) | |||
115 | } | 115 | } |
116 | 116 | ||
117 | static int | 117 | static int |
118 | deadline_merge(request_queue_t *q, struct request **req, struct bio *bio) | 118 | deadline_merge(struct request_queue *q, struct request **req, struct bio *bio) |
119 | { | 119 | { |
120 | struct deadline_data *dd = q->elevator->elevator_data; | 120 | struct deadline_data *dd = q->elevator->elevator_data; |
121 | struct request *__rq; | 121 | struct request *__rq; |
@@ -144,8 +144,8 @@ out: | |||
144 | return ret; | 144 | return ret; |
145 | } | 145 | } |
146 | 146 | ||
147 | static void deadline_merged_request(request_queue_t *q, struct request *req, | 147 | static void deadline_merged_request(struct request_queue *q, |
148 | int type) | 148 | struct request *req, int type) |
149 | { | 149 | { |
150 | struct deadline_data *dd = q->elevator->elevator_data; | 150 | struct deadline_data *dd = q->elevator->elevator_data; |
151 | 151 | ||
@@ -159,7 +159,7 @@ static void deadline_merged_request(request_queue_t *q, struct request *req, | |||
159 | } | 159 | } |
160 | 160 | ||
161 | static void | 161 | static void |
162 | deadline_merged_requests(request_queue_t *q, struct request *req, | 162 | deadline_merged_requests(struct request_queue *q, struct request *req, |
163 | struct request *next) | 163 | struct request *next) |
164 | { | 164 | { |
165 | /* | 165 | /* |
@@ -185,7 +185,7 @@ deadline_merged_requests(request_queue_t *q, struct request *req, | |||
185 | static inline void | 185 | static inline void |
186 | deadline_move_to_dispatch(struct deadline_data *dd, struct request *rq) | 186 | deadline_move_to_dispatch(struct deadline_data *dd, struct request *rq) |
187 | { | 187 | { |
188 | request_queue_t *q = rq->q; | 188 | struct request_queue *q = rq->q; |
189 | 189 | ||
190 | deadline_remove_request(q, rq); | 190 | deadline_remove_request(q, rq); |
191 | elv_dispatch_add_tail(q, rq); | 191 | elv_dispatch_add_tail(q, rq); |
@@ -236,7 +236,7 @@ static inline int deadline_check_fifo(struct deadline_data *dd, int ddir) | |||
236 | * deadline_dispatch_requests selects the best request according to | 236 | * deadline_dispatch_requests selects the best request according to |
237 | * read/write expire, fifo_batch, etc | 237 | * read/write expire, fifo_batch, etc |
238 | */ | 238 | */ |
239 | static int deadline_dispatch_requests(request_queue_t *q, int force) | 239 | static int deadline_dispatch_requests(struct request_queue *q, int force) |
240 | { | 240 | { |
241 | struct deadline_data *dd = q->elevator->elevator_data; | 241 | struct deadline_data *dd = q->elevator->elevator_data; |
242 | const int reads = !list_empty(&dd->fifo_list[READ]); | 242 | const int reads = !list_empty(&dd->fifo_list[READ]); |
@@ -335,7 +335,7 @@ dispatch_request: | |||
335 | return 1; | 335 | return 1; |
336 | } | 336 | } |
337 | 337 | ||
338 | static int deadline_queue_empty(request_queue_t *q) | 338 | static int deadline_queue_empty(struct request_queue *q) |
339 | { | 339 | { |
340 | struct deadline_data *dd = q->elevator->elevator_data; | 340 | struct deadline_data *dd = q->elevator->elevator_data; |
341 | 341 | ||
@@ -356,7 +356,7 @@ static void deadline_exit_queue(elevator_t *e) | |||
356 | /* | 356 | /* |
357 | * initialize elevator private data (deadline_data). | 357 | * initialize elevator private data (deadline_data). |
358 | */ | 358 | */ |
359 | static void *deadline_init_queue(request_queue_t *q) | 359 | static void *deadline_init_queue(struct request_queue *q) |
360 | { | 360 | { |
361 | struct deadline_data *dd; | 361 | struct deadline_data *dd; |
362 | 362 | ||
diff --git a/block/elevator.c b/block/elevator.c index d265963d1ed3..c6d153de9fd6 100644 --- a/block/elevator.c +++ b/block/elevator.c | |||
@@ -56,7 +56,7 @@ static const int elv_hash_shift = 6; | |||
56 | */ | 56 | */ |
57 | static int elv_iosched_allow_merge(struct request *rq, struct bio *bio) | 57 | static int elv_iosched_allow_merge(struct request *rq, struct bio *bio) |
58 | { | 58 | { |
59 | request_queue_t *q = rq->q; | 59 | struct request_queue *q = rq->q; |
60 | elevator_t *e = q->elevator; | 60 | elevator_t *e = q->elevator; |
61 | 61 | ||
62 | if (e->ops->elevator_allow_merge_fn) | 62 | if (e->ops->elevator_allow_merge_fn) |
@@ -141,12 +141,13 @@ static struct elevator_type *elevator_get(const char *name) | |||
141 | return e; | 141 | return e; |
142 | } | 142 | } |
143 | 143 | ||
144 | static void *elevator_init_queue(request_queue_t *q, struct elevator_queue *eq) | 144 | static void *elevator_init_queue(struct request_queue *q, |
145 | struct elevator_queue *eq) | ||
145 | { | 146 | { |
146 | return eq->ops->elevator_init_fn(q); | 147 | return eq->ops->elevator_init_fn(q); |
147 | } | 148 | } |
148 | 149 | ||
149 | static void elevator_attach(request_queue_t *q, struct elevator_queue *eq, | 150 | static void elevator_attach(struct request_queue *q, struct elevator_queue *eq, |
150 | void *data) | 151 | void *data) |
151 | { | 152 | { |
152 | q->elevator = eq; | 153 | q->elevator = eq; |
@@ -172,7 +173,8 @@ __setup("elevator=", elevator_setup); | |||
172 | 173 | ||
173 | static struct kobj_type elv_ktype; | 174 | static struct kobj_type elv_ktype; |
174 | 175 | ||
175 | static elevator_t *elevator_alloc(request_queue_t *q, struct elevator_type *e) | 176 | static elevator_t *elevator_alloc(struct request_queue *q, |
177 | struct elevator_type *e) | ||
176 | { | 178 | { |
177 | elevator_t *eq; | 179 | elevator_t *eq; |
178 | int i; | 180 | int i; |
@@ -212,7 +214,7 @@ static void elevator_release(struct kobject *kobj) | |||
212 | kfree(e); | 214 | kfree(e); |
213 | } | 215 | } |
214 | 216 | ||
215 | int elevator_init(request_queue_t *q, char *name) | 217 | int elevator_init(struct request_queue *q, char *name) |
216 | { | 218 | { |
217 | struct elevator_type *e = NULL; | 219 | struct elevator_type *e = NULL; |
218 | struct elevator_queue *eq; | 220 | struct elevator_queue *eq; |
@@ -264,7 +266,7 @@ void elevator_exit(elevator_t *e) | |||
264 | 266 | ||
265 | EXPORT_SYMBOL(elevator_exit); | 267 | EXPORT_SYMBOL(elevator_exit); |
266 | 268 | ||
267 | static void elv_activate_rq(request_queue_t *q, struct request *rq) | 269 | static void elv_activate_rq(struct request_queue *q, struct request *rq) |
268 | { | 270 | { |
269 | elevator_t *e = q->elevator; | 271 | elevator_t *e = q->elevator; |
270 | 272 | ||
@@ -272,7 +274,7 @@ static void elv_activate_rq(request_queue_t *q, struct request *rq) | |||
272 | e->ops->elevator_activate_req_fn(q, rq); | 274 | e->ops->elevator_activate_req_fn(q, rq); |
273 | } | 275 | } |
274 | 276 | ||
275 | static void elv_deactivate_rq(request_queue_t *q, struct request *rq) | 277 | static void elv_deactivate_rq(struct request_queue *q, struct request *rq) |
276 | { | 278 | { |
277 | elevator_t *e = q->elevator; | 279 | elevator_t *e = q->elevator; |
278 | 280 | ||
@@ -285,13 +287,13 @@ static inline void __elv_rqhash_del(struct request *rq) | |||
285 | hlist_del_init(&rq->hash); | 287 | hlist_del_init(&rq->hash); |
286 | } | 288 | } |
287 | 289 | ||
288 | static void elv_rqhash_del(request_queue_t *q, struct request *rq) | 290 | static void elv_rqhash_del(struct request_queue *q, struct request *rq) |
289 | { | 291 | { |
290 | if (ELV_ON_HASH(rq)) | 292 | if (ELV_ON_HASH(rq)) |
291 | __elv_rqhash_del(rq); | 293 | __elv_rqhash_del(rq); |
292 | } | 294 | } |
293 | 295 | ||
294 | static void elv_rqhash_add(request_queue_t *q, struct request *rq) | 296 | static void elv_rqhash_add(struct request_queue *q, struct request *rq) |
295 | { | 297 | { |
296 | elevator_t *e = q->elevator; | 298 | elevator_t *e = q->elevator; |
297 | 299 | ||
@@ -299,13 +301,13 @@ static void elv_rqhash_add(request_queue_t *q, struct request *rq) | |||
299 | hlist_add_head(&rq->hash, &e->hash[ELV_HASH_FN(rq_hash_key(rq))]); | 301 | hlist_add_head(&rq->hash, &e->hash[ELV_HASH_FN(rq_hash_key(rq))]); |
300 | } | 302 | } |
301 | 303 | ||
302 | static void elv_rqhash_reposition(request_queue_t *q, struct request *rq) | 304 | static void elv_rqhash_reposition(struct request_queue *q, struct request *rq) |
303 | { | 305 | { |
304 | __elv_rqhash_del(rq); | 306 | __elv_rqhash_del(rq); |
305 | elv_rqhash_add(q, rq); | 307 | elv_rqhash_add(q, rq); |
306 | } | 308 | } |
307 | 309 | ||
308 | static struct request *elv_rqhash_find(request_queue_t *q, sector_t offset) | 310 | static struct request *elv_rqhash_find(struct request_queue *q, sector_t offset) |
309 | { | 311 | { |
310 | elevator_t *e = q->elevator; | 312 | elevator_t *e = q->elevator; |
311 | struct hlist_head *hash_list = &e->hash[ELV_HASH_FN(offset)]; | 313 | struct hlist_head *hash_list = &e->hash[ELV_HASH_FN(offset)]; |
@@ -391,7 +393,7 @@ EXPORT_SYMBOL(elv_rb_find); | |||
391 | * entry. rq is sort insted into the dispatch queue. To be used by | 393 | * entry. rq is sort insted into the dispatch queue. To be used by |
392 | * specific elevators. | 394 | * specific elevators. |
393 | */ | 395 | */ |
394 | void elv_dispatch_sort(request_queue_t *q, struct request *rq) | 396 | void elv_dispatch_sort(struct request_queue *q, struct request *rq) |
395 | { | 397 | { |
396 | sector_t boundary; | 398 | sector_t boundary; |
397 | struct list_head *entry; | 399 | struct list_head *entry; |
@@ -449,7 +451,7 @@ void elv_dispatch_add_tail(struct request_queue *q, struct request *rq) | |||
449 | 451 | ||
450 | EXPORT_SYMBOL(elv_dispatch_add_tail); | 452 | EXPORT_SYMBOL(elv_dispatch_add_tail); |
451 | 453 | ||
452 | int elv_merge(request_queue_t *q, struct request **req, struct bio *bio) | 454 | int elv_merge(struct request_queue *q, struct request **req, struct bio *bio) |
453 | { | 455 | { |
454 | elevator_t *e = q->elevator; | 456 | elevator_t *e = q->elevator; |
455 | struct request *__rq; | 457 | struct request *__rq; |
@@ -481,7 +483,7 @@ int elv_merge(request_queue_t *q, struct request **req, struct bio *bio) | |||
481 | return ELEVATOR_NO_MERGE; | 483 | return ELEVATOR_NO_MERGE; |
482 | } | 484 | } |
483 | 485 | ||
484 | void elv_merged_request(request_queue_t *q, struct request *rq, int type) | 486 | void elv_merged_request(struct request_queue *q, struct request *rq, int type) |
485 | { | 487 | { |
486 | elevator_t *e = q->elevator; | 488 | elevator_t *e = q->elevator; |
487 | 489 | ||
@@ -494,7 +496,7 @@ void elv_merged_request(request_queue_t *q, struct request *rq, int type) | |||
494 | q->last_merge = rq; | 496 | q->last_merge = rq; |
495 | } | 497 | } |
496 | 498 | ||
497 | void elv_merge_requests(request_queue_t *q, struct request *rq, | 499 | void elv_merge_requests(struct request_queue *q, struct request *rq, |
498 | struct request *next) | 500 | struct request *next) |
499 | { | 501 | { |
500 | elevator_t *e = q->elevator; | 502 | elevator_t *e = q->elevator; |
@@ -509,7 +511,7 @@ void elv_merge_requests(request_queue_t *q, struct request *rq, | |||
509 | q->last_merge = rq; | 511 | q->last_merge = rq; |
510 | } | 512 | } |
511 | 513 | ||
512 | void elv_requeue_request(request_queue_t *q, struct request *rq) | 514 | void elv_requeue_request(struct request_queue *q, struct request *rq) |
513 | { | 515 | { |
514 | /* | 516 | /* |
515 | * it already went through dequeue, we need to decrement the | 517 | * it already went through dequeue, we need to decrement the |
@@ -526,7 +528,7 @@ void elv_requeue_request(request_queue_t *q, struct request *rq) | |||
526 | elv_insert(q, rq, ELEVATOR_INSERT_REQUEUE); | 528 | elv_insert(q, rq, ELEVATOR_INSERT_REQUEUE); |
527 | } | 529 | } |
528 | 530 | ||
529 | static void elv_drain_elevator(request_queue_t *q) | 531 | static void elv_drain_elevator(struct request_queue *q) |
530 | { | 532 | { |
531 | static int printed; | 533 | static int printed; |
532 | while (q->elevator->ops->elevator_dispatch_fn(q, 1)) | 534 | while (q->elevator->ops->elevator_dispatch_fn(q, 1)) |
@@ -540,7 +542,7 @@ static void elv_drain_elevator(request_queue_t *q) | |||
540 | } | 542 | } |
541 | } | 543 | } |
542 | 544 | ||
543 | void elv_insert(request_queue_t *q, struct request *rq, int where) | 545 | void elv_insert(struct request_queue *q, struct request *rq, int where) |
544 | { | 546 | { |
545 | struct list_head *pos; | 547 | struct list_head *pos; |
546 | unsigned ordseq; | 548 | unsigned ordseq; |
@@ -638,7 +640,7 @@ void elv_insert(request_queue_t *q, struct request *rq, int where) | |||
638 | } | 640 | } |
639 | } | 641 | } |
640 | 642 | ||
641 | void __elv_add_request(request_queue_t *q, struct request *rq, int where, | 643 | void __elv_add_request(struct request_queue *q, struct request *rq, int where, |
642 | int plug) | 644 | int plug) |
643 | { | 645 | { |
644 | if (q->ordcolor) | 646 | if (q->ordcolor) |
@@ -676,7 +678,7 @@ void __elv_add_request(request_queue_t *q, struct request *rq, int where, | |||
676 | 678 | ||
677 | EXPORT_SYMBOL(__elv_add_request); | 679 | EXPORT_SYMBOL(__elv_add_request); |
678 | 680 | ||
679 | void elv_add_request(request_queue_t *q, struct request *rq, int where, | 681 | void elv_add_request(struct request_queue *q, struct request *rq, int where, |
680 | int plug) | 682 | int plug) |
681 | { | 683 | { |
682 | unsigned long flags; | 684 | unsigned long flags; |
@@ -688,7 +690,7 @@ void elv_add_request(request_queue_t *q, struct request *rq, int where, | |||
688 | 690 | ||
689 | EXPORT_SYMBOL(elv_add_request); | 691 | EXPORT_SYMBOL(elv_add_request); |
690 | 692 | ||
691 | static inline struct request *__elv_next_request(request_queue_t *q) | 693 | static inline struct request *__elv_next_request(struct request_queue *q) |
692 | { | 694 | { |
693 | struct request *rq; | 695 | struct request *rq; |
694 | 696 | ||
@@ -704,7 +706,7 @@ static inline struct request *__elv_next_request(request_queue_t *q) | |||
704 | } | 706 | } |
705 | } | 707 | } |
706 | 708 | ||
707 | struct request *elv_next_request(request_queue_t *q) | 709 | struct request *elv_next_request(struct request_queue *q) |
708 | { | 710 | { |
709 | struct request *rq; | 711 | struct request *rq; |
710 | int ret; | 712 | int ret; |
@@ -770,7 +772,7 @@ struct request *elv_next_request(request_queue_t *q) | |||
770 | 772 | ||
771 | EXPORT_SYMBOL(elv_next_request); | 773 | EXPORT_SYMBOL(elv_next_request); |
772 | 774 | ||
773 | void elv_dequeue_request(request_queue_t *q, struct request *rq) | 775 | void elv_dequeue_request(struct request_queue *q, struct request *rq) |
774 | { | 776 | { |
775 | BUG_ON(list_empty(&rq->queuelist)); | 777 | BUG_ON(list_empty(&rq->queuelist)); |
776 | BUG_ON(ELV_ON_HASH(rq)); | 778 | BUG_ON(ELV_ON_HASH(rq)); |
@@ -788,7 +790,7 @@ void elv_dequeue_request(request_queue_t *q, struct request *rq) | |||
788 | 790 | ||
789 | EXPORT_SYMBOL(elv_dequeue_request); | 791 | EXPORT_SYMBOL(elv_dequeue_request); |
790 | 792 | ||
791 | int elv_queue_empty(request_queue_t *q) | 793 | int elv_queue_empty(struct request_queue *q) |
792 | { | 794 | { |
793 | elevator_t *e = q->elevator; | 795 | elevator_t *e = q->elevator; |
794 | 796 | ||
@@ -803,7 +805,7 @@ int elv_queue_empty(request_queue_t *q) | |||
803 | 805 | ||
804 | EXPORT_SYMBOL(elv_queue_empty); | 806 | EXPORT_SYMBOL(elv_queue_empty); |
805 | 807 | ||
806 | struct request *elv_latter_request(request_queue_t *q, struct request *rq) | 808 | struct request *elv_latter_request(struct request_queue *q, struct request *rq) |
807 | { | 809 | { |
808 | elevator_t *e = q->elevator; | 810 | elevator_t *e = q->elevator; |
809 | 811 | ||
@@ -812,7 +814,7 @@ struct request *elv_latter_request(request_queue_t *q, struct request *rq) | |||
812 | return NULL; | 814 | return NULL; |
813 | } | 815 | } |
814 | 816 | ||
815 | struct request *elv_former_request(request_queue_t *q, struct request *rq) | 817 | struct request *elv_former_request(struct request_queue *q, struct request *rq) |
816 | { | 818 | { |
817 | elevator_t *e = q->elevator; | 819 | elevator_t *e = q->elevator; |
818 | 820 | ||
@@ -821,7 +823,7 @@ struct request *elv_former_request(request_queue_t *q, struct request *rq) | |||
821 | return NULL; | 823 | return NULL; |
822 | } | 824 | } |
823 | 825 | ||
824 | int elv_set_request(request_queue_t *q, struct request *rq, gfp_t gfp_mask) | 826 | int elv_set_request(struct request_queue *q, struct request *rq, gfp_t gfp_mask) |
825 | { | 827 | { |
826 | elevator_t *e = q->elevator; | 828 | elevator_t *e = q->elevator; |
827 | 829 | ||
@@ -832,7 +834,7 @@ int elv_set_request(request_queue_t *q, struct request *rq, gfp_t gfp_mask) | |||
832 | return 0; | 834 | return 0; |
833 | } | 835 | } |
834 | 836 | ||
835 | void elv_put_request(request_queue_t *q, struct request *rq) | 837 | void elv_put_request(struct request_queue *q, struct request *rq) |
836 | { | 838 | { |
837 | elevator_t *e = q->elevator; | 839 | elevator_t *e = q->elevator; |
838 | 840 | ||
@@ -840,7 +842,7 @@ void elv_put_request(request_queue_t *q, struct request *rq) | |||
840 | e->ops->elevator_put_req_fn(rq); | 842 | e->ops->elevator_put_req_fn(rq); |
841 | } | 843 | } |
842 | 844 | ||
843 | int elv_may_queue(request_queue_t *q, int rw) | 845 | int elv_may_queue(struct request_queue *q, int rw) |
844 | { | 846 | { |
845 | elevator_t *e = q->elevator; | 847 | elevator_t *e = q->elevator; |
846 | 848 | ||
@@ -850,7 +852,7 @@ int elv_may_queue(request_queue_t *q, int rw) | |||
850 | return ELV_MQUEUE_MAY; | 852 | return ELV_MQUEUE_MAY; |
851 | } | 853 | } |
852 | 854 | ||
853 | void elv_completed_request(request_queue_t *q, struct request *rq) | 855 | void elv_completed_request(struct request_queue *q, struct request *rq) |
854 | { | 856 | { |
855 | elevator_t *e = q->elevator; | 857 | elevator_t *e = q->elevator; |
856 | 858 | ||
@@ -1006,7 +1008,7 @@ EXPORT_SYMBOL_GPL(elv_unregister); | |||
1006 | * need for the new one. this way we have a chance of going back to the old | 1008 | * need for the new one. this way we have a chance of going back to the old |
1007 | * one, if the new one fails init for some reason. | 1009 | * one, if the new one fails init for some reason. |
1008 | */ | 1010 | */ |
1009 | static int elevator_switch(request_queue_t *q, struct elevator_type *new_e) | 1011 | static int elevator_switch(struct request_queue *q, struct elevator_type *new_e) |
1010 | { | 1012 | { |
1011 | elevator_t *old_elevator, *e; | 1013 | elevator_t *old_elevator, *e; |
1012 | void *data; | 1014 | void *data; |
@@ -1078,7 +1080,8 @@ fail_register: | |||
1078 | return 0; | 1080 | return 0; |
1079 | } | 1081 | } |
1080 | 1082 | ||
1081 | ssize_t elv_iosched_store(request_queue_t *q, const char *name, size_t count) | 1083 | ssize_t elv_iosched_store(struct request_queue *q, const char *name, |
1084 | size_t count) | ||
1082 | { | 1085 | { |
1083 | char elevator_name[ELV_NAME_MAX]; | 1086 | char elevator_name[ELV_NAME_MAX]; |
1084 | size_t len; | 1087 | size_t len; |
@@ -1107,7 +1110,7 @@ ssize_t elv_iosched_store(request_queue_t *q, const char *name, size_t count) | |||
1107 | return count; | 1110 | return count; |
1108 | } | 1111 | } |
1109 | 1112 | ||
1110 | ssize_t elv_iosched_show(request_queue_t *q, char *name) | 1113 | ssize_t elv_iosched_show(struct request_queue *q, char *name) |
1111 | { | 1114 | { |
1112 | elevator_t *e = q->elevator; | 1115 | elevator_t *e = q->elevator; |
1113 | struct elevator_type *elv = e->elevator_type; | 1116 | struct elevator_type *elv = e->elevator_type; |
@@ -1127,7 +1130,8 @@ ssize_t elv_iosched_show(request_queue_t *q, char *name) | |||
1127 | return len; | 1130 | return len; |
1128 | } | 1131 | } |
1129 | 1132 | ||
1130 | struct request *elv_rb_former_request(request_queue_t *q, struct request *rq) | 1133 | struct request *elv_rb_former_request(struct request_queue *q, |
1134 | struct request *rq) | ||
1131 | { | 1135 | { |
1132 | struct rb_node *rbprev = rb_prev(&rq->rb_node); | 1136 | struct rb_node *rbprev = rb_prev(&rq->rb_node); |
1133 | 1137 | ||
@@ -1139,7 +1143,8 @@ struct request *elv_rb_former_request(request_queue_t *q, struct request *rq) | |||
1139 | 1143 | ||
1140 | EXPORT_SYMBOL(elv_rb_former_request); | 1144 | EXPORT_SYMBOL(elv_rb_former_request); |
1141 | 1145 | ||
1142 | struct request *elv_rb_latter_request(request_queue_t *q, struct request *rq) | 1146 | struct request *elv_rb_latter_request(struct request_queue *q, |
1147 | struct request *rq) | ||
1143 | { | 1148 | { |
1144 | struct rb_node *rbnext = rb_next(&rq->rb_node); | 1149 | struct rb_node *rbnext = rb_next(&rq->rb_node); |
1145 | 1150 | ||
diff --git a/block/ll_rw_blk.c b/block/ll_rw_blk.c index 66056ca5e631..8c2caff87cc3 100644 --- a/block/ll_rw_blk.c +++ b/block/ll_rw_blk.c | |||
@@ -40,7 +40,7 @@ static void blk_unplug_work(struct work_struct *work); | |||
40 | static void blk_unplug_timeout(unsigned long data); | 40 | static void blk_unplug_timeout(unsigned long data); |
41 | static void drive_stat_acct(struct request *rq, int nr_sectors, int new_io); | 41 | static void drive_stat_acct(struct request *rq, int nr_sectors, int new_io); |
42 | static void init_request_from_bio(struct request *req, struct bio *bio); | 42 | static void init_request_from_bio(struct request *req, struct bio *bio); |
43 | static int __make_request(request_queue_t *q, struct bio *bio); | 43 | static int __make_request(struct request_queue *q, struct bio *bio); |
44 | static struct io_context *current_io_context(gfp_t gfp_flags, int node); | 44 | static struct io_context *current_io_context(gfp_t gfp_flags, int node); |
45 | 45 | ||
46 | /* | 46 | /* |
@@ -121,7 +121,7 @@ static void blk_queue_congestion_threshold(struct request_queue *q) | |||
121 | struct backing_dev_info *blk_get_backing_dev_info(struct block_device *bdev) | 121 | struct backing_dev_info *blk_get_backing_dev_info(struct block_device *bdev) |
122 | { | 122 | { |
123 | struct backing_dev_info *ret = NULL; | 123 | struct backing_dev_info *ret = NULL; |
124 | request_queue_t *q = bdev_get_queue(bdev); | 124 | struct request_queue *q = bdev_get_queue(bdev); |
125 | 125 | ||
126 | if (q) | 126 | if (q) |
127 | ret = &q->backing_dev_info; | 127 | ret = &q->backing_dev_info; |
@@ -140,7 +140,7 @@ EXPORT_SYMBOL(blk_get_backing_dev_info); | |||
140 | * cdb from the request data for instance. | 140 | * cdb from the request data for instance. |
141 | * | 141 | * |
142 | */ | 142 | */ |
143 | void blk_queue_prep_rq(request_queue_t *q, prep_rq_fn *pfn) | 143 | void blk_queue_prep_rq(struct request_queue *q, prep_rq_fn *pfn) |
144 | { | 144 | { |
145 | q->prep_rq_fn = pfn; | 145 | q->prep_rq_fn = pfn; |
146 | } | 146 | } |
@@ -163,14 +163,14 @@ EXPORT_SYMBOL(blk_queue_prep_rq); | |||
163 | * no merge_bvec_fn is defined for a queue, and only the fixed limits are | 163 | * no merge_bvec_fn is defined for a queue, and only the fixed limits are |
164 | * honored. | 164 | * honored. |
165 | */ | 165 | */ |
166 | void blk_queue_merge_bvec(request_queue_t *q, merge_bvec_fn *mbfn) | 166 | void blk_queue_merge_bvec(struct request_queue *q, merge_bvec_fn *mbfn) |
167 | { | 167 | { |
168 | q->merge_bvec_fn = mbfn; | 168 | q->merge_bvec_fn = mbfn; |
169 | } | 169 | } |
170 | 170 | ||
171 | EXPORT_SYMBOL(blk_queue_merge_bvec); | 171 | EXPORT_SYMBOL(blk_queue_merge_bvec); |
172 | 172 | ||
173 | void blk_queue_softirq_done(request_queue_t *q, softirq_done_fn *fn) | 173 | void blk_queue_softirq_done(struct request_queue *q, softirq_done_fn *fn) |
174 | { | 174 | { |
175 | q->softirq_done_fn = fn; | 175 | q->softirq_done_fn = fn; |
176 | } | 176 | } |
@@ -199,7 +199,7 @@ EXPORT_SYMBOL(blk_queue_softirq_done); | |||
199 | * __bio_kmap_atomic() to get a temporary kernel mapping, or by calling | 199 | * __bio_kmap_atomic() to get a temporary kernel mapping, or by calling |
200 | * blk_queue_bounce() to create a buffer in normal memory. | 200 | * blk_queue_bounce() to create a buffer in normal memory. |
201 | **/ | 201 | **/ |
202 | void blk_queue_make_request(request_queue_t * q, make_request_fn * mfn) | 202 | void blk_queue_make_request(struct request_queue * q, make_request_fn * mfn) |
203 | { | 203 | { |
204 | /* | 204 | /* |
205 | * set defaults | 205 | * set defaults |
@@ -235,7 +235,7 @@ void blk_queue_make_request(request_queue_t * q, make_request_fn * mfn) | |||
235 | 235 | ||
236 | EXPORT_SYMBOL(blk_queue_make_request); | 236 | EXPORT_SYMBOL(blk_queue_make_request); |
237 | 237 | ||
238 | static void rq_init(request_queue_t *q, struct request *rq) | 238 | static void rq_init(struct request_queue *q, struct request *rq) |
239 | { | 239 | { |
240 | INIT_LIST_HEAD(&rq->queuelist); | 240 | INIT_LIST_HEAD(&rq->queuelist); |
241 | INIT_LIST_HEAD(&rq->donelist); | 241 | INIT_LIST_HEAD(&rq->donelist); |
@@ -272,7 +272,7 @@ static void rq_init(request_queue_t *q, struct request *rq) | |||
272 | * feature should call this function and indicate so. | 272 | * feature should call this function and indicate so. |
273 | * | 273 | * |
274 | **/ | 274 | **/ |
275 | int blk_queue_ordered(request_queue_t *q, unsigned ordered, | 275 | int blk_queue_ordered(struct request_queue *q, unsigned ordered, |
276 | prepare_flush_fn *prepare_flush_fn) | 276 | prepare_flush_fn *prepare_flush_fn) |
277 | { | 277 | { |
278 | if (ordered & (QUEUE_ORDERED_PREFLUSH | QUEUE_ORDERED_POSTFLUSH) && | 278 | if (ordered & (QUEUE_ORDERED_PREFLUSH | QUEUE_ORDERED_POSTFLUSH) && |
@@ -311,7 +311,7 @@ EXPORT_SYMBOL(blk_queue_ordered); | |||
311 | * to the block layer by defining it through this call. | 311 | * to the block layer by defining it through this call. |
312 | * | 312 | * |
313 | **/ | 313 | **/ |
314 | void blk_queue_issue_flush_fn(request_queue_t *q, issue_flush_fn *iff) | 314 | void blk_queue_issue_flush_fn(struct request_queue *q, issue_flush_fn *iff) |
315 | { | 315 | { |
316 | q->issue_flush_fn = iff; | 316 | q->issue_flush_fn = iff; |
317 | } | 317 | } |
@@ -321,7 +321,7 @@ EXPORT_SYMBOL(blk_queue_issue_flush_fn); | |||
321 | /* | 321 | /* |
322 | * Cache flushing for ordered writes handling | 322 | * Cache flushing for ordered writes handling |
323 | */ | 323 | */ |
324 | inline unsigned blk_ordered_cur_seq(request_queue_t *q) | 324 | inline unsigned blk_ordered_cur_seq(struct request_queue *q) |
325 | { | 325 | { |
326 | if (!q->ordseq) | 326 | if (!q->ordseq) |
327 | return 0; | 327 | return 0; |
@@ -330,7 +330,7 @@ inline unsigned blk_ordered_cur_seq(request_queue_t *q) | |||
330 | 330 | ||
331 | unsigned blk_ordered_req_seq(struct request *rq) | 331 | unsigned blk_ordered_req_seq(struct request *rq) |
332 | { | 332 | { |
333 | request_queue_t *q = rq->q; | 333 | struct request_queue *q = rq->q; |
334 | 334 | ||
335 | BUG_ON(q->ordseq == 0); | 335 | BUG_ON(q->ordseq == 0); |
336 | 336 | ||
@@ -357,7 +357,7 @@ unsigned blk_ordered_req_seq(struct request *rq) | |||
357 | return QUEUE_ORDSEQ_DONE; | 357 | return QUEUE_ORDSEQ_DONE; |
358 | } | 358 | } |
359 | 359 | ||
360 | void blk_ordered_complete_seq(request_queue_t *q, unsigned seq, int error) | 360 | void blk_ordered_complete_seq(struct request_queue *q, unsigned seq, int error) |
361 | { | 361 | { |
362 | struct request *rq; | 362 | struct request *rq; |
363 | int uptodate; | 363 | int uptodate; |
@@ -401,7 +401,7 @@ static void post_flush_end_io(struct request *rq, int error) | |||
401 | blk_ordered_complete_seq(rq->q, QUEUE_ORDSEQ_POSTFLUSH, error); | 401 | blk_ordered_complete_seq(rq->q, QUEUE_ORDSEQ_POSTFLUSH, error); |
402 | } | 402 | } |
403 | 403 | ||
404 | static void queue_flush(request_queue_t *q, unsigned which) | 404 | static void queue_flush(struct request_queue *q, unsigned which) |
405 | { | 405 | { |
406 | struct request *rq; | 406 | struct request *rq; |
407 | rq_end_io_fn *end_io; | 407 | rq_end_io_fn *end_io; |
@@ -425,7 +425,7 @@ static void queue_flush(request_queue_t *q, unsigned which) | |||
425 | elv_insert(q, rq, ELEVATOR_INSERT_FRONT); | 425 | elv_insert(q, rq, ELEVATOR_INSERT_FRONT); |
426 | } | 426 | } |
427 | 427 | ||
428 | static inline struct request *start_ordered(request_queue_t *q, | 428 | static inline struct request *start_ordered(struct request_queue *q, |
429 | struct request *rq) | 429 | struct request *rq) |
430 | { | 430 | { |
431 | q->bi_size = 0; | 431 | q->bi_size = 0; |
@@ -476,7 +476,7 @@ static inline struct request *start_ordered(request_queue_t *q, | |||
476 | return rq; | 476 | return rq; |
477 | } | 477 | } |
478 | 478 | ||
479 | int blk_do_ordered(request_queue_t *q, struct request **rqp) | 479 | int blk_do_ordered(struct request_queue *q, struct request **rqp) |
480 | { | 480 | { |
481 | struct request *rq = *rqp; | 481 | struct request *rq = *rqp; |
482 | int is_barrier = blk_fs_request(rq) && blk_barrier_rq(rq); | 482 | int is_barrier = blk_fs_request(rq) && blk_barrier_rq(rq); |
@@ -527,7 +527,7 @@ int blk_do_ordered(request_queue_t *q, struct request **rqp) | |||
527 | 527 | ||
528 | static int flush_dry_bio_endio(struct bio *bio, unsigned int bytes, int error) | 528 | static int flush_dry_bio_endio(struct bio *bio, unsigned int bytes, int error) |
529 | { | 529 | { |
530 | request_queue_t *q = bio->bi_private; | 530 | struct request_queue *q = bio->bi_private; |
531 | 531 | ||
532 | /* | 532 | /* |
533 | * This is dry run, restore bio_sector and size. We'll finish | 533 | * This is dry run, restore bio_sector and size. We'll finish |
@@ -551,7 +551,7 @@ static int flush_dry_bio_endio(struct bio *bio, unsigned int bytes, int error) | |||
551 | static int ordered_bio_endio(struct request *rq, struct bio *bio, | 551 | static int ordered_bio_endio(struct request *rq, struct bio *bio, |
552 | unsigned int nbytes, int error) | 552 | unsigned int nbytes, int error) |
553 | { | 553 | { |
554 | request_queue_t *q = rq->q; | 554 | struct request_queue *q = rq->q; |
555 | bio_end_io_t *endio; | 555 | bio_end_io_t *endio; |
556 | void *private; | 556 | void *private; |
557 | 557 | ||
@@ -588,7 +588,7 @@ static int ordered_bio_endio(struct request *rq, struct bio *bio, | |||
588 | * blk_queue_bounce_limit to have lower memory pages allocated as bounce | 588 | * blk_queue_bounce_limit to have lower memory pages allocated as bounce |
589 | * buffers for doing I/O to pages residing above @page. | 589 | * buffers for doing I/O to pages residing above @page. |
590 | **/ | 590 | **/ |
591 | void blk_queue_bounce_limit(request_queue_t *q, u64 dma_addr) | 591 | void blk_queue_bounce_limit(struct request_queue *q, u64 dma_addr) |
592 | { | 592 | { |
593 | unsigned long bounce_pfn = dma_addr >> PAGE_SHIFT; | 593 | unsigned long bounce_pfn = dma_addr >> PAGE_SHIFT; |
594 | int dma = 0; | 594 | int dma = 0; |
@@ -624,7 +624,7 @@ EXPORT_SYMBOL(blk_queue_bounce_limit); | |||
624 | * Enables a low level driver to set an upper limit on the size of | 624 | * Enables a low level driver to set an upper limit on the size of |
625 | * received requests. | 625 | * received requests. |
626 | **/ | 626 | **/ |
627 | void blk_queue_max_sectors(request_queue_t *q, unsigned int max_sectors) | 627 | void blk_queue_max_sectors(struct request_queue *q, unsigned int max_sectors) |
628 | { | 628 | { |
629 | if ((max_sectors << 9) < PAGE_CACHE_SIZE) { | 629 | if ((max_sectors << 9) < PAGE_CACHE_SIZE) { |
630 | max_sectors = 1 << (PAGE_CACHE_SHIFT - 9); | 630 | max_sectors = 1 << (PAGE_CACHE_SHIFT - 9); |
@@ -651,7 +651,8 @@ EXPORT_SYMBOL(blk_queue_max_sectors); | |||
651 | * physical data segments in a request. This would be the largest sized | 651 | * physical data segments in a request. This would be the largest sized |
652 | * scatter list the driver could handle. | 652 | * scatter list the driver could handle. |
653 | **/ | 653 | **/ |
654 | void blk_queue_max_phys_segments(request_queue_t *q, unsigned short max_segments) | 654 | void blk_queue_max_phys_segments(struct request_queue *q, |
655 | unsigned short max_segments) | ||
655 | { | 656 | { |
656 | if (!max_segments) { | 657 | if (!max_segments) { |
657 | max_segments = 1; | 658 | max_segments = 1; |
@@ -674,7 +675,8 @@ EXPORT_SYMBOL(blk_queue_max_phys_segments); | |||
674 | * address/length pairs the host adapter can actually give as once | 675 | * address/length pairs the host adapter can actually give as once |
675 | * to the device. | 676 | * to the device. |
676 | **/ | 677 | **/ |
677 | void blk_queue_max_hw_segments(request_queue_t *q, unsigned short max_segments) | 678 | void blk_queue_max_hw_segments(struct request_queue *q, |
679 | unsigned short max_segments) | ||
678 | { | 680 | { |
679 | if (!max_segments) { | 681 | if (!max_segments) { |
680 | max_segments = 1; | 682 | max_segments = 1; |
@@ -695,7 +697,7 @@ EXPORT_SYMBOL(blk_queue_max_hw_segments); | |||
695 | * Enables a low level driver to set an upper limit on the size of a | 697 | * Enables a low level driver to set an upper limit on the size of a |
696 | * coalesced segment | 698 | * coalesced segment |
697 | **/ | 699 | **/ |
698 | void blk_queue_max_segment_size(request_queue_t *q, unsigned int max_size) | 700 | void blk_queue_max_segment_size(struct request_queue *q, unsigned int max_size) |
699 | { | 701 | { |
700 | if (max_size < PAGE_CACHE_SIZE) { | 702 | if (max_size < PAGE_CACHE_SIZE) { |
701 | max_size = PAGE_CACHE_SIZE; | 703 | max_size = PAGE_CACHE_SIZE; |
@@ -718,7 +720,7 @@ EXPORT_SYMBOL(blk_queue_max_segment_size); | |||
718 | * even internal read-modify-write operations). Usually the default | 720 | * even internal read-modify-write operations). Usually the default |
719 | * of 512 covers most hardware. | 721 | * of 512 covers most hardware. |
720 | **/ | 722 | **/ |
721 | void blk_queue_hardsect_size(request_queue_t *q, unsigned short size) | 723 | void blk_queue_hardsect_size(struct request_queue *q, unsigned short size) |
722 | { | 724 | { |
723 | q->hardsect_size = size; | 725 | q->hardsect_size = size; |
724 | } | 726 | } |
@@ -735,7 +737,7 @@ EXPORT_SYMBOL(blk_queue_hardsect_size); | |||
735 | * @t: the stacking driver (top) | 737 | * @t: the stacking driver (top) |
736 | * @b: the underlying device (bottom) | 738 | * @b: the underlying device (bottom) |
737 | **/ | 739 | **/ |
738 | void blk_queue_stack_limits(request_queue_t *t, request_queue_t *b) | 740 | void blk_queue_stack_limits(struct request_queue *t, struct request_queue *b) |
739 | { | 741 | { |
740 | /* zero is "infinity" */ | 742 | /* zero is "infinity" */ |
741 | t->max_sectors = min_not_zero(t->max_sectors,b->max_sectors); | 743 | t->max_sectors = min_not_zero(t->max_sectors,b->max_sectors); |
@@ -756,7 +758,7 @@ EXPORT_SYMBOL(blk_queue_stack_limits); | |||
756 | * @q: the request queue for the device | 758 | * @q: the request queue for the device |
757 | * @mask: the memory boundary mask | 759 | * @mask: the memory boundary mask |
758 | **/ | 760 | **/ |
759 | void blk_queue_segment_boundary(request_queue_t *q, unsigned long mask) | 761 | void blk_queue_segment_boundary(struct request_queue *q, unsigned long mask) |
760 | { | 762 | { |
761 | if (mask < PAGE_CACHE_SIZE - 1) { | 763 | if (mask < PAGE_CACHE_SIZE - 1) { |
762 | mask = PAGE_CACHE_SIZE - 1; | 764 | mask = PAGE_CACHE_SIZE - 1; |
@@ -778,7 +780,7 @@ EXPORT_SYMBOL(blk_queue_segment_boundary); | |||
778 | * this is used when buiding direct io requests for the queue. | 780 | * this is used when buiding direct io requests for the queue. |
779 | * | 781 | * |
780 | **/ | 782 | **/ |
781 | void blk_queue_dma_alignment(request_queue_t *q, int mask) | 783 | void blk_queue_dma_alignment(struct request_queue *q, int mask) |
782 | { | 784 | { |
783 | q->dma_alignment = mask; | 785 | q->dma_alignment = mask; |
784 | } | 786 | } |
@@ -796,7 +798,7 @@ EXPORT_SYMBOL(blk_queue_dma_alignment); | |||
796 | * | 798 | * |
797 | * no locks need be held. | 799 | * no locks need be held. |
798 | **/ | 800 | **/ |
799 | struct request *blk_queue_find_tag(request_queue_t *q, int tag) | 801 | struct request *blk_queue_find_tag(struct request_queue *q, int tag) |
800 | { | 802 | { |
801 | return blk_map_queue_find_tag(q->queue_tags, tag); | 803 | return blk_map_queue_find_tag(q->queue_tags, tag); |
802 | } | 804 | } |
@@ -840,7 +842,7 @@ static int __blk_free_tags(struct blk_queue_tag *bqt) | |||
840 | * blk_cleanup_queue() will take care of calling this function, if tagging | 842 | * blk_cleanup_queue() will take care of calling this function, if tagging |
841 | * has been used. So there's no need to call this directly. | 843 | * has been used. So there's no need to call this directly. |
842 | **/ | 844 | **/ |
843 | static void __blk_queue_free_tags(request_queue_t *q) | 845 | static void __blk_queue_free_tags(struct request_queue *q) |
844 | { | 846 | { |
845 | struct blk_queue_tag *bqt = q->queue_tags; | 847 | struct blk_queue_tag *bqt = q->queue_tags; |
846 | 848 | ||
@@ -877,7 +879,7 @@ EXPORT_SYMBOL(blk_free_tags); | |||
877 | * This is used to disabled tagged queuing to a device, yet leave | 879 | * This is used to disabled tagged queuing to a device, yet leave |
878 | * queue in function. | 880 | * queue in function. |
879 | **/ | 881 | **/ |
880 | void blk_queue_free_tags(request_queue_t *q) | 882 | void blk_queue_free_tags(struct request_queue *q) |
881 | { | 883 | { |
882 | clear_bit(QUEUE_FLAG_QUEUED, &q->queue_flags); | 884 | clear_bit(QUEUE_FLAG_QUEUED, &q->queue_flags); |
883 | } | 885 | } |
@@ -885,7 +887,7 @@ void blk_queue_free_tags(request_queue_t *q) | |||
885 | EXPORT_SYMBOL(blk_queue_free_tags); | 887 | EXPORT_SYMBOL(blk_queue_free_tags); |
886 | 888 | ||
887 | static int | 889 | static int |
888 | init_tag_map(request_queue_t *q, struct blk_queue_tag *tags, int depth) | 890 | init_tag_map(struct request_queue *q, struct blk_queue_tag *tags, int depth) |
889 | { | 891 | { |
890 | struct request **tag_index; | 892 | struct request **tag_index; |
891 | unsigned long *tag_map; | 893 | unsigned long *tag_map; |
@@ -955,7 +957,7 @@ EXPORT_SYMBOL(blk_init_tags); | |||
955 | * @depth: the maximum queue depth supported | 957 | * @depth: the maximum queue depth supported |
956 | * @tags: the tag to use | 958 | * @tags: the tag to use |
957 | **/ | 959 | **/ |
958 | int blk_queue_init_tags(request_queue_t *q, int depth, | 960 | int blk_queue_init_tags(struct request_queue *q, int depth, |
959 | struct blk_queue_tag *tags) | 961 | struct blk_queue_tag *tags) |
960 | { | 962 | { |
961 | int rc; | 963 | int rc; |
@@ -996,7 +998,7 @@ EXPORT_SYMBOL(blk_queue_init_tags); | |||
996 | * Notes: | 998 | * Notes: |
997 | * Must be called with the queue lock held. | 999 | * Must be called with the queue lock held. |
998 | **/ | 1000 | **/ |
999 | int blk_queue_resize_tags(request_queue_t *q, int new_depth) | 1001 | int blk_queue_resize_tags(struct request_queue *q, int new_depth) |
1000 | { | 1002 | { |
1001 | struct blk_queue_tag *bqt = q->queue_tags; | 1003 | struct blk_queue_tag *bqt = q->queue_tags; |
1002 | struct request **tag_index; | 1004 | struct request **tag_index; |
@@ -1059,7 +1061,7 @@ EXPORT_SYMBOL(blk_queue_resize_tags); | |||
1059 | * Notes: | 1061 | * Notes: |
1060 | * queue lock must be held. | 1062 | * queue lock must be held. |
1061 | **/ | 1063 | **/ |
1062 | void blk_queue_end_tag(request_queue_t *q, struct request *rq) | 1064 | void blk_queue_end_tag(struct request_queue *q, struct request *rq) |
1063 | { | 1065 | { |
1064 | struct blk_queue_tag *bqt = q->queue_tags; | 1066 | struct blk_queue_tag *bqt = q->queue_tags; |
1065 | int tag = rq->tag; | 1067 | int tag = rq->tag; |
@@ -1111,7 +1113,7 @@ EXPORT_SYMBOL(blk_queue_end_tag); | |||
1111 | * Notes: | 1113 | * Notes: |
1112 | * queue lock must be held. | 1114 | * queue lock must be held. |
1113 | **/ | 1115 | **/ |
1114 | int blk_queue_start_tag(request_queue_t *q, struct request *rq) | 1116 | int blk_queue_start_tag(struct request_queue *q, struct request *rq) |
1115 | { | 1117 | { |
1116 | struct blk_queue_tag *bqt = q->queue_tags; | 1118 | struct blk_queue_tag *bqt = q->queue_tags; |
1117 | int tag; | 1119 | int tag; |
@@ -1158,7 +1160,7 @@ EXPORT_SYMBOL(blk_queue_start_tag); | |||
1158 | * Notes: | 1160 | * Notes: |
1159 | * queue lock must be held. | 1161 | * queue lock must be held. |
1160 | **/ | 1162 | **/ |
1161 | void blk_queue_invalidate_tags(request_queue_t *q) | 1163 | void blk_queue_invalidate_tags(struct request_queue *q) |
1162 | { | 1164 | { |
1163 | struct blk_queue_tag *bqt = q->queue_tags; | 1165 | struct blk_queue_tag *bqt = q->queue_tags; |
1164 | struct list_head *tmp, *n; | 1166 | struct list_head *tmp, *n; |
@@ -1205,7 +1207,7 @@ void blk_dump_rq_flags(struct request *rq, char *msg) | |||
1205 | 1207 | ||
1206 | EXPORT_SYMBOL(blk_dump_rq_flags); | 1208 | EXPORT_SYMBOL(blk_dump_rq_flags); |
1207 | 1209 | ||
1208 | void blk_recount_segments(request_queue_t *q, struct bio *bio) | 1210 | void blk_recount_segments(struct request_queue *q, struct bio *bio) |
1209 | { | 1211 | { |
1210 | struct bio_vec *bv, *bvprv = NULL; | 1212 | struct bio_vec *bv, *bvprv = NULL; |
1211 | int i, nr_phys_segs, nr_hw_segs, seg_size, hw_seg_size, cluster; | 1213 | int i, nr_phys_segs, nr_hw_segs, seg_size, hw_seg_size, cluster; |
@@ -1267,7 +1269,7 @@ new_hw_segment: | |||
1267 | } | 1269 | } |
1268 | EXPORT_SYMBOL(blk_recount_segments); | 1270 | EXPORT_SYMBOL(blk_recount_segments); |
1269 | 1271 | ||
1270 | static int blk_phys_contig_segment(request_queue_t *q, struct bio *bio, | 1272 | static int blk_phys_contig_segment(struct request_queue *q, struct bio *bio, |
1271 | struct bio *nxt) | 1273 | struct bio *nxt) |
1272 | { | 1274 | { |
1273 | if (!(q->queue_flags & (1 << QUEUE_FLAG_CLUSTER))) | 1275 | if (!(q->queue_flags & (1 << QUEUE_FLAG_CLUSTER))) |
@@ -1288,7 +1290,7 @@ static int blk_phys_contig_segment(request_queue_t *q, struct bio *bio, | |||
1288 | return 0; | 1290 | return 0; |
1289 | } | 1291 | } |
1290 | 1292 | ||
1291 | static int blk_hw_contig_segment(request_queue_t *q, struct bio *bio, | 1293 | static int blk_hw_contig_segment(struct request_queue *q, struct bio *bio, |
1292 | struct bio *nxt) | 1294 | struct bio *nxt) |
1293 | { | 1295 | { |
1294 | if (unlikely(!bio_flagged(bio, BIO_SEG_VALID))) | 1296 | if (unlikely(!bio_flagged(bio, BIO_SEG_VALID))) |
@@ -1308,7 +1310,8 @@ static int blk_hw_contig_segment(request_queue_t *q, struct bio *bio, | |||
1308 | * map a request to scatterlist, return number of sg entries setup. Caller | 1310 | * map a request to scatterlist, return number of sg entries setup. Caller |
1309 | * must make sure sg can hold rq->nr_phys_segments entries | 1311 | * must make sure sg can hold rq->nr_phys_segments entries |
1310 | */ | 1312 | */ |
1311 | int blk_rq_map_sg(request_queue_t *q, struct request *rq, struct scatterlist *sg) | 1313 | int blk_rq_map_sg(struct request_queue *q, struct request *rq, |
1314 | struct scatterlist *sg) | ||
1312 | { | 1315 | { |
1313 | struct bio_vec *bvec, *bvprv; | 1316 | struct bio_vec *bvec, *bvprv; |
1314 | struct bio *bio; | 1317 | struct bio *bio; |
@@ -1361,7 +1364,7 @@ EXPORT_SYMBOL(blk_rq_map_sg); | |||
1361 | * specific ones if so desired | 1364 | * specific ones if so desired |
1362 | */ | 1365 | */ |
1363 | 1366 | ||
1364 | static inline int ll_new_mergeable(request_queue_t *q, | 1367 | static inline int ll_new_mergeable(struct request_queue *q, |
1365 | struct request *req, | 1368 | struct request *req, |
1366 | struct bio *bio) | 1369 | struct bio *bio) |
1367 | { | 1370 | { |
@@ -1382,7 +1385,7 @@ static inline int ll_new_mergeable(request_queue_t *q, | |||
1382 | return 1; | 1385 | return 1; |
1383 | } | 1386 | } |
1384 | 1387 | ||
1385 | static inline int ll_new_hw_segment(request_queue_t *q, | 1388 | static inline int ll_new_hw_segment(struct request_queue *q, |
1386 | struct request *req, | 1389 | struct request *req, |
1387 | struct bio *bio) | 1390 | struct bio *bio) |
1388 | { | 1391 | { |
@@ -1406,7 +1409,7 @@ static inline int ll_new_hw_segment(request_queue_t *q, | |||
1406 | return 1; | 1409 | return 1; |
1407 | } | 1410 | } |
1408 | 1411 | ||
1409 | int ll_back_merge_fn(request_queue_t *q, struct request *req, struct bio *bio) | 1412 | int ll_back_merge_fn(struct request_queue *q, struct request *req, struct bio *bio) |
1410 | { | 1413 | { |
1411 | unsigned short max_sectors; | 1414 | unsigned short max_sectors; |
1412 | int len; | 1415 | int len; |
@@ -1444,7 +1447,7 @@ int ll_back_merge_fn(request_queue_t *q, struct request *req, struct bio *bio) | |||
1444 | } | 1447 | } |
1445 | EXPORT_SYMBOL(ll_back_merge_fn); | 1448 | EXPORT_SYMBOL(ll_back_merge_fn); |
1446 | 1449 | ||
1447 | static int ll_front_merge_fn(request_queue_t *q, struct request *req, | 1450 | static int ll_front_merge_fn(struct request_queue *q, struct request *req, |
1448 | struct bio *bio) | 1451 | struct bio *bio) |
1449 | { | 1452 | { |
1450 | unsigned short max_sectors; | 1453 | unsigned short max_sectors; |
@@ -1483,7 +1486,7 @@ static int ll_front_merge_fn(request_queue_t *q, struct request *req, | |||
1483 | return ll_new_hw_segment(q, req, bio); | 1486 | return ll_new_hw_segment(q, req, bio); |
1484 | } | 1487 | } |
1485 | 1488 | ||
1486 | static int ll_merge_requests_fn(request_queue_t *q, struct request *req, | 1489 | static int ll_merge_requests_fn(struct request_queue *q, struct request *req, |
1487 | struct request *next) | 1490 | struct request *next) |
1488 | { | 1491 | { |
1489 | int total_phys_segments; | 1492 | int total_phys_segments; |
@@ -1539,7 +1542,7 @@ static int ll_merge_requests_fn(request_queue_t *q, struct request *req, | |||
1539 | * This is called with interrupts off and no requests on the queue and | 1542 | * This is called with interrupts off and no requests on the queue and |
1540 | * with the queue lock held. | 1543 | * with the queue lock held. |
1541 | */ | 1544 | */ |
1542 | void blk_plug_device(request_queue_t *q) | 1545 | void blk_plug_device(struct request_queue *q) |
1543 | { | 1546 | { |
1544 | WARN_ON(!irqs_disabled()); | 1547 | WARN_ON(!irqs_disabled()); |
1545 | 1548 | ||
@@ -1562,7 +1565,7 @@ EXPORT_SYMBOL(blk_plug_device); | |||
1562 | * remove the queue from the plugged list, if present. called with | 1565 | * remove the queue from the plugged list, if present. called with |
1563 | * queue lock held and interrupts disabled. | 1566 | * queue lock held and interrupts disabled. |
1564 | */ | 1567 | */ |
1565 | int blk_remove_plug(request_queue_t *q) | 1568 | int blk_remove_plug(struct request_queue *q) |
1566 | { | 1569 | { |
1567 | WARN_ON(!irqs_disabled()); | 1570 | WARN_ON(!irqs_disabled()); |
1568 | 1571 | ||
@@ -1578,7 +1581,7 @@ EXPORT_SYMBOL(blk_remove_plug); | |||
1578 | /* | 1581 | /* |
1579 | * remove the plug and let it rip.. | 1582 | * remove the plug and let it rip.. |
1580 | */ | 1583 | */ |
1581 | void __generic_unplug_device(request_queue_t *q) | 1584 | void __generic_unplug_device(struct request_queue *q) |
1582 | { | 1585 | { |
1583 | if (unlikely(blk_queue_stopped(q))) | 1586 | if (unlikely(blk_queue_stopped(q))) |
1584 | return; | 1587 | return; |
@@ -1592,7 +1595,7 @@ EXPORT_SYMBOL(__generic_unplug_device); | |||
1592 | 1595 | ||
1593 | /** | 1596 | /** |
1594 | * generic_unplug_device - fire a request queue | 1597 | * generic_unplug_device - fire a request queue |
1595 | * @q: The &request_queue_t in question | 1598 | * @q: The &struct request_queue in question |
1596 | * | 1599 | * |
1597 | * Description: | 1600 | * Description: |
1598 | * Linux uses plugging to build bigger requests queues before letting | 1601 | * Linux uses plugging to build bigger requests queues before letting |
@@ -1601,7 +1604,7 @@ EXPORT_SYMBOL(__generic_unplug_device); | |||
1601 | * gets unplugged, the request_fn defined for the queue is invoked and | 1604 | * gets unplugged, the request_fn defined for the queue is invoked and |
1602 | * transfers started. | 1605 | * transfers started. |
1603 | **/ | 1606 | **/ |
1604 | void generic_unplug_device(request_queue_t *q) | 1607 | void generic_unplug_device(struct request_queue *q) |
1605 | { | 1608 | { |
1606 | spin_lock_irq(q->queue_lock); | 1609 | spin_lock_irq(q->queue_lock); |
1607 | __generic_unplug_device(q); | 1610 | __generic_unplug_device(q); |
@@ -1612,7 +1615,7 @@ EXPORT_SYMBOL(generic_unplug_device); | |||
1612 | static void blk_backing_dev_unplug(struct backing_dev_info *bdi, | 1615 | static void blk_backing_dev_unplug(struct backing_dev_info *bdi, |
1613 | struct page *page) | 1616 | struct page *page) |
1614 | { | 1617 | { |
1615 | request_queue_t *q = bdi->unplug_io_data; | 1618 | struct request_queue *q = bdi->unplug_io_data; |
1616 | 1619 | ||
1617 | /* | 1620 | /* |
1618 | * devices don't necessarily have an ->unplug_fn defined | 1621 | * devices don't necessarily have an ->unplug_fn defined |
@@ -1627,7 +1630,8 @@ static void blk_backing_dev_unplug(struct backing_dev_info *bdi, | |||
1627 | 1630 | ||
1628 | static void blk_unplug_work(struct work_struct *work) | 1631 | static void blk_unplug_work(struct work_struct *work) |
1629 | { | 1632 | { |
1630 | request_queue_t *q = container_of(work, request_queue_t, unplug_work); | 1633 | struct request_queue *q = |
1634 | container_of(work, struct request_queue, unplug_work); | ||
1631 | 1635 | ||
1632 | blk_add_trace_pdu_int(q, BLK_TA_UNPLUG_IO, NULL, | 1636 | blk_add_trace_pdu_int(q, BLK_TA_UNPLUG_IO, NULL, |
1633 | q->rq.count[READ] + q->rq.count[WRITE]); | 1637 | q->rq.count[READ] + q->rq.count[WRITE]); |
@@ -1637,7 +1641,7 @@ static void blk_unplug_work(struct work_struct *work) | |||
1637 | 1641 | ||
1638 | static void blk_unplug_timeout(unsigned long data) | 1642 | static void blk_unplug_timeout(unsigned long data) |
1639 | { | 1643 | { |
1640 | request_queue_t *q = (request_queue_t *)data; | 1644 | struct request_queue *q = (struct request_queue *)data; |
1641 | 1645 | ||
1642 | blk_add_trace_pdu_int(q, BLK_TA_UNPLUG_TIMER, NULL, | 1646 | blk_add_trace_pdu_int(q, BLK_TA_UNPLUG_TIMER, NULL, |
1643 | q->rq.count[READ] + q->rq.count[WRITE]); | 1647 | q->rq.count[READ] + q->rq.count[WRITE]); |
@@ -1647,14 +1651,14 @@ static void blk_unplug_timeout(unsigned long data) | |||
1647 | 1651 | ||
1648 | /** | 1652 | /** |
1649 | * blk_start_queue - restart a previously stopped queue | 1653 | * blk_start_queue - restart a previously stopped queue |
1650 | * @q: The &request_queue_t in question | 1654 | * @q: The &struct request_queue in question |
1651 | * | 1655 | * |
1652 | * Description: | 1656 | * Description: |
1653 | * blk_start_queue() will clear the stop flag on the queue, and call | 1657 | * blk_start_queue() will clear the stop flag on the queue, and call |
1654 | * the request_fn for the queue if it was in a stopped state when | 1658 | * the request_fn for the queue if it was in a stopped state when |
1655 | * entered. Also see blk_stop_queue(). Queue lock must be held. | 1659 | * entered. Also see blk_stop_queue(). Queue lock must be held. |
1656 | **/ | 1660 | **/ |
1657 | void blk_start_queue(request_queue_t *q) | 1661 | void blk_start_queue(struct request_queue *q) |
1658 | { | 1662 | { |
1659 | WARN_ON(!irqs_disabled()); | 1663 | WARN_ON(!irqs_disabled()); |
1660 | 1664 | ||
@@ -1677,7 +1681,7 @@ EXPORT_SYMBOL(blk_start_queue); | |||
1677 | 1681 | ||
1678 | /** | 1682 | /** |
1679 | * blk_stop_queue - stop a queue | 1683 | * blk_stop_queue - stop a queue |
1680 | * @q: The &request_queue_t in question | 1684 | * @q: The &struct request_queue in question |
1681 | * | 1685 | * |
1682 | * Description: | 1686 | * Description: |
1683 | * The Linux block layer assumes that a block driver will consume all | 1687 | * The Linux block layer assumes that a block driver will consume all |
@@ -1689,7 +1693,7 @@ EXPORT_SYMBOL(blk_start_queue); | |||
1689 | * the driver has signalled it's ready to go again. This happens by calling | 1693 | * the driver has signalled it's ready to go again. This happens by calling |
1690 | * blk_start_queue() to restart queue operations. Queue lock must be held. | 1694 | * blk_start_queue() to restart queue operations. Queue lock must be held. |
1691 | **/ | 1695 | **/ |
1692 | void blk_stop_queue(request_queue_t *q) | 1696 | void blk_stop_queue(struct request_queue *q) |
1693 | { | 1697 | { |
1694 | blk_remove_plug(q); | 1698 | blk_remove_plug(q); |
1695 | set_bit(QUEUE_FLAG_STOPPED, &q->queue_flags); | 1699 | set_bit(QUEUE_FLAG_STOPPED, &q->queue_flags); |
@@ -1746,7 +1750,7 @@ void blk_run_queue(struct request_queue *q) | |||
1746 | EXPORT_SYMBOL(blk_run_queue); | 1750 | EXPORT_SYMBOL(blk_run_queue); |
1747 | 1751 | ||
1748 | /** | 1752 | /** |
1749 | * blk_cleanup_queue: - release a &request_queue_t when it is no longer needed | 1753 | * blk_cleanup_queue: - release a &struct request_queue when it is no longer needed |
1750 | * @kobj: the kobj belonging of the request queue to be released | 1754 | * @kobj: the kobj belonging of the request queue to be released |
1751 | * | 1755 | * |
1752 | * Description: | 1756 | * Description: |
@@ -1762,7 +1766,8 @@ EXPORT_SYMBOL(blk_run_queue); | |||
1762 | **/ | 1766 | **/ |
1763 | static void blk_release_queue(struct kobject *kobj) | 1767 | static void blk_release_queue(struct kobject *kobj) |
1764 | { | 1768 | { |
1765 | request_queue_t *q = container_of(kobj, struct request_queue, kobj); | 1769 | struct request_queue *q = |
1770 | container_of(kobj, struct request_queue, kobj); | ||
1766 | struct request_list *rl = &q->rq; | 1771 | struct request_list *rl = &q->rq; |
1767 | 1772 | ||
1768 | blk_sync_queue(q); | 1773 | blk_sync_queue(q); |
@@ -1778,13 +1783,13 @@ static void blk_release_queue(struct kobject *kobj) | |||
1778 | kmem_cache_free(requestq_cachep, q); | 1783 | kmem_cache_free(requestq_cachep, q); |
1779 | } | 1784 | } |
1780 | 1785 | ||
1781 | void blk_put_queue(request_queue_t *q) | 1786 | void blk_put_queue(struct request_queue *q) |
1782 | { | 1787 | { |
1783 | kobject_put(&q->kobj); | 1788 | kobject_put(&q->kobj); |
1784 | } | 1789 | } |
1785 | EXPORT_SYMBOL(blk_put_queue); | 1790 | EXPORT_SYMBOL(blk_put_queue); |
1786 | 1791 | ||
1787 | void blk_cleanup_queue(request_queue_t * q) | 1792 | void blk_cleanup_queue(struct request_queue * q) |
1788 | { | 1793 | { |
1789 | mutex_lock(&q->sysfs_lock); | 1794 | mutex_lock(&q->sysfs_lock); |
1790 | set_bit(QUEUE_FLAG_DEAD, &q->queue_flags); | 1795 | set_bit(QUEUE_FLAG_DEAD, &q->queue_flags); |
@@ -1798,7 +1803,7 @@ void blk_cleanup_queue(request_queue_t * q) | |||
1798 | 1803 | ||
1799 | EXPORT_SYMBOL(blk_cleanup_queue); | 1804 | EXPORT_SYMBOL(blk_cleanup_queue); |
1800 | 1805 | ||
1801 | static int blk_init_free_list(request_queue_t *q) | 1806 | static int blk_init_free_list(struct request_queue *q) |
1802 | { | 1807 | { |
1803 | struct request_list *rl = &q->rq; | 1808 | struct request_list *rl = &q->rq; |
1804 | 1809 | ||
@@ -1817,7 +1822,7 @@ static int blk_init_free_list(request_queue_t *q) | |||
1817 | return 0; | 1822 | return 0; |
1818 | } | 1823 | } |
1819 | 1824 | ||
1820 | request_queue_t *blk_alloc_queue(gfp_t gfp_mask) | 1825 | struct request_queue *blk_alloc_queue(gfp_t gfp_mask) |
1821 | { | 1826 | { |
1822 | return blk_alloc_queue_node(gfp_mask, -1); | 1827 | return blk_alloc_queue_node(gfp_mask, -1); |
1823 | } | 1828 | } |
@@ -1825,9 +1830,9 @@ EXPORT_SYMBOL(blk_alloc_queue); | |||
1825 | 1830 | ||
1826 | static struct kobj_type queue_ktype; | 1831 | static struct kobj_type queue_ktype; |
1827 | 1832 | ||
1828 | request_queue_t *blk_alloc_queue_node(gfp_t gfp_mask, int node_id) | 1833 | struct request_queue *blk_alloc_queue_node(gfp_t gfp_mask, int node_id) |
1829 | { | 1834 | { |
1830 | request_queue_t *q; | 1835 | struct request_queue *q; |
1831 | 1836 | ||
1832 | q = kmem_cache_alloc_node(requestq_cachep, | 1837 | q = kmem_cache_alloc_node(requestq_cachep, |
1833 | gfp_mask | __GFP_ZERO, node_id); | 1838 | gfp_mask | __GFP_ZERO, node_id); |
@@ -1882,16 +1887,16 @@ EXPORT_SYMBOL(blk_alloc_queue_node); | |||
1882 | * when the block device is deactivated (such as at module unload). | 1887 | * when the block device is deactivated (such as at module unload). |
1883 | **/ | 1888 | **/ |
1884 | 1889 | ||
1885 | request_queue_t *blk_init_queue(request_fn_proc *rfn, spinlock_t *lock) | 1890 | struct request_queue *blk_init_queue(request_fn_proc *rfn, spinlock_t *lock) |
1886 | { | 1891 | { |
1887 | return blk_init_queue_node(rfn, lock, -1); | 1892 | return blk_init_queue_node(rfn, lock, -1); |
1888 | } | 1893 | } |
1889 | EXPORT_SYMBOL(blk_init_queue); | 1894 | EXPORT_SYMBOL(blk_init_queue); |
1890 | 1895 | ||
1891 | request_queue_t * | 1896 | struct request_queue * |
1892 | blk_init_queue_node(request_fn_proc *rfn, spinlock_t *lock, int node_id) | 1897 | blk_init_queue_node(request_fn_proc *rfn, spinlock_t *lock, int node_id) |
1893 | { | 1898 | { |
1894 | request_queue_t *q = blk_alloc_queue_node(GFP_KERNEL, node_id); | 1899 | struct request_queue *q = blk_alloc_queue_node(GFP_KERNEL, node_id); |
1895 | 1900 | ||
1896 | if (!q) | 1901 | if (!q) |
1897 | return NULL; | 1902 | return NULL; |
@@ -1940,7 +1945,7 @@ blk_init_queue_node(request_fn_proc *rfn, spinlock_t *lock, int node_id) | |||
1940 | } | 1945 | } |
1941 | EXPORT_SYMBOL(blk_init_queue_node); | 1946 | EXPORT_SYMBOL(blk_init_queue_node); |
1942 | 1947 | ||
1943 | int blk_get_queue(request_queue_t *q) | 1948 | int blk_get_queue(struct request_queue *q) |
1944 | { | 1949 | { |
1945 | if (likely(!test_bit(QUEUE_FLAG_DEAD, &q->queue_flags))) { | 1950 | if (likely(!test_bit(QUEUE_FLAG_DEAD, &q->queue_flags))) { |
1946 | kobject_get(&q->kobj); | 1951 | kobject_get(&q->kobj); |
@@ -1952,7 +1957,7 @@ int blk_get_queue(request_queue_t *q) | |||
1952 | 1957 | ||
1953 | EXPORT_SYMBOL(blk_get_queue); | 1958 | EXPORT_SYMBOL(blk_get_queue); |
1954 | 1959 | ||
1955 | static inline void blk_free_request(request_queue_t *q, struct request *rq) | 1960 | static inline void blk_free_request(struct request_queue *q, struct request *rq) |
1956 | { | 1961 | { |
1957 | if (rq->cmd_flags & REQ_ELVPRIV) | 1962 | if (rq->cmd_flags & REQ_ELVPRIV) |
1958 | elv_put_request(q, rq); | 1963 | elv_put_request(q, rq); |
@@ -1960,7 +1965,7 @@ static inline void blk_free_request(request_queue_t *q, struct request *rq) | |||
1960 | } | 1965 | } |
1961 | 1966 | ||
1962 | static struct request * | 1967 | static struct request * |
1963 | blk_alloc_request(request_queue_t *q, int rw, int priv, gfp_t gfp_mask) | 1968 | blk_alloc_request(struct request_queue *q, int rw, int priv, gfp_t gfp_mask) |
1964 | { | 1969 | { |
1965 | struct request *rq = mempool_alloc(q->rq.rq_pool, gfp_mask); | 1970 | struct request *rq = mempool_alloc(q->rq.rq_pool, gfp_mask); |
1966 | 1971 | ||
@@ -1988,7 +1993,7 @@ blk_alloc_request(request_queue_t *q, int rw, int priv, gfp_t gfp_mask) | |||
1988 | * ioc_batching returns true if the ioc is a valid batching request and | 1993 | * ioc_batching returns true if the ioc is a valid batching request and |
1989 | * should be given priority access to a request. | 1994 | * should be given priority access to a request. |
1990 | */ | 1995 | */ |
1991 | static inline int ioc_batching(request_queue_t *q, struct io_context *ioc) | 1996 | static inline int ioc_batching(struct request_queue *q, struct io_context *ioc) |
1992 | { | 1997 | { |
1993 | if (!ioc) | 1998 | if (!ioc) |
1994 | return 0; | 1999 | return 0; |
@@ -2009,7 +2014,7 @@ static inline int ioc_batching(request_queue_t *q, struct io_context *ioc) | |||
2009 | * is the behaviour we want though - once it gets a wakeup it should be given | 2014 | * is the behaviour we want though - once it gets a wakeup it should be given |
2010 | * a nice run. | 2015 | * a nice run. |
2011 | */ | 2016 | */ |
2012 | static void ioc_set_batching(request_queue_t *q, struct io_context *ioc) | 2017 | static void ioc_set_batching(struct request_queue *q, struct io_context *ioc) |
2013 | { | 2018 | { |
2014 | if (!ioc || ioc_batching(q, ioc)) | 2019 | if (!ioc || ioc_batching(q, ioc)) |
2015 | return; | 2020 | return; |
@@ -2018,7 +2023,7 @@ static void ioc_set_batching(request_queue_t *q, struct io_context *ioc) | |||
2018 | ioc->last_waited = jiffies; | 2023 | ioc->last_waited = jiffies; |
2019 | } | 2024 | } |
2020 | 2025 | ||
2021 | static void __freed_request(request_queue_t *q, int rw) | 2026 | static void __freed_request(struct request_queue *q, int rw) |
2022 | { | 2027 | { |
2023 | struct request_list *rl = &q->rq; | 2028 | struct request_list *rl = &q->rq; |
2024 | 2029 | ||
@@ -2037,7 +2042,7 @@ static void __freed_request(request_queue_t *q, int rw) | |||
2037 | * A request has just been released. Account for it, update the full and | 2042 | * A request has just been released. Account for it, update the full and |
2038 | * congestion status, wake up any waiters. Called under q->queue_lock. | 2043 | * congestion status, wake up any waiters. Called under q->queue_lock. |
2039 | */ | 2044 | */ |
2040 | static void freed_request(request_queue_t *q, int rw, int priv) | 2045 | static void freed_request(struct request_queue *q, int rw, int priv) |
2041 | { | 2046 | { |
2042 | struct request_list *rl = &q->rq; | 2047 | struct request_list *rl = &q->rq; |
2043 | 2048 | ||
@@ -2057,7 +2062,7 @@ static void freed_request(request_queue_t *q, int rw, int priv) | |||
2057 | * Returns NULL on failure, with queue_lock held. | 2062 | * Returns NULL on failure, with queue_lock held. |
2058 | * Returns !NULL on success, with queue_lock *not held*. | 2063 | * Returns !NULL on success, with queue_lock *not held*. |
2059 | */ | 2064 | */ |
2060 | static struct request *get_request(request_queue_t *q, int rw_flags, | 2065 | static struct request *get_request(struct request_queue *q, int rw_flags, |
2061 | struct bio *bio, gfp_t gfp_mask) | 2066 | struct bio *bio, gfp_t gfp_mask) |
2062 | { | 2067 | { |
2063 | struct request *rq = NULL; | 2068 | struct request *rq = NULL; |
@@ -2162,7 +2167,7 @@ out: | |||
2162 | * | 2167 | * |
2163 | * Called with q->queue_lock held, and returns with it unlocked. | 2168 | * Called with q->queue_lock held, and returns with it unlocked. |
2164 | */ | 2169 | */ |
2165 | static struct request *get_request_wait(request_queue_t *q, int rw_flags, | 2170 | static struct request *get_request_wait(struct request_queue *q, int rw_flags, |
2166 | struct bio *bio) | 2171 | struct bio *bio) |
2167 | { | 2172 | { |
2168 | const int rw = rw_flags & 0x01; | 2173 | const int rw = rw_flags & 0x01; |
@@ -2204,7 +2209,7 @@ static struct request *get_request_wait(request_queue_t *q, int rw_flags, | |||
2204 | return rq; | 2209 | return rq; |
2205 | } | 2210 | } |
2206 | 2211 | ||
2207 | struct request *blk_get_request(request_queue_t *q, int rw, gfp_t gfp_mask) | 2212 | struct request *blk_get_request(struct request_queue *q, int rw, gfp_t gfp_mask) |
2208 | { | 2213 | { |
2209 | struct request *rq; | 2214 | struct request *rq; |
2210 | 2215 | ||
@@ -2234,7 +2239,7 @@ EXPORT_SYMBOL(blk_get_request); | |||
2234 | * | 2239 | * |
2235 | * The queue lock must be held with interrupts disabled. | 2240 | * The queue lock must be held with interrupts disabled. |
2236 | */ | 2241 | */ |
2237 | void blk_start_queueing(request_queue_t *q) | 2242 | void blk_start_queueing(struct request_queue *q) |
2238 | { | 2243 | { |
2239 | if (!blk_queue_plugged(q)) | 2244 | if (!blk_queue_plugged(q)) |
2240 | q->request_fn(q); | 2245 | q->request_fn(q); |
@@ -2253,7 +2258,7 @@ EXPORT_SYMBOL(blk_start_queueing); | |||
2253 | * more, when that condition happens we need to put the request back | 2258 | * more, when that condition happens we need to put the request back |
2254 | * on the queue. Must be called with queue lock held. | 2259 | * on the queue. Must be called with queue lock held. |
2255 | */ | 2260 | */ |
2256 | void blk_requeue_request(request_queue_t *q, struct request *rq) | 2261 | void blk_requeue_request(struct request_queue *q, struct request *rq) |
2257 | { | 2262 | { |
2258 | blk_add_trace_rq(q, rq, BLK_TA_REQUEUE); | 2263 | blk_add_trace_rq(q, rq, BLK_TA_REQUEUE); |
2259 | 2264 | ||
@@ -2284,7 +2289,7 @@ EXPORT_SYMBOL(blk_requeue_request); | |||
2284 | * of the queue for things like a QUEUE_FULL message from a device, or a | 2289 | * of the queue for things like a QUEUE_FULL message from a device, or a |
2285 | * host that is unable to accept a particular command. | 2290 | * host that is unable to accept a particular command. |
2286 | */ | 2291 | */ |
2287 | void blk_insert_request(request_queue_t *q, struct request *rq, | 2292 | void blk_insert_request(struct request_queue *q, struct request *rq, |
2288 | int at_head, void *data) | 2293 | int at_head, void *data) |
2289 | { | 2294 | { |
2290 | int where = at_head ? ELEVATOR_INSERT_FRONT : ELEVATOR_INSERT_BACK; | 2295 | int where = at_head ? ELEVATOR_INSERT_FRONT : ELEVATOR_INSERT_BACK; |
@@ -2330,7 +2335,7 @@ static int __blk_rq_unmap_user(struct bio *bio) | |||
2330 | return ret; | 2335 | return ret; |
2331 | } | 2336 | } |
2332 | 2337 | ||
2333 | static int __blk_rq_map_user(request_queue_t *q, struct request *rq, | 2338 | static int __blk_rq_map_user(struct request_queue *q, struct request *rq, |
2334 | void __user *ubuf, unsigned int len) | 2339 | void __user *ubuf, unsigned int len) |
2335 | { | 2340 | { |
2336 | unsigned long uaddr; | 2341 | unsigned long uaddr; |
@@ -2403,8 +2408,8 @@ unmap_bio: | |||
2403 | * original bio must be passed back in to blk_rq_unmap_user() for proper | 2408 | * original bio must be passed back in to blk_rq_unmap_user() for proper |
2404 | * unmapping. | 2409 | * unmapping. |
2405 | */ | 2410 | */ |
2406 | int blk_rq_map_user(request_queue_t *q, struct request *rq, void __user *ubuf, | 2411 | int blk_rq_map_user(struct request_queue *q, struct request *rq, |
2407 | unsigned long len) | 2412 | void __user *ubuf, unsigned long len) |
2408 | { | 2413 | { |
2409 | unsigned long bytes_read = 0; | 2414 | unsigned long bytes_read = 0; |
2410 | struct bio *bio = NULL; | 2415 | struct bio *bio = NULL; |
@@ -2470,7 +2475,7 @@ EXPORT_SYMBOL(blk_rq_map_user); | |||
2470 | * original bio must be passed back in to blk_rq_unmap_user() for proper | 2475 | * original bio must be passed back in to blk_rq_unmap_user() for proper |
2471 | * unmapping. | 2476 | * unmapping. |
2472 | */ | 2477 | */ |
2473 | int blk_rq_map_user_iov(request_queue_t *q, struct request *rq, | 2478 | int blk_rq_map_user_iov(struct request_queue *q, struct request *rq, |
2474 | struct sg_iovec *iov, int iov_count, unsigned int len) | 2479 | struct sg_iovec *iov, int iov_count, unsigned int len) |
2475 | { | 2480 | { |
2476 | struct bio *bio; | 2481 | struct bio *bio; |
@@ -2540,7 +2545,7 @@ EXPORT_SYMBOL(blk_rq_unmap_user); | |||
2540 | * @len: length of user data | 2545 | * @len: length of user data |
2541 | * @gfp_mask: memory allocation flags | 2546 | * @gfp_mask: memory allocation flags |
2542 | */ | 2547 | */ |
2543 | int blk_rq_map_kern(request_queue_t *q, struct request *rq, void *kbuf, | 2548 | int blk_rq_map_kern(struct request_queue *q, struct request *rq, void *kbuf, |
2544 | unsigned int len, gfp_t gfp_mask) | 2549 | unsigned int len, gfp_t gfp_mask) |
2545 | { | 2550 | { |
2546 | struct bio *bio; | 2551 | struct bio *bio; |
@@ -2577,7 +2582,7 @@ EXPORT_SYMBOL(blk_rq_map_kern); | |||
2577 | * Insert a fully prepared request at the back of the io scheduler queue | 2582 | * Insert a fully prepared request at the back of the io scheduler queue |
2578 | * for execution. Don't wait for completion. | 2583 | * for execution. Don't wait for completion. |
2579 | */ | 2584 | */ |
2580 | void blk_execute_rq_nowait(request_queue_t *q, struct gendisk *bd_disk, | 2585 | void blk_execute_rq_nowait(struct request_queue *q, struct gendisk *bd_disk, |
2581 | struct request *rq, int at_head, | 2586 | struct request *rq, int at_head, |
2582 | rq_end_io_fn *done) | 2587 | rq_end_io_fn *done) |
2583 | { | 2588 | { |
@@ -2605,7 +2610,7 @@ EXPORT_SYMBOL_GPL(blk_execute_rq_nowait); | |||
2605 | * Insert a fully prepared request at the back of the io scheduler queue | 2610 | * Insert a fully prepared request at the back of the io scheduler queue |
2606 | * for execution and wait for completion. | 2611 | * for execution and wait for completion. |
2607 | */ | 2612 | */ |
2608 | int blk_execute_rq(request_queue_t *q, struct gendisk *bd_disk, | 2613 | int blk_execute_rq(struct request_queue *q, struct gendisk *bd_disk, |
2609 | struct request *rq, int at_head) | 2614 | struct request *rq, int at_head) |
2610 | { | 2615 | { |
2611 | DECLARE_COMPLETION_ONSTACK(wait); | 2616 | DECLARE_COMPLETION_ONSTACK(wait); |
@@ -2648,7 +2653,7 @@ EXPORT_SYMBOL(blk_execute_rq); | |||
2648 | */ | 2653 | */ |
2649 | int blkdev_issue_flush(struct block_device *bdev, sector_t *error_sector) | 2654 | int blkdev_issue_flush(struct block_device *bdev, sector_t *error_sector) |
2650 | { | 2655 | { |
2651 | request_queue_t *q; | 2656 | struct request_queue *q; |
2652 | 2657 | ||
2653 | if (bdev->bd_disk == NULL) | 2658 | if (bdev->bd_disk == NULL) |
2654 | return -ENXIO; | 2659 | return -ENXIO; |
@@ -2684,7 +2689,7 @@ static void drive_stat_acct(struct request *rq, int nr_sectors, int new_io) | |||
2684 | * queue lock is held and interrupts disabled, as we muck with the | 2689 | * queue lock is held and interrupts disabled, as we muck with the |
2685 | * request queue list. | 2690 | * request queue list. |
2686 | */ | 2691 | */ |
2687 | static inline void add_request(request_queue_t * q, struct request * req) | 2692 | static inline void add_request(struct request_queue * q, struct request * req) |
2688 | { | 2693 | { |
2689 | drive_stat_acct(req, req->nr_sectors, 1); | 2694 | drive_stat_acct(req, req->nr_sectors, 1); |
2690 | 2695 | ||
@@ -2730,7 +2735,7 @@ EXPORT_SYMBOL_GPL(disk_round_stats); | |||
2730 | /* | 2735 | /* |
2731 | * queue lock must be held | 2736 | * queue lock must be held |
2732 | */ | 2737 | */ |
2733 | void __blk_put_request(request_queue_t *q, struct request *req) | 2738 | void __blk_put_request(struct request_queue *q, struct request *req) |
2734 | { | 2739 | { |
2735 | if (unlikely(!q)) | 2740 | if (unlikely(!q)) |
2736 | return; | 2741 | return; |
@@ -2760,7 +2765,7 @@ EXPORT_SYMBOL_GPL(__blk_put_request); | |||
2760 | void blk_put_request(struct request *req) | 2765 | void blk_put_request(struct request *req) |
2761 | { | 2766 | { |
2762 | unsigned long flags; | 2767 | unsigned long flags; |
2763 | request_queue_t *q = req->q; | 2768 | struct request_queue *q = req->q; |
2764 | 2769 | ||
2765 | /* | 2770 | /* |
2766 | * Gee, IDE calls in w/ NULL q. Fix IDE and remove the | 2771 | * Gee, IDE calls in w/ NULL q. Fix IDE and remove the |
@@ -2798,7 +2803,7 @@ EXPORT_SYMBOL(blk_end_sync_rq); | |||
2798 | /* | 2803 | /* |
2799 | * Has to be called with the request spinlock acquired | 2804 | * Has to be called with the request spinlock acquired |
2800 | */ | 2805 | */ |
2801 | static int attempt_merge(request_queue_t *q, struct request *req, | 2806 | static int attempt_merge(struct request_queue *q, struct request *req, |
2802 | struct request *next) | 2807 | struct request *next) |
2803 | { | 2808 | { |
2804 | if (!rq_mergeable(req) || !rq_mergeable(next)) | 2809 | if (!rq_mergeable(req) || !rq_mergeable(next)) |
@@ -2851,7 +2856,8 @@ static int attempt_merge(request_queue_t *q, struct request *req, | |||
2851 | return 1; | 2856 | return 1; |
2852 | } | 2857 | } |
2853 | 2858 | ||
2854 | static inline int attempt_back_merge(request_queue_t *q, struct request *rq) | 2859 | static inline int attempt_back_merge(struct request_queue *q, |
2860 | struct request *rq) | ||
2855 | { | 2861 | { |
2856 | struct request *next = elv_latter_request(q, rq); | 2862 | struct request *next = elv_latter_request(q, rq); |
2857 | 2863 | ||
@@ -2861,7 +2867,8 @@ static inline int attempt_back_merge(request_queue_t *q, struct request *rq) | |||
2861 | return 0; | 2867 | return 0; |
2862 | } | 2868 | } |
2863 | 2869 | ||
2864 | static inline int attempt_front_merge(request_queue_t *q, struct request *rq) | 2870 | static inline int attempt_front_merge(struct request_queue *q, |
2871 | struct request *rq) | ||
2865 | { | 2872 | { |
2866 | struct request *prev = elv_former_request(q, rq); | 2873 | struct request *prev = elv_former_request(q, rq); |
2867 | 2874 | ||
@@ -2905,7 +2912,7 @@ static void init_request_from_bio(struct request *req, struct bio *bio) | |||
2905 | req->start_time = jiffies; | 2912 | req->start_time = jiffies; |
2906 | } | 2913 | } |
2907 | 2914 | ||
2908 | static int __make_request(request_queue_t *q, struct bio *bio) | 2915 | static int __make_request(struct request_queue *q, struct bio *bio) |
2909 | { | 2916 | { |
2910 | struct request *req; | 2917 | struct request *req; |
2911 | int el_ret, nr_sectors, barrier, err; | 2918 | int el_ret, nr_sectors, barrier, err; |
@@ -3119,7 +3126,7 @@ static inline int should_fail_request(struct bio *bio) | |||
3119 | */ | 3126 | */ |
3120 | static inline void __generic_make_request(struct bio *bio) | 3127 | static inline void __generic_make_request(struct bio *bio) |
3121 | { | 3128 | { |
3122 | request_queue_t *q; | 3129 | struct request_queue *q; |
3123 | sector_t maxsector; | 3130 | sector_t maxsector; |
3124 | sector_t old_sector; | 3131 | sector_t old_sector; |
3125 | int ret, nr_sectors = bio_sectors(bio); | 3132 | int ret, nr_sectors = bio_sectors(bio); |
@@ -3312,7 +3319,7 @@ static void blk_recalc_rq_segments(struct request *rq) | |||
3312 | struct bio *bio, *prevbio = NULL; | 3319 | struct bio *bio, *prevbio = NULL; |
3313 | int nr_phys_segs, nr_hw_segs; | 3320 | int nr_phys_segs, nr_hw_segs; |
3314 | unsigned int phys_size, hw_size; | 3321 | unsigned int phys_size, hw_size; |
3315 | request_queue_t *q = rq->q; | 3322 | struct request_queue *q = rq->q; |
3316 | 3323 | ||
3317 | if (!rq->bio) | 3324 | if (!rq->bio) |
3318 | return; | 3325 | return; |
@@ -3658,7 +3665,8 @@ void end_request(struct request *req, int uptodate) | |||
3658 | 3665 | ||
3659 | EXPORT_SYMBOL(end_request); | 3666 | EXPORT_SYMBOL(end_request); |
3660 | 3667 | ||
3661 | void blk_rq_bio_prep(request_queue_t *q, struct request *rq, struct bio *bio) | 3668 | void blk_rq_bio_prep(struct request_queue *q, struct request *rq, |
3669 | struct bio *bio) | ||
3662 | { | 3670 | { |
3663 | /* first two bits are identical in rq->cmd_flags and bio->bi_rw */ | 3671 | /* first two bits are identical in rq->cmd_flags and bio->bi_rw */ |
3664 | rq->cmd_flags |= (bio->bi_rw & 3); | 3672 | rq->cmd_flags |= (bio->bi_rw & 3); |
@@ -3701,7 +3709,7 @@ int __init blk_dev_init(void) | |||
3701 | sizeof(struct request), 0, SLAB_PANIC, NULL); | 3709 | sizeof(struct request), 0, SLAB_PANIC, NULL); |
3702 | 3710 | ||
3703 | requestq_cachep = kmem_cache_create("blkdev_queue", | 3711 | requestq_cachep = kmem_cache_create("blkdev_queue", |
3704 | sizeof(request_queue_t), 0, SLAB_PANIC, NULL); | 3712 | sizeof(struct request_queue), 0, SLAB_PANIC, NULL); |
3705 | 3713 | ||
3706 | iocontext_cachep = kmem_cache_create("blkdev_ioc", | 3714 | iocontext_cachep = kmem_cache_create("blkdev_ioc", |
3707 | sizeof(struct io_context), 0, SLAB_PANIC, NULL); | 3715 | sizeof(struct io_context), 0, SLAB_PANIC, NULL); |
@@ -4021,7 +4029,8 @@ static ssize_t | |||
4021 | queue_attr_show(struct kobject *kobj, struct attribute *attr, char *page) | 4029 | queue_attr_show(struct kobject *kobj, struct attribute *attr, char *page) |
4022 | { | 4030 | { |
4023 | struct queue_sysfs_entry *entry = to_queue(attr); | 4031 | struct queue_sysfs_entry *entry = to_queue(attr); |
4024 | request_queue_t *q = container_of(kobj, struct request_queue, kobj); | 4032 | struct request_queue *q = |
4033 | container_of(kobj, struct request_queue, kobj); | ||
4025 | ssize_t res; | 4034 | ssize_t res; |
4026 | 4035 | ||
4027 | if (!entry->show) | 4036 | if (!entry->show) |
@@ -4041,7 +4050,7 @@ queue_attr_store(struct kobject *kobj, struct attribute *attr, | |||
4041 | const char *page, size_t length) | 4050 | const char *page, size_t length) |
4042 | { | 4051 | { |
4043 | struct queue_sysfs_entry *entry = to_queue(attr); | 4052 | struct queue_sysfs_entry *entry = to_queue(attr); |
4044 | request_queue_t *q = container_of(kobj, struct request_queue, kobj); | 4053 | struct request_queue *q = container_of(kobj, struct request_queue, kobj); |
4045 | 4054 | ||
4046 | ssize_t res; | 4055 | ssize_t res; |
4047 | 4056 | ||
@@ -4072,7 +4081,7 @@ int blk_register_queue(struct gendisk *disk) | |||
4072 | { | 4081 | { |
4073 | int ret; | 4082 | int ret; |
4074 | 4083 | ||
4075 | request_queue_t *q = disk->queue; | 4084 | struct request_queue *q = disk->queue; |
4076 | 4085 | ||
4077 | if (!q || !q->request_fn) | 4086 | if (!q || !q->request_fn) |
4078 | return -ENXIO; | 4087 | return -ENXIO; |
@@ -4097,7 +4106,7 @@ int blk_register_queue(struct gendisk *disk) | |||
4097 | 4106 | ||
4098 | void blk_unregister_queue(struct gendisk *disk) | 4107 | void blk_unregister_queue(struct gendisk *disk) |
4099 | { | 4108 | { |
4100 | request_queue_t *q = disk->queue; | 4109 | struct request_queue *q = disk->queue; |
4101 | 4110 | ||
4102 | if (q && q->request_fn) { | 4111 | if (q && q->request_fn) { |
4103 | elv_unregister_queue(q); | 4112 | elv_unregister_queue(q); |
diff --git a/block/noop-iosched.c b/block/noop-iosched.c index 1c3de2b9a6b5..7563d8aa3944 100644 --- a/block/noop-iosched.c +++ b/block/noop-iosched.c | |||
@@ -11,13 +11,13 @@ struct noop_data { | |||
11 | struct list_head queue; | 11 | struct list_head queue; |
12 | }; | 12 | }; |
13 | 13 | ||
14 | static void noop_merged_requests(request_queue_t *q, struct request *rq, | 14 | static void noop_merged_requests(struct request_queue *q, struct request *rq, |
15 | struct request *next) | 15 | struct request *next) |
16 | { | 16 | { |
17 | list_del_init(&next->queuelist); | 17 | list_del_init(&next->queuelist); |
18 | } | 18 | } |
19 | 19 | ||
20 | static int noop_dispatch(request_queue_t *q, int force) | 20 | static int noop_dispatch(struct request_queue *q, int force) |
21 | { | 21 | { |
22 | struct noop_data *nd = q->elevator->elevator_data; | 22 | struct noop_data *nd = q->elevator->elevator_data; |
23 | 23 | ||
@@ -31,14 +31,14 @@ static int noop_dispatch(request_queue_t *q, int force) | |||
31 | return 0; | 31 | return 0; |
32 | } | 32 | } |
33 | 33 | ||
34 | static void noop_add_request(request_queue_t *q, struct request *rq) | 34 | static void noop_add_request(struct request_queue *q, struct request *rq) |
35 | { | 35 | { |
36 | struct noop_data *nd = q->elevator->elevator_data; | 36 | struct noop_data *nd = q->elevator->elevator_data; |
37 | 37 | ||
38 | list_add_tail(&rq->queuelist, &nd->queue); | 38 | list_add_tail(&rq->queuelist, &nd->queue); |
39 | } | 39 | } |
40 | 40 | ||
41 | static int noop_queue_empty(request_queue_t *q) | 41 | static int noop_queue_empty(struct request_queue *q) |
42 | { | 42 | { |
43 | struct noop_data *nd = q->elevator->elevator_data; | 43 | struct noop_data *nd = q->elevator->elevator_data; |
44 | 44 | ||
@@ -46,7 +46,7 @@ static int noop_queue_empty(request_queue_t *q) | |||
46 | } | 46 | } |
47 | 47 | ||
48 | static struct request * | 48 | static struct request * |
49 | noop_former_request(request_queue_t *q, struct request *rq) | 49 | noop_former_request(struct request_queue *q, struct request *rq) |
50 | { | 50 | { |
51 | struct noop_data *nd = q->elevator->elevator_data; | 51 | struct noop_data *nd = q->elevator->elevator_data; |
52 | 52 | ||
@@ -56,7 +56,7 @@ noop_former_request(request_queue_t *q, struct request *rq) | |||
56 | } | 56 | } |
57 | 57 | ||
58 | static struct request * | 58 | static struct request * |
59 | noop_latter_request(request_queue_t *q, struct request *rq) | 59 | noop_latter_request(struct request_queue *q, struct request *rq) |
60 | { | 60 | { |
61 | struct noop_data *nd = q->elevator->elevator_data; | 61 | struct noop_data *nd = q->elevator->elevator_data; |
62 | 62 | ||
@@ -65,7 +65,7 @@ noop_latter_request(request_queue_t *q, struct request *rq) | |||
65 | return list_entry(rq->queuelist.next, struct request, queuelist); | 65 | return list_entry(rq->queuelist.next, struct request, queuelist); |
66 | } | 66 | } |
67 | 67 | ||
68 | static void *noop_init_queue(request_queue_t *q) | 68 | static void *noop_init_queue(struct request_queue *q) |
69 | { | 69 | { |
70 | struct noop_data *nd; | 70 | struct noop_data *nd; |
71 | 71 | ||
diff --git a/block/scsi_ioctl.c b/block/scsi_ioctl.c index d359a715bbc8..91c73224f4c6 100644 --- a/block/scsi_ioctl.c +++ b/block/scsi_ioctl.c | |||
@@ -49,22 +49,22 @@ static int sg_get_version(int __user *p) | |||
49 | return put_user(sg_version_num, p); | 49 | return put_user(sg_version_num, p); |
50 | } | 50 | } |
51 | 51 | ||
52 | static int scsi_get_idlun(request_queue_t *q, int __user *p) | 52 | static int scsi_get_idlun(struct request_queue *q, int __user *p) |
53 | { | 53 | { |
54 | return put_user(0, p); | 54 | return put_user(0, p); |
55 | } | 55 | } |
56 | 56 | ||
57 | static int scsi_get_bus(request_queue_t *q, int __user *p) | 57 | static int scsi_get_bus(struct request_queue *q, int __user *p) |
58 | { | 58 | { |
59 | return put_user(0, p); | 59 | return put_user(0, p); |
60 | } | 60 | } |
61 | 61 | ||
62 | static int sg_get_timeout(request_queue_t *q) | 62 | static int sg_get_timeout(struct request_queue *q) |
63 | { | 63 | { |
64 | return q->sg_timeout / (HZ / USER_HZ); | 64 | return q->sg_timeout / (HZ / USER_HZ); |
65 | } | 65 | } |
66 | 66 | ||
67 | static int sg_set_timeout(request_queue_t *q, int __user *p) | 67 | static int sg_set_timeout(struct request_queue *q, int __user *p) |
68 | { | 68 | { |
69 | int timeout, err = get_user(timeout, p); | 69 | int timeout, err = get_user(timeout, p); |
70 | 70 | ||
@@ -74,14 +74,14 @@ static int sg_set_timeout(request_queue_t *q, int __user *p) | |||
74 | return err; | 74 | return err; |
75 | } | 75 | } |
76 | 76 | ||
77 | static int sg_get_reserved_size(request_queue_t *q, int __user *p) | 77 | static int sg_get_reserved_size(struct request_queue *q, int __user *p) |
78 | { | 78 | { |
79 | unsigned val = min(q->sg_reserved_size, q->max_sectors << 9); | 79 | unsigned val = min(q->sg_reserved_size, q->max_sectors << 9); |
80 | 80 | ||
81 | return put_user(val, p); | 81 | return put_user(val, p); |
82 | } | 82 | } |
83 | 83 | ||
84 | static int sg_set_reserved_size(request_queue_t *q, int __user *p) | 84 | static int sg_set_reserved_size(struct request_queue *q, int __user *p) |
85 | { | 85 | { |
86 | int size, err = get_user(size, p); | 86 | int size, err = get_user(size, p); |
87 | 87 | ||
@@ -101,7 +101,7 @@ static int sg_set_reserved_size(request_queue_t *q, int __user *p) | |||
101 | * will always return that we are ATAPI even for a real SCSI drive, I'm not | 101 | * will always return that we are ATAPI even for a real SCSI drive, I'm not |
102 | * so sure this is worth doing anything about (why would you care??) | 102 | * so sure this is worth doing anything about (why would you care??) |
103 | */ | 103 | */ |
104 | static int sg_emulated_host(request_queue_t *q, int __user *p) | 104 | static int sg_emulated_host(struct request_queue *q, int __user *p) |
105 | { | 105 | { |
106 | return put_user(1, p); | 106 | return put_user(1, p); |
107 | } | 107 | } |
@@ -214,7 +214,7 @@ int blk_verify_command(unsigned char *cmd, int has_write_perm) | |||
214 | } | 214 | } |
215 | EXPORT_SYMBOL_GPL(blk_verify_command); | 215 | EXPORT_SYMBOL_GPL(blk_verify_command); |
216 | 216 | ||
217 | static int blk_fill_sghdr_rq(request_queue_t *q, struct request *rq, | 217 | static int blk_fill_sghdr_rq(struct request_queue *q, struct request *rq, |
218 | struct sg_io_hdr *hdr, int has_write_perm) | 218 | struct sg_io_hdr *hdr, int has_write_perm) |
219 | { | 219 | { |
220 | memset(rq->cmd, 0, BLK_MAX_CDB); /* ATAPI hates garbage after CDB */ | 220 | memset(rq->cmd, 0, BLK_MAX_CDB); /* ATAPI hates garbage after CDB */ |
@@ -286,7 +286,7 @@ static int blk_complete_sghdr_rq(struct request *rq, struct sg_io_hdr *hdr, | |||
286 | return r; | 286 | return r; |
287 | } | 287 | } |
288 | 288 | ||
289 | static int sg_io(struct file *file, request_queue_t *q, | 289 | static int sg_io(struct file *file, struct request_queue *q, |
290 | struct gendisk *bd_disk, struct sg_io_hdr *hdr) | 290 | struct gendisk *bd_disk, struct sg_io_hdr *hdr) |
291 | { | 291 | { |
292 | unsigned long start_time; | 292 | unsigned long start_time; |
@@ -519,7 +519,8 @@ error: | |||
519 | EXPORT_SYMBOL_GPL(sg_scsi_ioctl); | 519 | EXPORT_SYMBOL_GPL(sg_scsi_ioctl); |
520 | 520 | ||
521 | /* Send basic block requests */ | 521 | /* Send basic block requests */ |
522 | static int __blk_send_generic(request_queue_t *q, struct gendisk *bd_disk, int cmd, int data) | 522 | static int __blk_send_generic(struct request_queue *q, struct gendisk *bd_disk, |
523 | int cmd, int data) | ||
523 | { | 524 | { |
524 | struct request *rq; | 525 | struct request *rq; |
525 | int err; | 526 | int err; |
@@ -539,7 +540,8 @@ static int __blk_send_generic(request_queue_t *q, struct gendisk *bd_disk, int c | |||
539 | return err; | 540 | return err; |
540 | } | 541 | } |
541 | 542 | ||
542 | static inline int blk_send_start_stop(request_queue_t *q, struct gendisk *bd_disk, int data) | 543 | static inline int blk_send_start_stop(struct request_queue *q, |
544 | struct gendisk *bd_disk, int data) | ||
543 | { | 545 | { |
544 | return __blk_send_generic(q, bd_disk, GPCMD_START_STOP_UNIT, data); | 546 | return __blk_send_generic(q, bd_disk, GPCMD_START_STOP_UNIT, data); |
545 | } | 547 | } |
diff --git a/drivers/acorn/block/fd1772.c b/drivers/acorn/block/fd1772.c index 423ed08fb6f7..d7e18ce8dad9 100644 --- a/drivers/acorn/block/fd1772.c +++ b/drivers/acorn/block/fd1772.c | |||
@@ -372,7 +372,7 @@ static int fd_test_drive_present(int drive); | |||
372 | static void config_types(void); | 372 | static void config_types(void); |
373 | static int floppy_open(struct inode *inode, struct file *filp); | 373 | static int floppy_open(struct inode *inode, struct file *filp); |
374 | static int floppy_release(struct inode *inode, struct file *filp); | 374 | static int floppy_release(struct inode *inode, struct file *filp); |
375 | static void do_fd_request(request_queue_t *); | 375 | static void do_fd_request(struct request_queue *); |
376 | 376 | ||
377 | /************************* End of Prototypes **************************/ | 377 | /************************* End of Prototypes **************************/ |
378 | 378 | ||
@@ -1271,7 +1271,7 @@ static void fd1772_checkint(void) | |||
1271 | } | 1271 | } |
1272 | } | 1272 | } |
1273 | 1273 | ||
1274 | static void do_fd_request(request_queue_t* q) | 1274 | static void do_fd_request(struct request_queue* q) |
1275 | { | 1275 | { |
1276 | unsigned long flags; | 1276 | unsigned long flags; |
1277 | 1277 | ||
diff --git a/drivers/acorn/block/mfmhd.c b/drivers/acorn/block/mfmhd.c index d85520f78e68..74058db674db 100644 --- a/drivers/acorn/block/mfmhd.c +++ b/drivers/acorn/block/mfmhd.c | |||
@@ -924,7 +924,7 @@ static void mfm_request(void) | |||
924 | DBG("mfm_request: Dropping out bottom\n"); | 924 | DBG("mfm_request: Dropping out bottom\n"); |
925 | } | 925 | } |
926 | 926 | ||
927 | static void do_mfm_request(request_queue_t *q) | 927 | static void do_mfm_request(struct request_queue *q) |
928 | { | 928 | { |
929 | DBG("do_mfm_request: about to mfm_request\n"); | 929 | DBG("do_mfm_request: about to mfm_request\n"); |
930 | mfm_request(); | 930 | mfm_request(); |
diff --git a/drivers/acpi/Kconfig b/drivers/acpi/Kconfig index 408b45168aba..22b401b2e088 100644 --- a/drivers/acpi/Kconfig +++ b/drivers/acpi/Kconfig | |||
@@ -42,51 +42,26 @@ menuconfig ACPI | |||
42 | 42 | ||
43 | if ACPI | 43 | if ACPI |
44 | 44 | ||
45 | config ACPI_SLEEP | ||
46 | bool "Sleep States" | ||
47 | depends on X86 && (!SMP || SUSPEND_SMP) | ||
48 | default y | ||
49 | ---help--- | ||
50 | This option adds support for ACPI suspend states. | ||
51 | |||
52 | With this option, you will be able to put the system "to sleep". | ||
53 | Sleep states are low power states for the system and devices. All | ||
54 | of the system operating state is saved to either memory or disk | ||
55 | (depending on the state), to allow the system to resume operation | ||
56 | quickly at your request. | ||
57 | |||
58 | Although this option sounds really nifty, barely any of the device | ||
59 | drivers have been converted to the new driver model and hence few | ||
60 | have proper power management support. | ||
61 | |||
62 | This option is not recommended for anyone except those doing driver | ||
63 | power management development. | ||
64 | |||
65 | config ACPI_SLEEP_PROC_FS | ||
66 | bool | ||
67 | depends on ACPI_SLEEP && PROC_FS | ||
68 | default y | ||
69 | |||
70 | config ACPI_SLEEP_PROC_SLEEP | ||
71 | bool "/proc/acpi/sleep (deprecated)" | ||
72 | depends on ACPI_SLEEP_PROC_FS | ||
73 | default n | ||
74 | ---help--- | ||
75 | Create /proc/acpi/sleep | ||
76 | Deprecated by /sys/power/state | ||
77 | |||
78 | config ACPI_PROCFS | 45 | config ACPI_PROCFS |
79 | bool "Procfs interface (deprecated)" | 46 | bool "Deprecated /proc/acpi files" |
80 | default y | 47 | depends on PROC_FS |
81 | ---help--- | 48 | ---help--- |
82 | The Procfs interface for ACPI is made optional for backward compatibility. | 49 | For backwards compatibility, this option allows |
83 | As the same functions are duplicated in the sysfs interface | 50 | depricated /proc/acpi/ files to exist, even when |
84 | and this proc interface will be removed some time later, | 51 | they have been replaced by functions in /sys. |
85 | it's marked as deprecated. | 52 | The deprecated files (and their replacements) include: |
86 | ( /proc/acpi/debug_layer && debug_level are deprecated by | 53 | |
87 | /sys/module/acpi/parameters/debug_layer && debug_level. | 54 | /proc/acpi/sleep (/sys/power/state) |
88 | /proc/acpi/info is deprecated by | 55 | /proc/acpi/info (/sys/modules/acpi/parameters/acpica_version) |
89 | /sys/module/acpi/parameters/acpica_version ) | 56 | /proc/acpi/dsdt (/sys/firmware/acpi/tables/DSDT) |
57 | /proc/acpi/fadt (/sys/firmware/acpi/tables/FACP) | ||
58 | /proc/acpi/debug_layer (/sys/module/acpi/parameters/debug_layer) | ||
59 | /proc/acpi/debug_level (/sys/module/acpi/parameters/debug_level) | ||
60 | |||
61 | This option has no effect on /proc/acpi/ files | ||
62 | and functions which do not yet exist in /sys. | ||
63 | |||
64 | Say N to delete /proc/acpi/ files that have moved to /sys/ | ||
90 | 65 | ||
91 | config ACPI_AC | 66 | config ACPI_AC |
92 | tristate "AC Adapter" | 67 | tristate "AC Adapter" |
diff --git a/drivers/acpi/ac.c b/drivers/acpi/ac.c index 37c7dc4f9fe5..d8b35093527a 100644 --- a/drivers/acpi/ac.c +++ b/drivers/acpi/ac.c | |||
@@ -34,7 +34,6 @@ | |||
34 | 34 | ||
35 | #define ACPI_AC_COMPONENT 0x00020000 | 35 | #define ACPI_AC_COMPONENT 0x00020000 |
36 | #define ACPI_AC_CLASS "ac_adapter" | 36 | #define ACPI_AC_CLASS "ac_adapter" |
37 | #define ACPI_AC_HID "ACPI0003" | ||
38 | #define ACPI_AC_DEVICE_NAME "AC Adapter" | 37 | #define ACPI_AC_DEVICE_NAME "AC Adapter" |
39 | #define ACPI_AC_FILE_STATE "state" | 38 | #define ACPI_AC_FILE_STATE "state" |
40 | #define ACPI_AC_NOTIFY_STATUS 0x80 | 39 | #define ACPI_AC_NOTIFY_STATUS 0x80 |
@@ -56,10 +55,16 @@ static int acpi_ac_add(struct acpi_device *device); | |||
56 | static int acpi_ac_remove(struct acpi_device *device, int type); | 55 | static int acpi_ac_remove(struct acpi_device *device, int type); |
57 | static int acpi_ac_open_fs(struct inode *inode, struct file *file); | 56 | static int acpi_ac_open_fs(struct inode *inode, struct file *file); |
58 | 57 | ||
58 | const static struct acpi_device_id ac_device_ids[] = { | ||
59 | {"ACPI0003", 0}, | ||
60 | {"", 0}, | ||
61 | }; | ||
62 | MODULE_DEVICE_TABLE(acpi, ac_device_ids); | ||
63 | |||
59 | static struct acpi_driver acpi_ac_driver = { | 64 | static struct acpi_driver acpi_ac_driver = { |
60 | .name = "ac", | 65 | .name = "ac", |
61 | .class = ACPI_AC_CLASS, | 66 | .class = ACPI_AC_CLASS, |
62 | .ids = ACPI_AC_HID, | 67 | .ids = ac_device_ids, |
63 | .ops = { | 68 | .ops = { |
64 | .add = acpi_ac_add, | 69 | .add = acpi_ac_add, |
65 | .remove = acpi_ac_remove, | 70 | .remove = acpi_ac_remove, |
diff --git a/drivers/acpi/acpi_memhotplug.c b/drivers/acpi/acpi_memhotplug.c index e65628a03085..5f1127ad5a95 100644 --- a/drivers/acpi/acpi_memhotplug.c +++ b/drivers/acpi/acpi_memhotplug.c | |||
@@ -53,10 +53,16 @@ static int acpi_memory_device_add(struct acpi_device *device); | |||
53 | static int acpi_memory_device_remove(struct acpi_device *device, int type); | 53 | static int acpi_memory_device_remove(struct acpi_device *device, int type); |
54 | static int acpi_memory_device_start(struct acpi_device *device); | 54 | static int acpi_memory_device_start(struct acpi_device *device); |
55 | 55 | ||
56 | static const struct acpi_device_id memory_device_ids[] = { | ||
57 | {ACPI_MEMORY_DEVICE_HID, 0}, | ||
58 | {"", 0}, | ||
59 | }; | ||
60 | MODULE_DEVICE_TABLE(acpi, memory_device_ids); | ||
61 | |||
56 | static struct acpi_driver acpi_memory_device_driver = { | 62 | static struct acpi_driver acpi_memory_device_driver = { |
57 | .name = "acpi_memhotplug", | 63 | .name = "acpi_memhotplug", |
58 | .class = ACPI_MEMORY_DEVICE_CLASS, | 64 | .class = ACPI_MEMORY_DEVICE_CLASS, |
59 | .ids = ACPI_MEMORY_DEVICE_HID, | 65 | .ids = memory_device_ids, |
60 | .ops = { | 66 | .ops = { |
61 | .add = acpi_memory_device_add, | 67 | .add = acpi_memory_device_add, |
62 | .remove = acpi_memory_device_remove, | 68 | .remove = acpi_memory_device_remove, |
diff --git a/drivers/acpi/asus_acpi.c b/drivers/acpi/asus_acpi.c index 3cd79caad70c..9c4bd220c44f 100644 --- a/drivers/acpi/asus_acpi.c +++ b/drivers/acpi/asus_acpi.c | |||
@@ -56,7 +56,6 @@ | |||
56 | #define ACPI_HOTK_NAME "Asus Laptop ACPI Extras Driver" | 56 | #define ACPI_HOTK_NAME "Asus Laptop ACPI Extras Driver" |
57 | #define ACPI_HOTK_CLASS "hotkey" | 57 | #define ACPI_HOTK_CLASS "hotkey" |
58 | #define ACPI_HOTK_DEVICE_NAME "Hotkey" | 58 | #define ACPI_HOTK_DEVICE_NAME "Hotkey" |
59 | #define ACPI_HOTK_HID "ATK0100" | ||
60 | 59 | ||
61 | /* | 60 | /* |
62 | * Some events we use, same for all Asus | 61 | * Some events we use, same for all Asus |
@@ -426,14 +425,20 @@ static struct acpi_table_header *asus_info; | |||
426 | static struct asus_hotk *hotk; | 425 | static struct asus_hotk *hotk; |
427 | 426 | ||
428 | /* | 427 | /* |
429 | * The hotkey driver declaration | 428 | * The hotkey driver and autoloading declaration |
430 | */ | 429 | */ |
431 | static int asus_hotk_add(struct acpi_device *device); | 430 | static int asus_hotk_add(struct acpi_device *device); |
432 | static int asus_hotk_remove(struct acpi_device *device, int type); | 431 | static int asus_hotk_remove(struct acpi_device *device, int type); |
432 | static const struct acpi_device_id asus_device_ids[] = { | ||
433 | {"ATK0100", 0}, | ||
434 | {"", 0}, | ||
435 | }; | ||
436 | MODULE_DEVICE_TABLE(acpi, asus_device_ids); | ||
437 | |||
433 | static struct acpi_driver asus_hotk_driver = { | 438 | static struct acpi_driver asus_hotk_driver = { |
434 | .name = "asus_acpi", | 439 | .name = "asus_acpi", |
435 | .class = ACPI_HOTK_CLASS, | 440 | .class = ACPI_HOTK_CLASS, |
436 | .ids = ACPI_HOTK_HID, | 441 | .ids = asus_device_ids, |
437 | .ops = { | 442 | .ops = { |
438 | .add = asus_hotk_add, | 443 | .add = asus_hotk_add, |
439 | .remove = asus_hotk_remove, | 444 | .remove = asus_hotk_remove, |
diff --git a/drivers/acpi/battery.c b/drivers/acpi/battery.c index cad932de383d..81651032791b 100644 --- a/drivers/acpi/battery.c +++ b/drivers/acpi/battery.c | |||
@@ -41,7 +41,6 @@ | |||
41 | 41 | ||
42 | #define ACPI_BATTERY_COMPONENT 0x00040000 | 42 | #define ACPI_BATTERY_COMPONENT 0x00040000 |
43 | #define ACPI_BATTERY_CLASS "battery" | 43 | #define ACPI_BATTERY_CLASS "battery" |
44 | #define ACPI_BATTERY_HID "PNP0C0A" | ||
45 | #define ACPI_BATTERY_DEVICE_NAME "Battery" | 44 | #define ACPI_BATTERY_DEVICE_NAME "Battery" |
46 | #define ACPI_BATTERY_NOTIFY_STATUS 0x80 | 45 | #define ACPI_BATTERY_NOTIFY_STATUS 0x80 |
47 | #define ACPI_BATTERY_NOTIFY_INFO 0x81 | 46 | #define ACPI_BATTERY_NOTIFY_INFO 0x81 |
@@ -74,10 +73,16 @@ static int acpi_battery_add(struct acpi_device *device); | |||
74 | static int acpi_battery_remove(struct acpi_device *device, int type); | 73 | static int acpi_battery_remove(struct acpi_device *device, int type); |
75 | static int acpi_battery_resume(struct acpi_device *device); | 74 | static int acpi_battery_resume(struct acpi_device *device); |
76 | 75 | ||
76 | static const struct acpi_device_id battery_device_ids[] = { | ||
77 | {"PNP0C0A", 0}, | ||
78 | {"", 0}, | ||
79 | }; | ||
80 | MODULE_DEVICE_TABLE(acpi, battery_device_ids); | ||
81 | |||
77 | static struct acpi_driver acpi_battery_driver = { | 82 | static struct acpi_driver acpi_battery_driver = { |
78 | .name = "battery", | 83 | .name = "battery", |
79 | .class = ACPI_BATTERY_CLASS, | 84 | .class = ACPI_BATTERY_CLASS, |
80 | .ids = ACPI_BATTERY_HID, | 85 | .ids = battery_device_ids, |
81 | .ops = { | 86 | .ops = { |
82 | .add = acpi_battery_add, | 87 | .add = acpi_battery_add, |
83 | .resume = acpi_battery_resume, | 88 | .resume = acpi_battery_resume, |
diff --git a/drivers/acpi/button.c b/drivers/acpi/button.c index cb4110b50cd0..540581338ef5 100644 --- a/drivers/acpi/button.c +++ b/drivers/acpi/button.c | |||
@@ -66,6 +66,16 @@ MODULE_AUTHOR("Paul Diefenbaugh"); | |||
66 | MODULE_DESCRIPTION("ACPI Button Driver"); | 66 | MODULE_DESCRIPTION("ACPI Button Driver"); |
67 | MODULE_LICENSE("GPL"); | 67 | MODULE_LICENSE("GPL"); |
68 | 68 | ||
69 | static const struct acpi_device_id button_device_ids[] = { | ||
70 | {ACPI_BUTTON_HID_LID, 0}, | ||
71 | {ACPI_BUTTON_HID_SLEEP, 0}, | ||
72 | {ACPI_BUTTON_HID_SLEEPF, 0}, | ||
73 | {ACPI_BUTTON_HID_POWER, 0}, | ||
74 | {ACPI_BUTTON_HID_POWERF, 0}, | ||
75 | {"", 0}, | ||
76 | }; | ||
77 | MODULE_DEVICE_TABLE(acpi, button_device_ids); | ||
78 | |||
69 | static int acpi_button_add(struct acpi_device *device); | 79 | static int acpi_button_add(struct acpi_device *device); |
70 | static int acpi_button_remove(struct acpi_device *device, int type); | 80 | static int acpi_button_remove(struct acpi_device *device, int type); |
71 | static int acpi_button_info_open_fs(struct inode *inode, struct file *file); | 81 | static int acpi_button_info_open_fs(struct inode *inode, struct file *file); |
@@ -74,7 +84,7 @@ static int acpi_button_state_open_fs(struct inode *inode, struct file *file); | |||
74 | static struct acpi_driver acpi_button_driver = { | 84 | static struct acpi_driver acpi_button_driver = { |
75 | .name = "button", | 85 | .name = "button", |
76 | .class = ACPI_BUTTON_CLASS, | 86 | .class = ACPI_BUTTON_CLASS, |
77 | .ids = "button_power,button_sleep,PNP0C0D,PNP0C0C,PNP0C0E", | 87 | .ids = button_device_ids, |
78 | .ops = { | 88 | .ops = { |
79 | .add = acpi_button_add, | 89 | .add = acpi_button_add, |
80 | .remove = acpi_button_remove, | 90 | .remove = acpi_button_remove, |
diff --git a/drivers/acpi/container.c b/drivers/acpi/container.c index 0dd3bf7c0ed1..3c25ec7a1871 100644 --- a/drivers/acpi/container.c +++ b/drivers/acpi/container.c | |||
@@ -52,10 +52,18 @@ MODULE_LICENSE("GPL"); | |||
52 | static int acpi_container_add(struct acpi_device *device); | 52 | static int acpi_container_add(struct acpi_device *device); |
53 | static int acpi_container_remove(struct acpi_device *device, int type); | 53 | static int acpi_container_remove(struct acpi_device *device, int type); |
54 | 54 | ||
55 | static const struct acpi_device_id container_device_ids[] = { | ||
56 | {"ACPI0004", 0}, | ||
57 | {"PNP0A05", 0}, | ||
58 | {"PNP0A06", 0}, | ||
59 | {"", 0}, | ||
60 | }; | ||
61 | MODULE_DEVICE_TABLE(acpi, container_device_ids); | ||
62 | |||
55 | static struct acpi_driver acpi_container_driver = { | 63 | static struct acpi_driver acpi_container_driver = { |
56 | .name = "container", | 64 | .name = "container", |
57 | .class = ACPI_CONTAINER_CLASS, | 65 | .class = ACPI_CONTAINER_CLASS, |
58 | .ids = "ACPI0004,PNP0A05,PNP0A06", | 66 | .ids = container_device_ids, |
59 | .ops = { | 67 | .ops = { |
60 | .add = acpi_container_add, | 68 | .add = acpi_container_add, |
61 | .remove = acpi_container_remove, | 69 | .remove = acpi_container_remove, |
diff --git a/drivers/acpi/ec.c b/drivers/acpi/ec.c index 10e851021eca..469f3f57f881 100644 --- a/drivers/acpi/ec.c +++ b/drivers/acpi/ec.c | |||
@@ -41,7 +41,6 @@ | |||
41 | #include <acpi/actypes.h> | 41 | #include <acpi/actypes.h> |
42 | 42 | ||
43 | #define ACPI_EC_CLASS "embedded_controller" | 43 | #define ACPI_EC_CLASS "embedded_controller" |
44 | #define ACPI_EC_HID "PNP0C09" | ||
45 | #define ACPI_EC_DEVICE_NAME "Embedded Controller" | 44 | #define ACPI_EC_DEVICE_NAME "Embedded Controller" |
46 | #define ACPI_EC_FILE_INFO "info" | 45 | #define ACPI_EC_FILE_INFO "info" |
47 | 46 | ||
@@ -82,10 +81,15 @@ static int acpi_ec_start(struct acpi_device *device); | |||
82 | static int acpi_ec_stop(struct acpi_device *device, int type); | 81 | static int acpi_ec_stop(struct acpi_device *device, int type); |
83 | static int acpi_ec_add(struct acpi_device *device); | 82 | static int acpi_ec_add(struct acpi_device *device); |
84 | 83 | ||
84 | static const struct acpi_device_id ec_device_ids[] = { | ||
85 | {"PNP0C09", 0}, | ||
86 | {"", 0}, | ||
87 | }; | ||
88 | |||
85 | static struct acpi_driver acpi_ec_driver = { | 89 | static struct acpi_driver acpi_ec_driver = { |
86 | .name = "ec", | 90 | .name = "ec", |
87 | .class = ACPI_EC_CLASS, | 91 | .class = ACPI_EC_CLASS, |
88 | .ids = ACPI_EC_HID, | 92 | .ids = ec_device_ids, |
89 | .ops = { | 93 | .ops = { |
90 | .add = acpi_ec_add, | 94 | .add = acpi_ec_add, |
91 | .remove = acpi_ec_remove, | 95 | .remove = acpi_ec_remove, |
diff --git a/drivers/acpi/events/evrgnini.c b/drivers/acpi/events/evrgnini.c index 23ee7bc4a705..b1aaa0e84588 100644 --- a/drivers/acpi/events/evrgnini.c +++ b/drivers/acpi/events/evrgnini.c | |||
@@ -378,7 +378,7 @@ static u8 acpi_ev_match_pci_root_bridge(char *id) | |||
378 | static u8 acpi_ev_is_pci_root_bridge(struct acpi_namespace_node *node) | 378 | static u8 acpi_ev_is_pci_root_bridge(struct acpi_namespace_node *node) |
379 | { | 379 | { |
380 | acpi_status status; | 380 | acpi_status status; |
381 | struct acpi_device_id hid; | 381 | struct acpica_device_id hid; |
382 | struct acpi_compatible_id_list *cid; | 382 | struct acpi_compatible_id_list *cid; |
383 | acpi_native_uint i; | 383 | acpi_native_uint i; |
384 | 384 | ||
diff --git a/drivers/acpi/fan.c b/drivers/acpi/fan.c index ec655c539492..c81f6bdb68b8 100644 --- a/drivers/acpi/fan.c +++ b/drivers/acpi/fan.c | |||
@@ -50,10 +50,16 @@ static int acpi_fan_remove(struct acpi_device *device, int type); | |||
50 | static int acpi_fan_suspend(struct acpi_device *device, pm_message_t state); | 50 | static int acpi_fan_suspend(struct acpi_device *device, pm_message_t state); |
51 | static int acpi_fan_resume(struct acpi_device *device); | 51 | static int acpi_fan_resume(struct acpi_device *device); |
52 | 52 | ||
53 | static const struct acpi_device_id fan_device_ids[] = { | ||
54 | {"PNP0C0B", 0}, | ||
55 | {"", 0}, | ||
56 | }; | ||
57 | MODULE_DEVICE_TABLE(acpi, fan_device_ids); | ||
58 | |||
53 | static struct acpi_driver acpi_fan_driver = { | 59 | static struct acpi_driver acpi_fan_driver = { |
54 | .name = "fan", | 60 | .name = "fan", |
55 | .class = ACPI_FAN_CLASS, | 61 | .class = ACPI_FAN_CLASS, |
56 | .ids = "PNP0C0B", | 62 | .ids = fan_device_ids, |
57 | .ops = { | 63 | .ops = { |
58 | .add = acpi_fan_add, | 64 | .add = acpi_fan_add, |
59 | .remove = acpi_fan_remove, | 65 | .remove = acpi_fan_remove, |
diff --git a/drivers/acpi/namespace/nsxfeval.c b/drivers/acpi/namespace/nsxfeval.c index be4f2899de74..ab65b2c2560e 100644 --- a/drivers/acpi/namespace/nsxfeval.c +++ b/drivers/acpi/namespace/nsxfeval.c | |||
@@ -440,7 +440,7 @@ acpi_ns_get_device_callback(acpi_handle obj_handle, | |||
440 | acpi_status status; | 440 | acpi_status status; |
441 | struct acpi_namespace_node *node; | 441 | struct acpi_namespace_node *node; |
442 | u32 flags; | 442 | u32 flags; |
443 | struct acpi_device_id hid; | 443 | struct acpica_device_id hid; |
444 | struct acpi_compatible_id_list *cid; | 444 | struct acpi_compatible_id_list *cid; |
445 | acpi_native_uint i; | 445 | acpi_native_uint i; |
446 | 446 | ||
diff --git a/drivers/acpi/pci_link.c b/drivers/acpi/pci_link.c index 3448edd61dc4..c9f526e55392 100644 --- a/drivers/acpi/pci_link.c +++ b/drivers/acpi/pci_link.c | |||
@@ -46,7 +46,6 @@ | |||
46 | #define _COMPONENT ACPI_PCI_COMPONENT | 46 | #define _COMPONENT ACPI_PCI_COMPONENT |
47 | ACPI_MODULE_NAME("pci_link"); | 47 | ACPI_MODULE_NAME("pci_link"); |
48 | #define ACPI_PCI_LINK_CLASS "pci_irq_routing" | 48 | #define ACPI_PCI_LINK_CLASS "pci_irq_routing" |
49 | #define ACPI_PCI_LINK_HID "PNP0C0F" | ||
50 | #define ACPI_PCI_LINK_DEVICE_NAME "PCI Interrupt Link" | 49 | #define ACPI_PCI_LINK_DEVICE_NAME "PCI Interrupt Link" |
51 | #define ACPI_PCI_LINK_FILE_INFO "info" | 50 | #define ACPI_PCI_LINK_FILE_INFO "info" |
52 | #define ACPI_PCI_LINK_FILE_STATUS "state" | 51 | #define ACPI_PCI_LINK_FILE_STATUS "state" |
@@ -54,10 +53,16 @@ ACPI_MODULE_NAME("pci_link"); | |||
54 | static int acpi_pci_link_add(struct acpi_device *device); | 53 | static int acpi_pci_link_add(struct acpi_device *device); |
55 | static int acpi_pci_link_remove(struct acpi_device *device, int type); | 54 | static int acpi_pci_link_remove(struct acpi_device *device, int type); |
56 | 55 | ||
56 | static struct acpi_device_id link_device_ids[] = { | ||
57 | {"PNP0C0F", 0}, | ||
58 | {"", 0}, | ||
59 | }; | ||
60 | MODULE_DEVICE_TABLE(acpi, link_device_ids); | ||
61 | |||
57 | static struct acpi_driver acpi_pci_link_driver = { | 62 | static struct acpi_driver acpi_pci_link_driver = { |
58 | .name = "pci_link", | 63 | .name = "pci_link", |
59 | .class = ACPI_PCI_LINK_CLASS, | 64 | .class = ACPI_PCI_LINK_CLASS, |
60 | .ids = ACPI_PCI_LINK_HID, | 65 | .ids = link_device_ids, |
61 | .ops = { | 66 | .ops = { |
62 | .add = acpi_pci_link_add, | 67 | .add = acpi_pci_link_add, |
63 | .remove = acpi_pci_link_remove, | 68 | .remove = acpi_pci_link_remove, |
diff --git a/drivers/acpi/pci_root.c b/drivers/acpi/pci_root.c index ad4145a37786..f14ff1ffab29 100644 --- a/drivers/acpi/pci_root.c +++ b/drivers/acpi/pci_root.c | |||
@@ -38,16 +38,21 @@ | |||
38 | #define _COMPONENT ACPI_PCI_COMPONENT | 38 | #define _COMPONENT ACPI_PCI_COMPONENT |
39 | ACPI_MODULE_NAME("pci_root"); | 39 | ACPI_MODULE_NAME("pci_root"); |
40 | #define ACPI_PCI_ROOT_CLASS "pci_bridge" | 40 | #define ACPI_PCI_ROOT_CLASS "pci_bridge" |
41 | #define ACPI_PCI_ROOT_HID "PNP0A03" | ||
42 | #define ACPI_PCI_ROOT_DEVICE_NAME "PCI Root Bridge" | 41 | #define ACPI_PCI_ROOT_DEVICE_NAME "PCI Root Bridge" |
43 | static int acpi_pci_root_add(struct acpi_device *device); | 42 | static int acpi_pci_root_add(struct acpi_device *device); |
44 | static int acpi_pci_root_remove(struct acpi_device *device, int type); | 43 | static int acpi_pci_root_remove(struct acpi_device *device, int type); |
45 | static int acpi_pci_root_start(struct acpi_device *device); | 44 | static int acpi_pci_root_start(struct acpi_device *device); |
46 | 45 | ||
46 | static struct acpi_device_id root_device_ids[] = { | ||
47 | {"PNP0A03", 0}, | ||
48 | {"", 0}, | ||
49 | }; | ||
50 | MODULE_DEVICE_TABLE(acpi, root_device_ids); | ||
51 | |||
47 | static struct acpi_driver acpi_pci_root_driver = { | 52 | static struct acpi_driver acpi_pci_root_driver = { |
48 | .name = "pci_root", | 53 | .name = "pci_root", |
49 | .class = ACPI_PCI_ROOT_CLASS, | 54 | .class = ACPI_PCI_ROOT_CLASS, |
50 | .ids = ACPI_PCI_ROOT_HID, | 55 | .ids = root_device_ids, |
51 | .ops = { | 56 | .ops = { |
52 | .add = acpi_pci_root_add, | 57 | .add = acpi_pci_root_add, |
53 | .remove = acpi_pci_root_remove, | 58 | .remove = acpi_pci_root_remove, |
diff --git a/drivers/acpi/power.c b/drivers/acpi/power.c index 4ffecd179702..57b9a2998fd0 100644 --- a/drivers/acpi/power.c +++ b/drivers/acpi/power.c | |||
@@ -59,10 +59,16 @@ static int acpi_power_remove(struct acpi_device *device, int type); | |||
59 | static int acpi_power_resume(struct acpi_device *device); | 59 | static int acpi_power_resume(struct acpi_device *device); |
60 | static int acpi_power_open_fs(struct inode *inode, struct file *file); | 60 | static int acpi_power_open_fs(struct inode *inode, struct file *file); |
61 | 61 | ||
62 | static struct acpi_device_id power_device_ids[] = { | ||
63 | {ACPI_POWER_HID, 0}, | ||
64 | {"", 0}, | ||
65 | }; | ||
66 | MODULE_DEVICE_TABLE(acpi, power_device_ids); | ||
67 | |||
62 | static struct acpi_driver acpi_power_driver = { | 68 | static struct acpi_driver acpi_power_driver = { |
63 | .name = "power", | 69 | .name = "power", |
64 | .class = ACPI_POWER_CLASS, | 70 | .class = ACPI_POWER_CLASS, |
65 | .ids = ACPI_POWER_HID, | 71 | .ids = power_device_ids, |
66 | .ops = { | 72 | .ops = { |
67 | .add = acpi_power_add, | 73 | .add = acpi_power_add, |
68 | .remove = acpi_power_remove, | 74 | .remove = acpi_power_remove, |
diff --git a/drivers/acpi/processor_core.c b/drivers/acpi/processor_core.c index 81aceb5da7c7..498422343f38 100644 --- a/drivers/acpi/processor_core.c +++ b/drivers/acpi/processor_core.c | |||
@@ -88,10 +88,16 @@ static int acpi_processor_handle_eject(struct acpi_processor *pr); | |||
88 | extern int acpi_processor_tstate_has_changed(struct acpi_processor *pr); | 88 | extern int acpi_processor_tstate_has_changed(struct acpi_processor *pr); |
89 | 89 | ||
90 | 90 | ||
91 | static const struct acpi_device_id processor_device_ids[] = { | ||
92 | {ACPI_PROCESSOR_HID, 0}, | ||
93 | {"", 0}, | ||
94 | }; | ||
95 | MODULE_DEVICE_TABLE(acpi, processor_device_ids); | ||
96 | |||
91 | static struct acpi_driver acpi_processor_driver = { | 97 | static struct acpi_driver acpi_processor_driver = { |
92 | .name = "processor", | 98 | .name = "processor", |
93 | .class = ACPI_PROCESSOR_CLASS, | 99 | .class = ACPI_PROCESSOR_CLASS, |
94 | .ids = ACPI_PROCESSOR_HID, | 100 | .ids = processor_device_ids, |
95 | .ops = { | 101 | .ops = { |
96 | .add = acpi_processor_add, | 102 | .add = acpi_processor_add, |
97 | .remove = acpi_processor_remove, | 103 | .remove = acpi_processor_remove, |
diff --git a/drivers/acpi/processor_throttling.c b/drivers/acpi/processor_throttling.c index 3f55d1f90c11..0b8204e7082a 100644 --- a/drivers/acpi/processor_throttling.c +++ b/drivers/acpi/processor_throttling.c | |||
@@ -47,6 +47,9 @@ ACPI_MODULE_NAME("processor_throttling"); | |||
47 | static int acpi_processor_get_throttling(struct acpi_processor *pr); | 47 | static int acpi_processor_get_throttling(struct acpi_processor *pr); |
48 | int acpi_processor_set_throttling(struct acpi_processor *pr, int state); | 48 | int acpi_processor_set_throttling(struct acpi_processor *pr, int state); |
49 | 49 | ||
50 | /* | ||
51 | * _TPC - Throttling Present Capabilities | ||
52 | */ | ||
50 | static int acpi_processor_get_platform_limit(struct acpi_processor *pr) | 53 | static int acpi_processor_get_platform_limit(struct acpi_processor *pr) |
51 | { | 54 | { |
52 | acpi_status status = 0; | 55 | acpi_status status = 0; |
@@ -55,8 +58,10 @@ static int acpi_processor_get_platform_limit(struct acpi_processor *pr) | |||
55 | if (!pr) | 58 | if (!pr) |
56 | return -EINVAL; | 59 | return -EINVAL; |
57 | status = acpi_evaluate_integer(pr->handle, "_TPC", NULL, &tpc); | 60 | status = acpi_evaluate_integer(pr->handle, "_TPC", NULL, &tpc); |
58 | if (ACPI_FAILURE(status) && status != AE_NOT_FOUND) { | 61 | if (ACPI_FAILURE(status)) { |
59 | ACPI_EXCEPTION((AE_INFO, status, "Evaluating _TPC")); | 62 | if (status != AE_NOT_FOUND) { |
63 | ACPI_EXCEPTION((AE_INFO, status, "Evaluating _TPC")); | ||
64 | } | ||
60 | return -ENODEV; | 65 | return -ENODEV; |
61 | } | 66 | } |
62 | pr->throttling_platform_limit = (int)tpc; | 67 | pr->throttling_platform_limit = (int)tpc; |
@@ -68,9 +73,9 @@ int acpi_processor_tstate_has_changed(struct acpi_processor *pr) | |||
68 | return acpi_processor_get_platform_limit(pr); | 73 | return acpi_processor_get_platform_limit(pr); |
69 | } | 74 | } |
70 | 75 | ||
71 | /* -------------------------------------------------------------------------- | 76 | /* |
72 | _PTC, _TSS, _TSD support | 77 | * _PTC - Processor Throttling Control (and status) register location |
73 | -------------------------------------------------------------------------- */ | 78 | */ |
74 | static int acpi_processor_get_throttling_control(struct acpi_processor *pr) | 79 | static int acpi_processor_get_throttling_control(struct acpi_processor *pr) |
75 | { | 80 | { |
76 | int result = 0; | 81 | int result = 0; |
@@ -81,7 +86,9 @@ static int acpi_processor_get_throttling_control(struct acpi_processor *pr) | |||
81 | 86 | ||
82 | status = acpi_evaluate_object(pr->handle, "_PTC", NULL, &buffer); | 87 | status = acpi_evaluate_object(pr->handle, "_PTC", NULL, &buffer); |
83 | if (ACPI_FAILURE(status)) { | 88 | if (ACPI_FAILURE(status)) { |
84 | ACPI_EXCEPTION((AE_INFO, status, "Evaluating _PTC")); | 89 | if (status != AE_NOT_FOUND) { |
90 | ACPI_EXCEPTION((AE_INFO, status, "Evaluating _PTC")); | ||
91 | } | ||
85 | return -ENODEV; | 92 | return -ENODEV; |
86 | } | 93 | } |
87 | 94 | ||
@@ -132,6 +139,10 @@ static int acpi_processor_get_throttling_control(struct acpi_processor *pr) | |||
132 | 139 | ||
133 | return result; | 140 | return result; |
134 | } | 141 | } |
142 | |||
143 | /* | ||
144 | * _TSS - Throttling Supported States | ||
145 | */ | ||
135 | static int acpi_processor_get_throttling_states(struct acpi_processor *pr) | 146 | static int acpi_processor_get_throttling_states(struct acpi_processor *pr) |
136 | { | 147 | { |
137 | int result = 0; | 148 | int result = 0; |
@@ -144,7 +155,9 @@ static int acpi_processor_get_throttling_states(struct acpi_processor *pr) | |||
144 | 155 | ||
145 | status = acpi_evaluate_object(pr->handle, "_TSS", NULL, &buffer); | 156 | status = acpi_evaluate_object(pr->handle, "_TSS", NULL, &buffer); |
146 | if (ACPI_FAILURE(status)) { | 157 | if (ACPI_FAILURE(status)) { |
147 | ACPI_EXCEPTION((AE_INFO, status, "Evaluating _TSS")); | 158 | if (status != AE_NOT_FOUND) { |
159 | ACPI_EXCEPTION((AE_INFO, status, "Evaluating _TSS")); | ||
160 | } | ||
148 | return -ENODEV; | 161 | return -ENODEV; |
149 | } | 162 | } |
150 | 163 | ||
@@ -201,6 +214,10 @@ static int acpi_processor_get_throttling_states(struct acpi_processor *pr) | |||
201 | 214 | ||
202 | return result; | 215 | return result; |
203 | } | 216 | } |
217 | |||
218 | /* | ||
219 | * _TSD - T-State Dependencies | ||
220 | */ | ||
204 | static int acpi_processor_get_tsd(struct acpi_processor *pr) | 221 | static int acpi_processor_get_tsd(struct acpi_processor *pr) |
205 | { | 222 | { |
206 | int result = 0; | 223 | int result = 0; |
@@ -213,6 +230,9 @@ static int acpi_processor_get_tsd(struct acpi_processor *pr) | |||
213 | 230 | ||
214 | status = acpi_evaluate_object(pr->handle, "_TSD", NULL, &buffer); | 231 | status = acpi_evaluate_object(pr->handle, "_TSD", NULL, &buffer); |
215 | if (ACPI_FAILURE(status)) { | 232 | if (ACPI_FAILURE(status)) { |
233 | if (status != AE_NOT_FOUND) { | ||
234 | ACPI_EXCEPTION((AE_INFO, status, "Evaluating _TSD")); | ||
235 | } | ||
216 | return -ENODEV; | 236 | return -ENODEV; |
217 | } | 237 | } |
218 | 238 | ||
@@ -525,9 +545,6 @@ int acpi_processor_get_throttling_info(struct acpi_processor *pr) | |||
525 | int result = 0; | 545 | int result = 0; |
526 | int step = 0; | 546 | int step = 0; |
527 | int i = 0; | 547 | int i = 0; |
528 | int no_ptc = 0; | ||
529 | int no_tss = 0; | ||
530 | int no_tsd = 0; | ||
531 | 548 | ||
532 | ACPI_DEBUG_PRINT((ACPI_DB_INFO, | 549 | ACPI_DEBUG_PRINT((ACPI_DB_INFO, |
533 | "pblk_address[0x%08x] duty_offset[%d] duty_width[%d]\n", | 550 | "pblk_address[0x%08x] duty_offset[%d] duty_width[%d]\n", |
@@ -538,12 +555,14 @@ int acpi_processor_get_throttling_info(struct acpi_processor *pr) | |||
538 | if (!pr) | 555 | if (!pr) |
539 | return -EINVAL; | 556 | return -EINVAL; |
540 | 557 | ||
541 | /* TBD: Support ACPI 2.0 objects */ | 558 | /* |
542 | no_ptc = acpi_processor_get_throttling_control(pr); | 559 | * Evaluate _PTC, _TSS and _TPC |
543 | no_tss = acpi_processor_get_throttling_states(pr); | 560 | * They must all be present or none of them can be used. |
544 | no_tsd = acpi_processor_get_tsd(pr); | 561 | */ |
545 | 562 | if (acpi_processor_get_throttling_control(pr) || | |
546 | if (no_ptc || no_tss) { | 563 | acpi_processor_get_throttling_states(pr) || |
564 | acpi_processor_get_platform_limit(pr)) | ||
565 | { | ||
547 | pr->throttling.acpi_processor_get_throttling = | 566 | pr->throttling.acpi_processor_get_throttling = |
548 | &acpi_processor_get_throttling_fadt; | 567 | &acpi_processor_get_throttling_fadt; |
549 | pr->throttling.acpi_processor_set_throttling = | 568 | pr->throttling.acpi_processor_set_throttling = |
@@ -555,6 +574,8 @@ int acpi_processor_get_throttling_info(struct acpi_processor *pr) | |||
555 | &acpi_processor_set_throttling_ptc; | 574 | &acpi_processor_set_throttling_ptc; |
556 | } | 575 | } |
557 | 576 | ||
577 | acpi_processor_get_tsd(pr); | ||
578 | |||
558 | if (!pr->throttling.address) { | 579 | if (!pr->throttling.address) { |
559 | ACPI_DEBUG_PRINT((ACPI_DB_INFO, "No throttling register\n")); | 580 | ACPI_DEBUG_PRINT((ACPI_DB_INFO, "No throttling register\n")); |
560 | return 0; | 581 | return 0; |
@@ -658,18 +679,20 @@ static int acpi_processor_throttling_seq_show(struct seq_file *seq, | |||
658 | pr->throttling.state_count - 1); | 679 | pr->throttling.state_count - 1); |
659 | 680 | ||
660 | seq_puts(seq, "states:\n"); | 681 | seq_puts(seq, "states:\n"); |
661 | if (acpi_processor_get_throttling == acpi_processor_get_throttling_fadt) | 682 | if (pr->throttling.acpi_processor_get_throttling == |
683 | acpi_processor_get_throttling_fadt) { | ||
662 | for (i = 0; i < pr->throttling.state_count; i++) | 684 | for (i = 0; i < pr->throttling.state_count; i++) |
663 | seq_printf(seq, " %cT%d: %02d%%\n", | 685 | seq_printf(seq, " %cT%d: %02d%%\n", |
664 | (i == pr->throttling.state ? '*' : ' '), i, | 686 | (i == pr->throttling.state ? '*' : ' '), i, |
665 | (pr->throttling.states[i].performance ? pr-> | 687 | (pr->throttling.states[i].performance ? pr-> |
666 | throttling.states[i].performance / 10 : 0)); | 688 | throttling.states[i].performance / 10 : 0)); |
667 | else | 689 | } else { |
668 | for (i = 0; i < pr->throttling.state_count; i++) | 690 | for (i = 0; i < pr->throttling.state_count; i++) |
669 | seq_printf(seq, " %cT%d: %02d%%\n", | 691 | seq_printf(seq, " %cT%d: %02d%%\n", |
670 | (i == pr->throttling.state ? '*' : ' '), i, | 692 | (i == pr->throttling.state ? '*' : ' '), i, |
671 | (int)pr->throttling.states_tss[i]. | 693 | (int)pr->throttling.states_tss[i]. |
672 | freqpercentage); | 694 | freqpercentage); |
695 | } | ||
673 | 696 | ||
674 | end: | 697 | end: |
675 | return 0; | 698 | return 0; |
diff --git a/drivers/acpi/sbs.c b/drivers/acpi/sbs.c index 974d00ccfe84..7d8e78ea13a5 100644 --- a/drivers/acpi/sbs.c +++ b/drivers/acpi/sbs.c | |||
@@ -38,7 +38,6 @@ | |||
38 | #define ACPI_SBS_CLASS "sbs" | 38 | #define ACPI_SBS_CLASS "sbs" |
39 | #define ACPI_AC_CLASS "ac_adapter" | 39 | #define ACPI_AC_CLASS "ac_adapter" |
40 | #define ACPI_BATTERY_CLASS "battery" | 40 | #define ACPI_BATTERY_CLASS "battery" |
41 | #define ACPI_SBS_HID "ACPI0002" | ||
42 | #define ACPI_SBS_DEVICE_NAME "Smart Battery System" | 41 | #define ACPI_SBS_DEVICE_NAME "Smart Battery System" |
43 | #define ACPI_SBS_FILE_INFO "info" | 42 | #define ACPI_SBS_FILE_INFO "info" |
44 | #define ACPI_SBS_FILE_STATE "state" | 43 | #define ACPI_SBS_FILE_STATE "state" |
@@ -124,10 +123,17 @@ static int acpi_sbs_add(struct acpi_device *device); | |||
124 | static int acpi_sbs_remove(struct acpi_device *device, int type); | 123 | static int acpi_sbs_remove(struct acpi_device *device, int type); |
125 | static int acpi_sbs_resume(struct acpi_device *device); | 124 | static int acpi_sbs_resume(struct acpi_device *device); |
126 | 125 | ||
126 | static const struct acpi_device_id sbs_device_ids[] = { | ||
127 | {"ACPI0001", 0}, | ||
128 | {"ACPI0005", 0}, | ||
129 | {"", 0}, | ||
130 | }; | ||
131 | MODULE_DEVICE_TABLE(acpi, sbs_device_ids); | ||
132 | |||
127 | static struct acpi_driver acpi_sbs_driver = { | 133 | static struct acpi_driver acpi_sbs_driver = { |
128 | .name = "sbs", | 134 | .name = "sbs", |
129 | .class = ACPI_SBS_CLASS, | 135 | .class = ACPI_SBS_CLASS, |
130 | .ids = "ACPI0001,ACPI0005", | 136 | .ids = sbs_device_ids, |
131 | .ops = { | 137 | .ops = { |
132 | .add = acpi_sbs_add, | 138 | .add = acpi_sbs_add, |
133 | .remove = acpi_sbs_remove, | 139 | .remove = acpi_sbs_remove, |
diff --git a/drivers/acpi/scan.c b/drivers/acpi/scan.c index 6b3b8a522476..be74347d1354 100644 --- a/drivers/acpi/scan.c +++ b/drivers/acpi/scan.c | |||
@@ -16,7 +16,7 @@ ACPI_MODULE_NAME("scan"); | |||
16 | extern struct acpi_device *acpi_root; | 16 | extern struct acpi_device *acpi_root; |
17 | 17 | ||
18 | #define ACPI_BUS_CLASS "system_bus" | 18 | #define ACPI_BUS_CLASS "system_bus" |
19 | #define ACPI_BUS_HID "ACPI_BUS" | 19 | #define ACPI_BUS_HID "LNXSYBUS" |
20 | #define ACPI_BUS_DEVICE_NAME "System Bus" | 20 | #define ACPI_BUS_DEVICE_NAME "System Bus" |
21 | 21 | ||
22 | static LIST_HEAD(acpi_device_list); | 22 | static LIST_HEAD(acpi_device_list); |
@@ -29,6 +29,62 @@ struct acpi_device_bus_id{ | |||
29 | unsigned int instance_no; | 29 | unsigned int instance_no; |
30 | struct list_head node; | 30 | struct list_head node; |
31 | }; | 31 | }; |
32 | |||
33 | /* | ||
34 | * Creates hid/cid(s) string needed for modalias and uevent | ||
35 | * e.g. on a device with hid:IBM0001 and cid:ACPI0001 you get: | ||
36 | * char *modalias: "acpi:IBM0001:ACPI0001" | ||
37 | */ | ||
38 | int create_modalias(struct acpi_device *acpi_dev, char *modalias, int size){ | ||
39 | |||
40 | int len; | ||
41 | |||
42 | if (!acpi_dev->flags.hardware_id) | ||
43 | return -ENODEV; | ||
44 | |||
45 | len = snprintf(modalias, size, "acpi:%s:", | ||
46 | acpi_dev->pnp.hardware_id); | ||
47 | if (len < 0 || len >= size) | ||
48 | return -EINVAL; | ||
49 | size -= len; | ||
50 | |||
51 | if (acpi_dev->flags.compatible_ids) { | ||
52 | struct acpi_compatible_id_list *cid_list; | ||
53 | int i; | ||
54 | int count; | ||
55 | |||
56 | cid_list = acpi_dev->pnp.cid_list; | ||
57 | for (i = 0; i < cid_list->count; i++) { | ||
58 | count = snprintf(&modalias[len], size, "%s:", | ||
59 | cid_list->id[i].value); | ||
60 | if (count < 0 || count >= size) { | ||
61 | printk(KERN_ERR "acpi: %s cid[%i] exceeds event buffer size", | ||
62 | acpi_dev->pnp.device_name, i); | ||
63 | break; | ||
64 | } | ||
65 | len += count; | ||
66 | size -= count; | ||
67 | } | ||
68 | } | ||
69 | |||
70 | modalias[len] = '\0'; | ||
71 | return len; | ||
72 | } | ||
73 | |||
74 | static ssize_t | ||
75 | acpi_device_modalias_show(struct device *dev, struct device_attribute *attr, char *buf) { | ||
76 | struct acpi_device *acpi_dev = to_acpi_device(dev); | ||
77 | int len; | ||
78 | |||
79 | /* Device has no HID and no CID or string is >1024 */ | ||
80 | len = create_modalias(acpi_dev, buf, 1024); | ||
81 | if (len <= 0) | ||
82 | return 0; | ||
83 | buf[len++] = '\n'; | ||
84 | return len; | ||
85 | } | ||
86 | static DEVICE_ATTR(modalias, 0444, acpi_device_modalias_show, NULL); | ||
87 | |||
32 | static int acpi_eject_operation(acpi_handle handle, int lockable) | 88 | static int acpi_eject_operation(acpi_handle handle, int lockable) |
33 | { | 89 | { |
34 | struct acpi_object_list arg_list; | 90 | struct acpi_object_list arg_list; |
@@ -154,6 +210,12 @@ static int acpi_device_setup_files(struct acpi_device *dev) | |||
154 | goto end; | 210 | goto end; |
155 | } | 211 | } |
156 | 212 | ||
213 | if (dev->flags.hardware_id || dev->flags.compatible_ids){ | ||
214 | result = device_create_file(&dev->dev, &dev_attr_modalias); | ||
215 | if(result) | ||
216 | goto end; | ||
217 | } | ||
218 | |||
157 | /* | 219 | /* |
158 | * If device has _EJ0, 'eject' file is created that is used to trigger | 220 | * If device has _EJ0, 'eject' file is created that is used to trigger |
159 | * hot-removal function from userland. | 221 | * hot-removal function from userland. |
@@ -178,6 +240,9 @@ static void acpi_device_remove_files(struct acpi_device *dev) | |||
178 | if (ACPI_SUCCESS(status)) | 240 | if (ACPI_SUCCESS(status)) |
179 | device_remove_file(&dev->dev, &dev_attr_eject); | 241 | device_remove_file(&dev->dev, &dev_attr_eject); |
180 | 242 | ||
243 | if (dev->flags.hardware_id || dev->flags.compatible_ids) | ||
244 | device_remove_file(&dev->dev, &dev_attr_modalias); | ||
245 | |||
181 | if(dev->flags.hardware_id) | 246 | if(dev->flags.hardware_id) |
182 | device_remove_file(&dev->dev, &dev_attr_hid); | 247 | device_remove_file(&dev->dev, &dev_attr_hid); |
183 | if(dev->handle) | 248 | if(dev->handle) |
@@ -186,6 +251,37 @@ static void acpi_device_remove_files(struct acpi_device *dev) | |||
186 | /* -------------------------------------------------------------------------- | 251 | /* -------------------------------------------------------------------------- |
187 | ACPI Bus operations | 252 | ACPI Bus operations |
188 | -------------------------------------------------------------------------- */ | 253 | -------------------------------------------------------------------------- */ |
254 | |||
255 | int acpi_match_device_ids(struct acpi_device *device, | ||
256 | const struct acpi_device_id *ids) | ||
257 | { | ||
258 | const struct acpi_device_id *id; | ||
259 | |||
260 | if (device->flags.hardware_id) { | ||
261 | for (id = ids; id->id[0]; id++) { | ||
262 | if (!strcmp((char*)id->id, device->pnp.hardware_id)) | ||
263 | return 0; | ||
264 | } | ||
265 | } | ||
266 | |||
267 | if (device->flags.compatible_ids) { | ||
268 | struct acpi_compatible_id_list *cid_list = device->pnp.cid_list; | ||
269 | int i; | ||
270 | |||
271 | for (id = ids; id->id[0]; id++) { | ||
272 | /* compare multiple _CID entries against driver ids */ | ||
273 | for (i = 0; i < cid_list->count; i++) { | ||
274 | if (!strcmp((char*)id->id, | ||
275 | cid_list->id[i].value)) | ||
276 | return 0; | ||
277 | } | ||
278 | } | ||
279 | } | ||
280 | |||
281 | return -ENOENT; | ||
282 | } | ||
283 | EXPORT_SYMBOL(acpi_match_device_ids); | ||
284 | |||
189 | static void acpi_device_release(struct device *dev) | 285 | static void acpi_device_release(struct device *dev) |
190 | { | 286 | { |
191 | struct acpi_device *acpi_dev = to_acpi_device(dev); | 287 | struct acpi_device *acpi_dev = to_acpi_device(dev); |
@@ -219,37 +315,19 @@ static int acpi_bus_match(struct device *dev, struct device_driver *drv) | |||
219 | struct acpi_device *acpi_dev = to_acpi_device(dev); | 315 | struct acpi_device *acpi_dev = to_acpi_device(dev); |
220 | struct acpi_driver *acpi_drv = to_acpi_driver(drv); | 316 | struct acpi_driver *acpi_drv = to_acpi_driver(drv); |
221 | 317 | ||
222 | return !acpi_match_ids(acpi_dev, acpi_drv->ids); | 318 | return !acpi_match_device_ids(acpi_dev, acpi_drv->ids); |
223 | } | 319 | } |
224 | 320 | ||
225 | static int acpi_device_uevent(struct device *dev, char **envp, int num_envp, | 321 | static int acpi_device_uevent(struct device *dev, char **envp, int num_envp, |
226 | char *buffer, int buffer_size) | 322 | char *buffer, int buffer_size) |
227 | { | 323 | { |
228 | struct acpi_device *acpi_dev = to_acpi_device(dev); | 324 | struct acpi_device *acpi_dev = to_acpi_device(dev); |
229 | int i = 0, length = 0, ret = 0; | ||
230 | |||
231 | if (acpi_dev->flags.hardware_id) | ||
232 | ret = add_uevent_var(envp, num_envp, &i, | ||
233 | buffer, buffer_size, &length, | ||
234 | "HWID=%s", acpi_dev->pnp.hardware_id); | ||
235 | if (ret) | ||
236 | return -ENOMEM; | ||
237 | if (acpi_dev->flags.compatible_ids) { | ||
238 | int j; | ||
239 | struct acpi_compatible_id_list *cid_list; | ||
240 | 325 | ||
241 | cid_list = acpi_dev->pnp.cid_list; | 326 | strcpy(buffer, "MODALIAS="); |
242 | 327 | if (create_modalias(acpi_dev, buffer + 9, buffer_size - 9) > 0) { | |
243 | for (j = 0; j < cid_list->count; j++) { | 328 | envp[0] = buffer; |
244 | ret = add_uevent_var(envp, num_envp, &i, buffer, | 329 | envp[1] = NULL; |
245 | buffer_size, &length, "COMPTID=%s", | ||
246 | cid_list->id[j].value); | ||
247 | if (ret) | ||
248 | return -ENOMEM; | ||
249 | } | ||
250 | } | 330 | } |
251 | |||
252 | envp[i] = NULL; | ||
253 | return 0; | 331 | return 0; |
254 | } | 332 | } |
255 | 333 | ||
@@ -543,25 +621,6 @@ void acpi_bus_data_handler(acpi_handle handle, u32 function, void *context) | |||
543 | return; | 621 | return; |
544 | } | 622 | } |
545 | 623 | ||
546 | int acpi_match_ids(struct acpi_device *device, char *ids) | ||
547 | { | ||
548 | if (device->flags.hardware_id) | ||
549 | if (strstr(ids, device->pnp.hardware_id)) | ||
550 | return 0; | ||
551 | |||
552 | if (device->flags.compatible_ids) { | ||
553 | struct acpi_compatible_id_list *cid_list = device->pnp.cid_list; | ||
554 | int i; | ||
555 | |||
556 | /* compare multiple _CID entries against driver ids */ | ||
557 | for (i = 0; i < cid_list->count; i++) { | ||
558 | if (strstr(ids, cid_list->id[i].value)) | ||
559 | return 0; | ||
560 | } | ||
561 | } | ||
562 | return -ENOENT; | ||
563 | } | ||
564 | |||
565 | static int acpi_bus_get_perf_flags(struct acpi_device *device) | 624 | static int acpi_bus_get_perf_flags(struct acpi_device *device) |
566 | { | 625 | { |
567 | device->performance.state = ACPI_STATE_UNKNOWN; | 626 | device->performance.state = ACPI_STATE_UNKNOWN; |
@@ -624,6 +683,13 @@ static int acpi_bus_get_wakeup_device_flags(struct acpi_device *device) | |||
624 | struct acpi_buffer buffer = { ACPI_ALLOCATE_BUFFER, NULL }; | 683 | struct acpi_buffer buffer = { ACPI_ALLOCATE_BUFFER, NULL }; |
625 | union acpi_object *package = NULL; | 684 | union acpi_object *package = NULL; |
626 | 685 | ||
686 | struct acpi_device_id button_device_ids[] = { | ||
687 | {"PNP0C0D", 0}, | ||
688 | {"PNP0C0C", 0}, | ||
689 | {"PNP0C0E", 0}, | ||
690 | {"", 0}, | ||
691 | }; | ||
692 | |||
627 | 693 | ||
628 | /* _PRW */ | 694 | /* _PRW */ |
629 | status = acpi_evaluate_object(device->handle, "_PRW", NULL, &buffer); | 695 | status = acpi_evaluate_object(device->handle, "_PRW", NULL, &buffer); |
@@ -643,7 +709,7 @@ static int acpi_bus_get_wakeup_device_flags(struct acpi_device *device) | |||
643 | 709 | ||
644 | device->wakeup.flags.valid = 1; | 710 | device->wakeup.flags.valid = 1; |
645 | /* Power button, Lid switch always enable wakeup */ | 711 | /* Power button, Lid switch always enable wakeup */ |
646 | if (!acpi_match_ids(device, "PNP0C0D,PNP0C0C,PNP0C0E")) | 712 | if (!acpi_match_device_ids(device, button_device_ids)) |
647 | device->wakeup.flags.run_wake = 1; | 713 | device->wakeup.flags.run_wake = 1; |
648 | 714 | ||
649 | end: | 715 | end: |
diff --git a/drivers/acpi/sleep/Makefile b/drivers/acpi/sleep/Makefile index d6c017709c85..01a993a1d086 100644 --- a/drivers/acpi/sleep/Makefile +++ b/drivers/acpi/sleep/Makefile | |||
@@ -1,5 +1,5 @@ | |||
1 | obj-y := poweroff.o wakeup.o | 1 | obj-y := poweroff.o wakeup.o |
2 | obj-$(CONFIG_ACPI_SLEEP) += main.o | 2 | obj-y += main.o |
3 | obj-$(CONFIG_ACPI_SLEEP_PROC_FS) += proc.o | 3 | obj-$(CONFIG_X86) += proc.o |
4 | 4 | ||
5 | EXTRA_CFLAGS += $(ACPI_CFLAGS) | 5 | EXTRA_CFLAGS += $(ACPI_CFLAGS) |
diff --git a/drivers/acpi/sleep/main.c b/drivers/acpi/sleep/main.c index 3279e72a94f8..ab21357c5c7b 100644 --- a/drivers/acpi/sleep/main.c +++ b/drivers/acpi/sleep/main.c | |||
@@ -34,35 +34,55 @@ static u32 acpi_suspend_states[] = { | |||
34 | 34 | ||
35 | static int init_8259A_after_S1; | 35 | static int init_8259A_after_S1; |
36 | 36 | ||
37 | extern int acpi_sleep_prepare(u32 acpi_state); | ||
38 | extern void acpi_power_off(void); | ||
39 | |||
40 | static u32 acpi_target_sleep_state = ACPI_STATE_S0; | ||
41 | |||
42 | /** | ||
43 | * acpi_pm_set_target - Set the target system sleep state to the state | ||
44 | * associated with given @pm_state, if supported. | ||
45 | */ | ||
46 | |||
47 | static int acpi_pm_set_target(suspend_state_t pm_state) | ||
48 | { | ||
49 | u32 acpi_state = acpi_suspend_states[pm_state]; | ||
50 | int error = 0; | ||
51 | |||
52 | if (sleep_states[acpi_state]) { | ||
53 | acpi_target_sleep_state = acpi_state; | ||
54 | } else { | ||
55 | printk(KERN_ERR "ACPI does not support this state: %d\n", | ||
56 | pm_state); | ||
57 | error = -ENOSYS; | ||
58 | } | ||
59 | return error; | ||
60 | } | ||
61 | |||
37 | /** | 62 | /** |
38 | * acpi_pm_prepare - Do preliminary suspend work. | 63 | * acpi_pm_prepare - Do preliminary suspend work. |
39 | * @pm_state: suspend state we're entering. | 64 | * @pm_state: ignored |
40 | * | 65 | * |
41 | * Make sure we support the state. If we do, and we need it, set the | 66 | * If necessary, set the firmware waking vector and do arch-specific |
42 | * firmware waking vector and do arch-specific nastiness to get the | 67 | * nastiness to get the wakeup code to the waking vector. |
43 | * wakeup code to the waking vector. | ||
44 | */ | 68 | */ |
45 | 69 | ||
46 | extern int acpi_sleep_prepare(u32 acpi_state); | ||
47 | extern void acpi_power_off(void); | ||
48 | |||
49 | static int acpi_pm_prepare(suspend_state_t pm_state) | 70 | static int acpi_pm_prepare(suspend_state_t pm_state) |
50 | { | 71 | { |
51 | u32 acpi_state = acpi_suspend_states[pm_state]; | 72 | int error = acpi_sleep_prepare(acpi_target_sleep_state); |
52 | 73 | ||
53 | if (!sleep_states[acpi_state]) { | 74 | if (error) |
54 | printk("acpi_pm_prepare does not support %d \n", pm_state); | 75 | acpi_target_sleep_state = ACPI_STATE_S0; |
55 | return -EPERM; | 76 | |
56 | } | 77 | return error; |
57 | return acpi_sleep_prepare(acpi_state); | ||
58 | } | 78 | } |
59 | 79 | ||
60 | /** | 80 | /** |
61 | * acpi_pm_enter - Actually enter a sleep state. | 81 | * acpi_pm_enter - Actually enter a sleep state. |
62 | * @pm_state: State we're entering. | 82 | * @pm_state: ignored |
63 | * | 83 | * |
64 | * Flush caches and go to sleep. For STR or STD, we have to call | 84 | * Flush caches and go to sleep. For STR we have to call arch-specific |
65 | * arch-specific assembly, which in turn call acpi_enter_sleep_state(). | 85 | * assembly, which in turn call acpi_enter_sleep_state(). |
66 | * It's unfortunate, but it works. Please fix if you're feeling frisky. | 86 | * It's unfortunate, but it works. Please fix if you're feeling frisky. |
67 | */ | 87 | */ |
68 | 88 | ||
@@ -70,31 +90,31 @@ static int acpi_pm_enter(suspend_state_t pm_state) | |||
70 | { | 90 | { |
71 | acpi_status status = AE_OK; | 91 | acpi_status status = AE_OK; |
72 | unsigned long flags = 0; | 92 | unsigned long flags = 0; |
73 | u32 acpi_state = acpi_suspend_states[pm_state]; | 93 | u32 acpi_state = acpi_target_sleep_state; |
74 | 94 | ||
75 | ACPI_FLUSH_CPU_CACHE(); | 95 | ACPI_FLUSH_CPU_CACHE(); |
76 | 96 | ||
77 | /* Do arch specific saving of state. */ | 97 | /* Do arch specific saving of state. */ |
78 | if (pm_state > PM_SUSPEND_STANDBY) { | 98 | if (acpi_state == ACPI_STATE_S3) { |
79 | int error = acpi_save_state_mem(); | 99 | int error = acpi_save_state_mem(); |
80 | if (error) | 100 | |
101 | if (error) { | ||
102 | acpi_target_sleep_state = ACPI_STATE_S0; | ||
81 | return error; | 103 | return error; |
104 | } | ||
82 | } | 105 | } |
83 | 106 | ||
84 | local_irq_save(flags); | 107 | local_irq_save(flags); |
85 | acpi_enable_wakeup_device(acpi_state); | 108 | acpi_enable_wakeup_device(acpi_state); |
86 | switch (pm_state) { | 109 | switch (acpi_state) { |
87 | case PM_SUSPEND_STANDBY: | 110 | case ACPI_STATE_S1: |
88 | barrier(); | 111 | barrier(); |
89 | status = acpi_enter_sleep_state(acpi_state); | 112 | status = acpi_enter_sleep_state(acpi_state); |
90 | break; | 113 | break; |
91 | 114 | ||
92 | case PM_SUSPEND_MEM: | 115 | case ACPI_STATE_S3: |
93 | do_suspend_lowlevel(); | 116 | do_suspend_lowlevel(); |
94 | break; | 117 | break; |
95 | |||
96 | default: | ||
97 | return -EINVAL; | ||
98 | } | 118 | } |
99 | 119 | ||
100 | /* ACPI 3.0 specs (P62) says that it's the responsabilty | 120 | /* ACPI 3.0 specs (P62) says that it's the responsabilty |
@@ -107,12 +127,8 @@ static int acpi_pm_enter(suspend_state_t pm_state) | |||
107 | local_irq_restore(flags); | 127 | local_irq_restore(flags); |
108 | printk(KERN_DEBUG "Back to C!\n"); | 128 | printk(KERN_DEBUG "Back to C!\n"); |
109 | 129 | ||
110 | /* restore processor state | 130 | /* restore processor state */ |
111 | * We should only be here if we're coming back from STR or STD. | 131 | if (acpi_state == ACPI_STATE_S3) |
112 | * And, in the case of the latter, the memory image should have already | ||
113 | * been loaded from disk. | ||
114 | */ | ||
115 | if (pm_state > PM_SUSPEND_STANDBY) | ||
116 | acpi_restore_state_mem(); | 132 | acpi_restore_state_mem(); |
117 | 133 | ||
118 | return ACPI_SUCCESS(status) ? 0 : -EFAULT; | 134 | return ACPI_SUCCESS(status) ? 0 : -EFAULT; |
@@ -120,7 +136,7 @@ static int acpi_pm_enter(suspend_state_t pm_state) | |||
120 | 136 | ||
121 | /** | 137 | /** |
122 | * acpi_pm_finish - Finish up suspend sequence. | 138 | * acpi_pm_finish - Finish up suspend sequence. |
123 | * @pm_state: State we're coming out of. | 139 | * @pm_state: ignored |
124 | * | 140 | * |
125 | * This is called after we wake back up (or if entering the sleep state | 141 | * This is called after we wake back up (or if entering the sleep state |
126 | * failed). | 142 | * failed). |
@@ -128,7 +144,7 @@ static int acpi_pm_enter(suspend_state_t pm_state) | |||
128 | 144 | ||
129 | static int acpi_pm_finish(suspend_state_t pm_state) | 145 | static int acpi_pm_finish(suspend_state_t pm_state) |
130 | { | 146 | { |
131 | u32 acpi_state = acpi_suspend_states[pm_state]; | 147 | u32 acpi_state = acpi_target_sleep_state; |
132 | 148 | ||
133 | acpi_leave_sleep_state(acpi_state); | 149 | acpi_leave_sleep_state(acpi_state); |
134 | acpi_disable_wakeup_device(acpi_state); | 150 | acpi_disable_wakeup_device(acpi_state); |
@@ -136,10 +152,14 @@ static int acpi_pm_finish(suspend_state_t pm_state) | |||
136 | /* reset firmware waking vector */ | 152 | /* reset firmware waking vector */ |
137 | acpi_set_firmware_waking_vector((acpi_physical_address) 0); | 153 | acpi_set_firmware_waking_vector((acpi_physical_address) 0); |
138 | 154 | ||
155 | acpi_target_sleep_state = ACPI_STATE_S0; | ||
156 | |||
157 | #ifdef CONFIG_X86 | ||
139 | if (init_8259A_after_S1) { | 158 | if (init_8259A_after_S1) { |
140 | printk("Broken toshiba laptop -> kicking interrupts\n"); | 159 | printk("Broken toshiba laptop -> kicking interrupts\n"); |
141 | init_8259A(0); | 160 | init_8259A(0); |
142 | } | 161 | } |
162 | #endif | ||
143 | return 0; | 163 | return 0; |
144 | } | 164 | } |
145 | 165 | ||
@@ -176,6 +196,7 @@ static int acpi_pm_state_valid(suspend_state_t pm_state) | |||
176 | 196 | ||
177 | static struct pm_ops acpi_pm_ops = { | 197 | static struct pm_ops acpi_pm_ops = { |
178 | .valid = acpi_pm_state_valid, | 198 | .valid = acpi_pm_state_valid, |
199 | .set_target = acpi_pm_set_target, | ||
179 | .prepare = acpi_pm_prepare, | 200 | .prepare = acpi_pm_prepare, |
180 | .enter = acpi_pm_enter, | 201 | .enter = acpi_pm_enter, |
181 | .finish = acpi_pm_finish, | 202 | .finish = acpi_pm_finish, |
@@ -235,6 +256,81 @@ static struct hibernation_ops acpi_hibernation_ops = { | |||
235 | }; | 256 | }; |
236 | #endif /* CONFIG_SOFTWARE_SUSPEND */ | 257 | #endif /* CONFIG_SOFTWARE_SUSPEND */ |
237 | 258 | ||
259 | /** | ||
260 | * acpi_pm_device_sleep_state - return preferred power state of ACPI device | ||
261 | * in the system sleep state given by %acpi_target_sleep_state | ||
262 | * @dev: device to examine | ||
263 | * @wake: if set, the device should be able to wake up the system | ||
264 | * @d_min_p: used to store the upper limit of allowed states range | ||
265 | * Return value: preferred power state of the device on success, -ENODEV on | ||
266 | * failure (ie. if there's no 'struct acpi_device' for @dev) | ||
267 | * | ||
268 | * Find the lowest power (highest number) ACPI device power state that | ||
269 | * device @dev can be in while the system is in the sleep state represented | ||
270 | * by %acpi_target_sleep_state. If @wake is nonzero, the device should be | ||
271 | * able to wake up the system from this sleep state. If @d_min_p is set, | ||
272 | * the highest power (lowest number) device power state of @dev allowed | ||
273 | * in this system sleep state is stored at the location pointed to by it. | ||
274 | * | ||
275 | * The caller must ensure that @dev is valid before using this function. | ||
276 | * The caller is also responsible for figuring out if the device is | ||
277 | * supposed to be able to wake up the system and passing this information | ||
278 | * via @wake. | ||
279 | */ | ||
280 | |||
281 | int acpi_pm_device_sleep_state(struct device *dev, int wake, int *d_min_p) | ||
282 | { | ||
283 | acpi_handle handle = DEVICE_ACPI_HANDLE(dev); | ||
284 | struct acpi_device *adev; | ||
285 | char acpi_method[] = "_SxD"; | ||
286 | unsigned long d_min, d_max; | ||
287 | |||
288 | if (!handle || ACPI_FAILURE(acpi_bus_get_device(handle, &adev))) { | ||
289 | printk(KERN_ERR "ACPI handle has no context!\n"); | ||
290 | return -ENODEV; | ||
291 | } | ||
292 | |||
293 | acpi_method[2] = '0' + acpi_target_sleep_state; | ||
294 | /* | ||
295 | * If the sleep state is S0, we will return D3, but if the device has | ||
296 | * _S0W, we will use the value from _S0W | ||
297 | */ | ||
298 | d_min = ACPI_STATE_D0; | ||
299 | d_max = ACPI_STATE_D3; | ||
300 | |||
301 | /* | ||
302 | * If present, _SxD methods return the minimum D-state (highest power | ||
303 | * state) we can use for the corresponding S-states. Otherwise, the | ||
304 | * minimum D-state is D0 (ACPI 3.x). | ||
305 | * | ||
306 | * NOTE: We rely on acpi_evaluate_integer() not clobbering the integer | ||
307 | * provided -- that's our fault recovery, we ignore retval. | ||
308 | */ | ||
309 | if (acpi_target_sleep_state > ACPI_STATE_S0) | ||
310 | acpi_evaluate_integer(handle, acpi_method, NULL, &d_min); | ||
311 | |||
312 | /* | ||
313 | * If _PRW says we can wake up the system from the target sleep state, | ||
314 | * the D-state returned by _SxD is sufficient for that (we assume a | ||
315 | * wakeup-aware driver if wake is set). Still, if _SxW exists | ||
316 | * (ACPI 3.x), it should return the maximum (lowest power) D-state that | ||
317 | * can wake the system. _S0W may be valid, too. | ||
318 | */ | ||
319 | if (acpi_target_sleep_state == ACPI_STATE_S0 || | ||
320 | (wake && adev->wakeup.state.enabled && | ||
321 | adev->wakeup.sleep_state <= acpi_target_sleep_state)) { | ||
322 | acpi_method[3] = 'W'; | ||
323 | acpi_evaluate_integer(handle, acpi_method, NULL, &d_max); | ||
324 | /* Sanity check */ | ||
325 | if (d_max < d_min) | ||
326 | d_min = d_max; | ||
327 | } | ||
328 | |||
329 | if (d_min_p) | ||
330 | *d_min_p = d_min; | ||
331 | return d_max; | ||
332 | } | ||
333 | |||
238 | /* | 334 | /* |
239 | * Toshiba fails to preserve interrupts over S1, reinitialization | 335 | * Toshiba fails to preserve interrupts over S1, reinitialization |
240 | * of 8259 is needed after S1 resume. | 336 | * of 8259 is needed after S1 resume. |
diff --git a/drivers/acpi/sleep/poweroff.c b/drivers/acpi/sleep/poweroff.c index 39e40d56b034..b3f68ef0669e 100644 --- a/drivers/acpi/sleep/poweroff.c +++ b/drivers/acpi/sleep/poweroff.c | |||
@@ -18,7 +18,6 @@ | |||
18 | 18 | ||
19 | int acpi_sleep_prepare(u32 acpi_state) | 19 | int acpi_sleep_prepare(u32 acpi_state) |
20 | { | 20 | { |
21 | #ifdef CONFIG_ACPI_SLEEP | ||
22 | /* do we have a wakeup address for S2 and S3? */ | 21 | /* do we have a wakeup address for S2 and S3? */ |
23 | if (acpi_state == ACPI_STATE_S3) { | 22 | if (acpi_state == ACPI_STATE_S3) { |
24 | if (!acpi_wakeup_address) { | 23 | if (!acpi_wakeup_address) { |
@@ -31,7 +30,6 @@ int acpi_sleep_prepare(u32 acpi_state) | |||
31 | } | 30 | } |
32 | ACPI_FLUSH_CPU_CACHE(); | 31 | ACPI_FLUSH_CPU_CACHE(); |
33 | acpi_enable_wakeup_device_prep(acpi_state); | 32 | acpi_enable_wakeup_device_prep(acpi_state); |
34 | #endif | ||
35 | acpi_gpe_sleep_prepare(acpi_state); | 33 | acpi_gpe_sleep_prepare(acpi_state); |
36 | acpi_enter_sleep_state_prep(acpi_state); | 34 | acpi_enter_sleep_state_prep(acpi_state); |
37 | return 0; | 35 | return 0; |
diff --git a/drivers/acpi/sleep/proc.c b/drivers/acpi/sleep/proc.c index 61f1822cc350..ed58e1168aed 100644 --- a/drivers/acpi/sleep/proc.c +++ b/drivers/acpi/sleep/proc.c | |||
@@ -14,8 +14,16 @@ | |||
14 | #include "sleep.h" | 14 | #include "sleep.h" |
15 | 15 | ||
16 | #define _COMPONENT ACPI_SYSTEM_COMPONENT | 16 | #define _COMPONENT ACPI_SYSTEM_COMPONENT |
17 | |||
18 | /* | ||
19 | * this file provides support for: | ||
20 | * /proc/acpi/sleep | ||
21 | * /proc/acpi/alarm | ||
22 | * /proc/acpi/wakeup | ||
23 | */ | ||
24 | |||
17 | ACPI_MODULE_NAME("sleep") | 25 | ACPI_MODULE_NAME("sleep") |
18 | #ifdef CONFIG_ACPI_SLEEP_PROC_SLEEP | 26 | #ifdef CONFIG_ACPI_PROCFS |
19 | static int acpi_system_sleep_seq_show(struct seq_file *seq, void *offset) | 27 | static int acpi_system_sleep_seq_show(struct seq_file *seq, void *offset) |
20 | { | 28 | { |
21 | int i; | 29 | int i; |
@@ -68,7 +76,7 @@ acpi_system_write_sleep(struct file *file, | |||
68 | Done: | 76 | Done: |
69 | return error ? error : count; | 77 | return error ? error : count; |
70 | } | 78 | } |
71 | #endif /* CONFIG_ACPI_SLEEP_PROC_SLEEP */ | 79 | #endif /* CONFIG_ACPI_PROCFS */ |
72 | 80 | ||
73 | #if defined(CONFIG_RTC_DRV_CMOS) || defined(CONFIG_RTC_DRV_CMOS_MODULE) | 81 | #if defined(CONFIG_RTC_DRV_CMOS) || defined(CONFIG_RTC_DRV_CMOS_MODULE) |
74 | /* use /sys/class/rtc/rtcX/wakealarm instead; it's not ACPI-specific */ | 82 | /* use /sys/class/rtc/rtcX/wakealarm instead; it's not ACPI-specific */ |
@@ -463,7 +471,7 @@ static const struct file_operations acpi_system_wakeup_device_fops = { | |||
463 | .release = single_release, | 471 | .release = single_release, |
464 | }; | 472 | }; |
465 | 473 | ||
466 | #ifdef CONFIG_ACPI_SLEEP_PROC_SLEEP | 474 | #ifdef CONFIG_ACPI_PROCFS |
467 | static const struct file_operations acpi_system_sleep_fops = { | 475 | static const struct file_operations acpi_system_sleep_fops = { |
468 | .open = acpi_system_sleep_open_fs, | 476 | .open = acpi_system_sleep_open_fs, |
469 | .read = seq_read, | 477 | .read = seq_read, |
@@ -471,7 +479,7 @@ static const struct file_operations acpi_system_sleep_fops = { | |||
471 | .llseek = seq_lseek, | 479 | .llseek = seq_lseek, |
472 | .release = single_release, | 480 | .release = single_release, |
473 | }; | 481 | }; |
474 | #endif /* CONFIG_ACPI_SLEEP_PROC_SLEEP */ | 482 | #endif /* CONFIG_ACPI_PROCFS */ |
475 | 483 | ||
476 | #ifdef HAVE_ACPI_LEGACY_ALARM | 484 | #ifdef HAVE_ACPI_LEGACY_ALARM |
477 | static const struct file_operations acpi_system_alarm_fops = { | 485 | static const struct file_operations acpi_system_alarm_fops = { |
@@ -498,14 +506,14 @@ static int __init acpi_sleep_proc_init(void) | |||
498 | if (acpi_disabled) | 506 | if (acpi_disabled) |
499 | return 0; | 507 | return 0; |
500 | 508 | ||
501 | #ifdef CONFIG_ACPI_SLEEP_PROC_SLEEP | 509 | #ifdef CONFIG_ACPI_PROCFS |
502 | /* 'sleep' [R/W] */ | 510 | /* 'sleep' [R/W] */ |
503 | entry = | 511 | entry = |
504 | create_proc_entry("sleep", S_IFREG | S_IRUGO | S_IWUSR, | 512 | create_proc_entry("sleep", S_IFREG | S_IRUGO | S_IWUSR, |
505 | acpi_root_dir); | 513 | acpi_root_dir); |
506 | if (entry) | 514 | if (entry) |
507 | entry->proc_fops = &acpi_system_sleep_fops; | 515 | entry->proc_fops = &acpi_system_sleep_fops; |
508 | #endif | 516 | #endif /* CONFIG_ACPI_PROCFS */ |
509 | 517 | ||
510 | #ifdef HAVE_ACPI_LEGACY_ALARM | 518 | #ifdef HAVE_ACPI_LEGACY_ALARM |
511 | /* 'alarm' [R/W] */ | 519 | /* 'alarm' [R/W] */ |
diff --git a/drivers/acpi/sleep/wakeup.c b/drivers/acpi/sleep/wakeup.c index fab8f2694f03..97c27ddb144d 100644 --- a/drivers/acpi/sleep/wakeup.c +++ b/drivers/acpi/sleep/wakeup.c | |||
@@ -17,7 +17,6 @@ ACPI_MODULE_NAME("wakeup_devices") | |||
17 | extern struct list_head acpi_wakeup_device_list; | 17 | extern struct list_head acpi_wakeup_device_list; |
18 | extern spinlock_t acpi_device_lock; | 18 | extern spinlock_t acpi_device_lock; |
19 | 19 | ||
20 | #ifdef CONFIG_ACPI_SLEEP | ||
21 | /** | 20 | /** |
22 | * acpi_enable_wakeup_device_prep - prepare wakeup devices | 21 | * acpi_enable_wakeup_device_prep - prepare wakeup devices |
23 | * @sleep_state: ACPI state | 22 | * @sleep_state: ACPI state |
@@ -180,7 +179,6 @@ static int __init acpi_wakeup_device_init(void) | |||
180 | } | 179 | } |
181 | 180 | ||
182 | late_initcall(acpi_wakeup_device_init); | 181 | late_initcall(acpi_wakeup_device_init); |
183 | #endif | ||
184 | 182 | ||
185 | /* | 183 | /* |
186 | * Disable all wakeup GPEs before entering requested sleep state. | 184 | * Disable all wakeup GPEs before entering requested sleep state. |
diff --git a/drivers/acpi/thermal.c b/drivers/acpi/thermal.c index 58f1338981bc..5a62de1b7f2a 100644 --- a/drivers/acpi/thermal.c +++ b/drivers/acpi/thermal.c | |||
@@ -92,10 +92,16 @@ static int acpi_thermal_polling_open_fs(struct inode *inode, struct file *file); | |||
92 | static ssize_t acpi_thermal_write_polling(struct file *, const char __user *, | 92 | static ssize_t acpi_thermal_write_polling(struct file *, const char __user *, |
93 | size_t, loff_t *); | 93 | size_t, loff_t *); |
94 | 94 | ||
95 | static const struct acpi_device_id thermal_device_ids[] = { | ||
96 | {ACPI_THERMAL_HID, 0}, | ||
97 | {"", 0}, | ||
98 | }; | ||
99 | MODULE_DEVICE_TABLE(acpi, thermal_device_ids); | ||
100 | |||
95 | static struct acpi_driver acpi_thermal_driver = { | 101 | static struct acpi_driver acpi_thermal_driver = { |
96 | .name = "thermal", | 102 | .name = "thermal", |
97 | .class = ACPI_THERMAL_CLASS, | 103 | .class = ACPI_THERMAL_CLASS, |
98 | .ids = ACPI_THERMAL_HID, | 104 | .ids = thermal_device_ids, |
99 | .ops = { | 105 | .ops = { |
100 | .add = acpi_thermal_add, | 106 | .add = acpi_thermal_add, |
101 | .remove = acpi_thermal_remove, | 107 | .remove = acpi_thermal_remove, |
diff --git a/drivers/acpi/utilities/uteval.c b/drivers/acpi/utilities/uteval.c index f112af433e36..0042b7e78b26 100644 --- a/drivers/acpi/utilities/uteval.c +++ b/drivers/acpi/utilities/uteval.c | |||
@@ -407,7 +407,7 @@ acpi_ut_copy_id_string(char *destination, char *source, acpi_size max_length) | |||
407 | 407 | ||
408 | acpi_status | 408 | acpi_status |
409 | acpi_ut_execute_HID(struct acpi_namespace_node *device_node, | 409 | acpi_ut_execute_HID(struct acpi_namespace_node *device_node, |
410 | struct acpi_device_id *hid) | 410 | struct acpica_device_id *hid) |
411 | { | 411 | { |
412 | union acpi_operand_object *obj_desc; | 412 | union acpi_operand_object *obj_desc; |
413 | acpi_status status; | 413 | acpi_status status; |
@@ -609,7 +609,7 @@ acpi_ut_execute_CID(struct acpi_namespace_node * device_node, | |||
609 | 609 | ||
610 | acpi_status | 610 | acpi_status |
611 | acpi_ut_execute_UID(struct acpi_namespace_node *device_node, | 611 | acpi_ut_execute_UID(struct acpi_namespace_node *device_node, |
612 | struct acpi_device_id *uid) | 612 | struct acpica_device_id *uid) |
613 | { | 613 | { |
614 | union acpi_operand_object *obj_desc; | 614 | union acpi_operand_object *obj_desc; |
615 | acpi_status status; | 615 | acpi_status status; |
diff --git a/drivers/acpi/video.c b/drivers/acpi/video.c index 04ea697f72bf..d98701941981 100644 --- a/drivers/acpi/video.c +++ b/drivers/acpi/video.c | |||
@@ -74,10 +74,16 @@ MODULE_LICENSE("GPL"); | |||
74 | static int acpi_video_bus_add(struct acpi_device *device); | 74 | static int acpi_video_bus_add(struct acpi_device *device); |
75 | static int acpi_video_bus_remove(struct acpi_device *device, int type); | 75 | static int acpi_video_bus_remove(struct acpi_device *device, int type); |
76 | 76 | ||
77 | static const struct acpi_device_id video_device_ids[] = { | ||
78 | {ACPI_VIDEO_HID, 0}, | ||
79 | {"", 0}, | ||
80 | }; | ||
81 | MODULE_DEVICE_TABLE(acpi, video_device_ids); | ||
82 | |||
77 | static struct acpi_driver acpi_video_bus = { | 83 | static struct acpi_driver acpi_video_bus = { |
78 | .name = "video", | 84 | .name = "video", |
79 | .class = ACPI_VIDEO_CLASS, | 85 | .class = ACPI_VIDEO_CLASS, |
80 | .ids = ACPI_VIDEO_HID, | 86 | .ids = video_device_ids, |
81 | .ops = { | 87 | .ops = { |
82 | .add = acpi_video_bus_add, | 88 | .add = acpi_video_bus_add, |
83 | .remove = acpi_video_bus_remove, | 89 | .remove = acpi_video_bus_remove, |
diff --git a/drivers/ata/ata_piix.c b/drivers/ata/ata_piix.c index d9fa329fd157..ad070861bb53 100644 --- a/drivers/ata/ata_piix.c +++ b/drivers/ata/ata_piix.c | |||
@@ -91,6 +91,7 @@ | |||
91 | #include <linux/device.h> | 91 | #include <linux/device.h> |
92 | #include <scsi/scsi_host.h> | 92 | #include <scsi/scsi_host.h> |
93 | #include <linux/libata.h> | 93 | #include <linux/libata.h> |
94 | #include <linux/dmi.h> | ||
94 | 95 | ||
95 | #define DRV_NAME "ata_piix" | 96 | #define DRV_NAME "ata_piix" |
96 | #define DRV_VERSION "2.11" | 97 | #define DRV_VERSION "2.11" |
@@ -140,6 +141,9 @@ enum { | |||
140 | RV = -3, /* reserved */ | 141 | RV = -3, /* reserved */ |
141 | 142 | ||
142 | PIIX_AHCI_DEVICE = 6, | 143 | PIIX_AHCI_DEVICE = 6, |
144 | |||
145 | /* host->flags bits */ | ||
146 | PIIX_HOST_BROKEN_SUSPEND = (1 << 24), | ||
143 | }; | 147 | }; |
144 | 148 | ||
145 | struct piix_map_db { | 149 | struct piix_map_db { |
@@ -159,6 +163,10 @@ static void piix_set_piomode (struct ata_port *ap, struct ata_device *adev); | |||
159 | static void piix_set_dmamode (struct ata_port *ap, struct ata_device *adev); | 163 | static void piix_set_dmamode (struct ata_port *ap, struct ata_device *adev); |
160 | static void ich_set_dmamode (struct ata_port *ap, struct ata_device *adev); | 164 | static void ich_set_dmamode (struct ata_port *ap, struct ata_device *adev); |
161 | static int ich_pata_cable_detect(struct ata_port *ap); | 165 | static int ich_pata_cable_detect(struct ata_port *ap); |
166 | #ifdef CONFIG_PM | ||
167 | static int piix_pci_device_suspend(struct pci_dev *pdev, pm_message_t mesg); | ||
168 | static int piix_pci_device_resume(struct pci_dev *pdev); | ||
169 | #endif | ||
162 | 170 | ||
163 | static unsigned int in_module_init = 1; | 171 | static unsigned int in_module_init = 1; |
164 | 172 | ||
@@ -255,8 +263,8 @@ static struct pci_driver piix_pci_driver = { | |||
255 | .probe = piix_init_one, | 263 | .probe = piix_init_one, |
256 | .remove = ata_pci_remove_one, | 264 | .remove = ata_pci_remove_one, |
257 | #ifdef CONFIG_PM | 265 | #ifdef CONFIG_PM |
258 | .suspend = ata_pci_device_suspend, | 266 | .suspend = piix_pci_device_suspend, |
259 | .resume = ata_pci_device_resume, | 267 | .resume = piix_pci_device_resume, |
260 | #endif | 268 | #endif |
261 | }; | 269 | }; |
262 | 270 | ||
@@ -881,6 +889,107 @@ static void ich_set_dmamode (struct ata_port *ap, struct ata_device *adev) | |||
881 | do_pata_set_dmamode(ap, adev, 1); | 889 | do_pata_set_dmamode(ap, adev, 1); |
882 | } | 890 | } |
883 | 891 | ||
892 | #ifdef CONFIG_PM | ||
893 | static struct dmi_system_id piix_broken_suspend_dmi_table[] = { | ||
894 | { | ||
895 | .ident = "TECRA M5", | ||
896 | .matches = { | ||
897 | DMI_MATCH(DMI_SYS_VENDOR, "TOSHIBA"), | ||
898 | DMI_MATCH(DMI_PRODUCT_NAME, "TECRA M5"), | ||
899 | }, | ||
900 | }, | ||
901 | { | ||
902 | .ident = "Satellite U200", | ||
903 | .matches = { | ||
904 | DMI_MATCH(DMI_SYS_VENDOR, "TOSHIBA"), | ||
905 | DMI_MATCH(DMI_PRODUCT_NAME, "Satellite U200"), | ||
906 | }, | ||
907 | }, | ||
908 | { | ||
909 | .ident = "Satellite U205", | ||
910 | .matches = { | ||
911 | DMI_MATCH(DMI_SYS_VENDOR, "TOSHIBA"), | ||
912 | DMI_MATCH(DMI_PRODUCT_NAME, "Satellite U205"), | ||
913 | }, | ||
914 | }, | ||
915 | { | ||
916 | .ident = "Portege M500", | ||
917 | .matches = { | ||
918 | DMI_MATCH(DMI_SYS_VENDOR, "TOSHIBA"), | ||
919 | DMI_MATCH(DMI_PRODUCT_NAME, "PORTEGE M500"), | ||
920 | }, | ||
921 | }, | ||
922 | { } | ||
923 | }; | ||
924 | |||
925 | static int piix_pci_device_suspend(struct pci_dev *pdev, pm_message_t mesg) | ||
926 | { | ||
927 | struct ata_host *host = dev_get_drvdata(&pdev->dev); | ||
928 | unsigned long flags; | ||
929 | int rc = 0; | ||
930 | |||
931 | rc = ata_host_suspend(host, mesg); | ||
932 | if (rc) | ||
933 | return rc; | ||
934 | |||
935 | /* Some braindamaged ACPI suspend implementations expect the | ||
936 | * controller to be awake on entry; otherwise, it burns cpu | ||
937 | * cycles and power trying to do something to the sleeping | ||
938 | * beauty. | ||
939 | */ | ||
940 | if (dmi_check_system(piix_broken_suspend_dmi_table) && | ||
941 | mesg.event == PM_EVENT_SUSPEND) { | ||
942 | pci_save_state(pdev); | ||
943 | |||
944 | /* mark its power state as "unknown", since we don't | ||
945 | * know if e.g. the BIOS will change its device state | ||
946 | * when we suspend. | ||
947 | */ | ||
948 | if (pdev->current_state == PCI_D0) | ||
949 | pdev->current_state = PCI_UNKNOWN; | ||
950 | |||
951 | /* tell resume that it's waking up from broken suspend */ | ||
952 | spin_lock_irqsave(&host->lock, flags); | ||
953 | host->flags |= PIIX_HOST_BROKEN_SUSPEND; | ||
954 | spin_unlock_irqrestore(&host->lock, flags); | ||
955 | } else | ||
956 | ata_pci_device_do_suspend(pdev, mesg); | ||
957 | |||
958 | return 0; | ||
959 | } | ||
960 | |||
961 | static int piix_pci_device_resume(struct pci_dev *pdev) | ||
962 | { | ||
963 | struct ata_host *host = dev_get_drvdata(&pdev->dev); | ||
964 | unsigned long flags; | ||
965 | int rc; | ||
966 | |||
967 | if (host->flags & PIIX_HOST_BROKEN_SUSPEND) { | ||
968 | spin_lock_irqsave(&host->lock, flags); | ||
969 | host->flags &= ~PIIX_HOST_BROKEN_SUSPEND; | ||
970 | spin_unlock_irqrestore(&host->lock, flags); | ||
971 | |||
972 | pci_set_power_state(pdev, PCI_D0); | ||
973 | pci_restore_state(pdev); | ||
974 | |||
975 | /* PCI device wasn't disabled during suspend. Use | ||
976 | * __pci_reenable_device() to avoid affecting the | ||
977 | * enable count. | ||
978 | */ | ||
979 | rc = __pci_reenable_device(pdev); | ||
980 | if (rc) | ||
981 | dev_printk(KERN_ERR, &pdev->dev, "failed to enable " | ||
982 | "device after resume (%d)\n", rc); | ||
983 | } else | ||
984 | rc = ata_pci_device_do_resume(pdev); | ||
985 | |||
986 | if (rc == 0) | ||
987 | ata_host_resume(host); | ||
988 | |||
989 | return rc; | ||
990 | } | ||
991 | #endif | ||
992 | |||
884 | #define AHCI_PCI_BAR 5 | 993 | #define AHCI_PCI_BAR 5 |
885 | #define AHCI_GLOBAL_CTL 0x04 | 994 | #define AHCI_GLOBAL_CTL 0x04 |
886 | #define AHCI_ENABLE (1 << 31) | 995 | #define AHCI_ENABLE (1 << 31) |
diff --git a/drivers/ata/libata-scsi.c b/drivers/ata/libata-scsi.c index 12ac0b511f79..e83647651b31 100644 --- a/drivers/ata/libata-scsi.c +++ b/drivers/ata/libata-scsi.c | |||
@@ -768,7 +768,7 @@ static void ata_scsi_dev_config(struct scsi_device *sdev, | |||
768 | * Decrement max hw segments accordingly. | 768 | * Decrement max hw segments accordingly. |
769 | */ | 769 | */ |
770 | if (dev->class == ATA_DEV_ATAPI) { | 770 | if (dev->class == ATA_DEV_ATAPI) { |
771 | request_queue_t *q = sdev->request_queue; | 771 | struct request_queue *q = sdev->request_queue; |
772 | blk_queue_max_hw_segments(q, q->max_hw_segments - 1); | 772 | blk_queue_max_hw_segments(q, q->max_hw_segments - 1); |
773 | } | 773 | } |
774 | 774 | ||
diff --git a/drivers/ata/pata_ali.c b/drivers/ata/pata_ali.c index 010436795d20..e8a28e94fe47 100644 --- a/drivers/ata/pata_ali.c +++ b/drivers/ata/pata_ali.c | |||
@@ -45,7 +45,7 @@ static struct dmi_system_id cable_dmi_table[] = { | |||
45 | .ident = "HP Pavilion N5430", | 45 | .ident = "HP Pavilion N5430", |
46 | .matches = { | 46 | .matches = { |
47 | DMI_MATCH(DMI_BOARD_VENDOR, "Hewlett-Packard"), | 47 | DMI_MATCH(DMI_BOARD_VENDOR, "Hewlett-Packard"), |
48 | DMI_MATCH(DMI_BOARD_NAME, "OmniBook N32N-736"), | 48 | DMI_MATCH(DMI_BOARD_VERSION, "OmniBook N32N-736"), |
49 | }, | 49 | }, |
50 | }, | 50 | }, |
51 | { } | 51 | { } |
diff --git a/drivers/ata/pata_hpt37x.c b/drivers/ata/pata_hpt37x.c index b0af65aadde3..84d9c5568567 100644 --- a/drivers/ata/pata_hpt37x.c +++ b/drivers/ata/pata_hpt37x.c | |||
@@ -26,7 +26,7 @@ | |||
26 | #include <linux/libata.h> | 26 | #include <linux/libata.h> |
27 | 27 | ||
28 | #define DRV_NAME "pata_hpt37x" | 28 | #define DRV_NAME "pata_hpt37x" |
29 | #define DRV_VERSION "0.6.6" | 29 | #define DRV_VERSION "0.6.7" |
30 | 30 | ||
31 | struct hpt_clock { | 31 | struct hpt_clock { |
32 | u8 xfer_speed; | 32 | u8 xfer_speed; |
@@ -1103,17 +1103,17 @@ static int hpt37x_init_one(struct pci_dev *dev, const struct pci_device_id *id) | |||
1103 | 1103 | ||
1104 | /* Select the DPLL clock. */ | 1104 | /* Select the DPLL clock. */ |
1105 | pci_write_config_byte(dev, 0x5b, 0x21); | 1105 | pci_write_config_byte(dev, 0x5b, 0x21); |
1106 | pci_write_config_dword(dev, 0x5C, (f_high << 16) | f_low); | 1106 | pci_write_config_dword(dev, 0x5C, (f_high << 16) | f_low | 0x100); |
1107 | 1107 | ||
1108 | for(adjust = 0; adjust < 8; adjust++) { | 1108 | for(adjust = 0; adjust < 8; adjust++) { |
1109 | if (hpt37x_calibrate_dpll(dev)) | 1109 | if (hpt37x_calibrate_dpll(dev)) |
1110 | break; | 1110 | break; |
1111 | /* See if it'll settle at a fractionally different clock */ | 1111 | /* See if it'll settle at a fractionally different clock */ |
1112 | if ((adjust & 3) == 3) { | 1112 | if (adjust & 1) |
1113 | f_low --; | 1113 | f_low -= adjust >> 1; |
1114 | f_high ++; | 1114 | else |
1115 | } | 1115 | f_high += adjust >> 1; |
1116 | pci_write_config_dword(dev, 0x5C, (f_high << 16) | f_low); | 1116 | pci_write_config_dword(dev, 0x5C, (f_high << 16) | f_low | 0x100); |
1117 | } | 1117 | } |
1118 | if (adjust == 8) { | 1118 | if (adjust == 8) { |
1119 | printk(KERN_WARNING "hpt37x: DPLL did not stabilize.\n"); | 1119 | printk(KERN_WARNING "hpt37x: DPLL did not stabilize.\n"); |
diff --git a/drivers/base/power/shutdown.c b/drivers/base/power/shutdown.c index a47ee1b70d20..56e8eaaac012 100644 --- a/drivers/base/power/shutdown.c +++ b/drivers/base/power/shutdown.c | |||
@@ -44,7 +44,5 @@ void device_shutdown(void) | |||
44 | dev->driver->shutdown(dev); | 44 | dev->driver->shutdown(dev); |
45 | } | 45 | } |
46 | } | 46 | } |
47 | |||
48 | sysdev_shutdown(); | ||
49 | } | 47 | } |
50 | 48 | ||
diff --git a/drivers/block/amiflop.c b/drivers/block/amiflop.c index 6ce8b897e262..c9751b2b57e6 100644 --- a/drivers/block/amiflop.c +++ b/drivers/block/amiflop.c | |||
@@ -1422,7 +1422,7 @@ static void redo_fd_request(void) | |||
1422 | goto repeat; | 1422 | goto repeat; |
1423 | } | 1423 | } |
1424 | 1424 | ||
1425 | static void do_fd_request(request_queue_t * q) | 1425 | static void do_fd_request(struct request_queue * q) |
1426 | { | 1426 | { |
1427 | redo_fd_request(); | 1427 | redo_fd_request(); |
1428 | } | 1428 | } |
diff --git a/drivers/block/aoe/aoe.h b/drivers/block/aoe/aoe.h index 1d8466817943..ba07f762c4cb 100644 --- a/drivers/block/aoe/aoe.h +++ b/drivers/block/aoe/aoe.h | |||
@@ -138,7 +138,7 @@ struct aoedev { | |||
138 | u16 maxbcnt; | 138 | u16 maxbcnt; |
139 | struct work_struct work;/* disk create work struct */ | 139 | struct work_struct work;/* disk create work struct */ |
140 | struct gendisk *gd; | 140 | struct gendisk *gd; |
141 | request_queue_t blkq; | 141 | struct request_queue blkq; |
142 | struct hd_geometry geo; | 142 | struct hd_geometry geo; |
143 | sector_t ssize; | 143 | sector_t ssize; |
144 | struct timer_list timer; | 144 | struct timer_list timer; |
diff --git a/drivers/block/aoe/aoeblk.c b/drivers/block/aoe/aoeblk.c index 4f598270fa31..007faaf008e7 100644 --- a/drivers/block/aoe/aoeblk.c +++ b/drivers/block/aoe/aoeblk.c | |||
@@ -125,7 +125,7 @@ aoeblk_release(struct inode *inode, struct file *filp) | |||
125 | } | 125 | } |
126 | 126 | ||
127 | static int | 127 | static int |
128 | aoeblk_make_request(request_queue_t *q, struct bio *bio) | 128 | aoeblk_make_request(struct request_queue *q, struct bio *bio) |
129 | { | 129 | { |
130 | struct aoedev *d; | 130 | struct aoedev *d; |
131 | struct buf *buf; | 131 | struct buf *buf; |
diff --git a/drivers/block/ataflop.c b/drivers/block/ataflop.c index 14d6b9492750..94268c75d04f 100644 --- a/drivers/block/ataflop.c +++ b/drivers/block/ataflop.c | |||
@@ -1466,7 +1466,7 @@ repeat: | |||
1466 | } | 1466 | } |
1467 | 1467 | ||
1468 | 1468 | ||
1469 | void do_fd_request(request_queue_t * q) | 1469 | void do_fd_request(struct request_queue * q) |
1470 | { | 1470 | { |
1471 | unsigned long flags; | 1471 | unsigned long flags; |
1472 | 1472 | ||
diff --git a/drivers/block/cciss.c b/drivers/block/cciss.c index a2d6612b80d2..1be82d544dc3 100644 --- a/drivers/block/cciss.c +++ b/drivers/block/cciss.c | |||
@@ -139,7 +139,7 @@ static struct board_type products[] = { | |||
139 | 139 | ||
140 | static ctlr_info_t *hba[MAX_CTLR]; | 140 | static ctlr_info_t *hba[MAX_CTLR]; |
141 | 141 | ||
142 | static void do_cciss_request(request_queue_t *q); | 142 | static void do_cciss_request(struct request_queue *q); |
143 | static irqreturn_t do_cciss_intr(int irq, void *dev_id); | 143 | static irqreturn_t do_cciss_intr(int irq, void *dev_id); |
144 | static int cciss_open(struct inode *inode, struct file *filep); | 144 | static int cciss_open(struct inode *inode, struct file *filep); |
145 | static int cciss_release(struct inode *inode, struct file *filep); | 145 | static int cciss_release(struct inode *inode, struct file *filep); |
@@ -1584,7 +1584,7 @@ static int deregister_disk(struct gendisk *disk, drive_info_struct *drv, | |||
1584 | */ | 1584 | */ |
1585 | if (h->gendisk[0] != disk) { | 1585 | if (h->gendisk[0] != disk) { |
1586 | if (disk) { | 1586 | if (disk) { |
1587 | request_queue_t *q = disk->queue; | 1587 | struct request_queue *q = disk->queue; |
1588 | if (disk->flags & GENHD_FL_UP) | 1588 | if (disk->flags & GENHD_FL_UP) |
1589 | del_gendisk(disk); | 1589 | del_gendisk(disk); |
1590 | if (q) { | 1590 | if (q) { |
@@ -2511,7 +2511,7 @@ after_error_processing: | |||
2511 | /* | 2511 | /* |
2512 | * Get a request and submit it to the controller. | 2512 | * Get a request and submit it to the controller. |
2513 | */ | 2513 | */ |
2514 | static void do_cciss_request(request_queue_t *q) | 2514 | static void do_cciss_request(struct request_queue *q) |
2515 | { | 2515 | { |
2516 | ctlr_info_t *h = q->queuedata; | 2516 | ctlr_info_t *h = q->queuedata; |
2517 | CommandList_struct *c; | 2517 | CommandList_struct *c; |
@@ -3380,7 +3380,7 @@ static int __devinit cciss_init_one(struct pci_dev *pdev, | |||
3380 | do { | 3380 | do { |
3381 | drive_info_struct *drv = &(hba[i]->drv[j]); | 3381 | drive_info_struct *drv = &(hba[i]->drv[j]); |
3382 | struct gendisk *disk = hba[i]->gendisk[j]; | 3382 | struct gendisk *disk = hba[i]->gendisk[j]; |
3383 | request_queue_t *q; | 3383 | struct request_queue *q; |
3384 | 3384 | ||
3385 | /* Check if the disk was allocated already */ | 3385 | /* Check if the disk was allocated already */ |
3386 | if (!disk){ | 3386 | if (!disk){ |
@@ -3523,7 +3523,7 @@ static void __devexit cciss_remove_one(struct pci_dev *pdev) | |||
3523 | for (j = 0; j < CISS_MAX_LUN; j++) { | 3523 | for (j = 0; j < CISS_MAX_LUN; j++) { |
3524 | struct gendisk *disk = hba[i]->gendisk[j]; | 3524 | struct gendisk *disk = hba[i]->gendisk[j]; |
3525 | if (disk) { | 3525 | if (disk) { |
3526 | request_queue_t *q = disk->queue; | 3526 | struct request_queue *q = disk->queue; |
3527 | 3527 | ||
3528 | if (disk->flags & GENHD_FL_UP) | 3528 | if (disk->flags & GENHD_FL_UP) |
3529 | del_gendisk(disk); | 3529 | del_gendisk(disk); |
diff --git a/drivers/block/cpqarray.c b/drivers/block/cpqarray.c index b94cd1c32131..be4e3477d83b 100644 --- a/drivers/block/cpqarray.c +++ b/drivers/block/cpqarray.c | |||
@@ -161,7 +161,7 @@ static int ida_ioctl(struct inode *inode, struct file *filep, unsigned int cmd, | |||
161 | static int ida_getgeo(struct block_device *bdev, struct hd_geometry *geo); | 161 | static int ida_getgeo(struct block_device *bdev, struct hd_geometry *geo); |
162 | static int ida_ctlr_ioctl(ctlr_info_t *h, int dsk, ida_ioctl_t *io); | 162 | static int ida_ctlr_ioctl(ctlr_info_t *h, int dsk, ida_ioctl_t *io); |
163 | 163 | ||
164 | static void do_ida_request(request_queue_t *q); | 164 | static void do_ida_request(struct request_queue *q); |
165 | static void start_io(ctlr_info_t *h); | 165 | static void start_io(ctlr_info_t *h); |
166 | 166 | ||
167 | static inline void addQ(cmdlist_t **Qptr, cmdlist_t *c); | 167 | static inline void addQ(cmdlist_t **Qptr, cmdlist_t *c); |
@@ -391,7 +391,7 @@ static void __devexit cpqarray_remove_one_eisa (int i) | |||
391 | /* pdev is NULL for eisa */ | 391 | /* pdev is NULL for eisa */ |
392 | static int __init cpqarray_register_ctlr( int i, struct pci_dev *pdev) | 392 | static int __init cpqarray_register_ctlr( int i, struct pci_dev *pdev) |
393 | { | 393 | { |
394 | request_queue_t *q; | 394 | struct request_queue *q; |
395 | int j; | 395 | int j; |
396 | 396 | ||
397 | /* | 397 | /* |
@@ -886,7 +886,7 @@ static inline cmdlist_t *removeQ(cmdlist_t **Qptr, cmdlist_t *c) | |||
886 | * are in here (either via the dummy do_ida_request functions or by being | 886 | * are in here (either via the dummy do_ida_request functions or by being |
887 | * called from the interrupt handler | 887 | * called from the interrupt handler |
888 | */ | 888 | */ |
889 | static void do_ida_request(request_queue_t *q) | 889 | static void do_ida_request(struct request_queue *q) |
890 | { | 890 | { |
891 | ctlr_info_t *h = q->queuedata; | 891 | ctlr_info_t *h = q->queuedata; |
892 | cmdlist_t *c; | 892 | cmdlist_t *c; |
diff --git a/drivers/block/floppy.c b/drivers/block/floppy.c index fe088045dd08..085b7794fb3e 100644 --- a/drivers/block/floppy.c +++ b/drivers/block/floppy.c | |||
@@ -251,7 +251,7 @@ static int irqdma_allocated; | |||
251 | 251 | ||
252 | static struct request *current_req; | 252 | static struct request *current_req; |
253 | static struct request_queue *floppy_queue; | 253 | static struct request_queue *floppy_queue; |
254 | static void do_fd_request(request_queue_t * q); | 254 | static void do_fd_request(struct request_queue * q); |
255 | 255 | ||
256 | #ifndef fd_get_dma_residue | 256 | #ifndef fd_get_dma_residue |
257 | #define fd_get_dma_residue() get_dma_residue(FLOPPY_DMA) | 257 | #define fd_get_dma_residue() get_dma_residue(FLOPPY_DMA) |
@@ -2981,7 +2981,7 @@ static void process_fd_request(void) | |||
2981 | schedule_bh(redo_fd_request); | 2981 | schedule_bh(redo_fd_request); |
2982 | } | 2982 | } |
2983 | 2983 | ||
2984 | static void do_fd_request(request_queue_t * q) | 2984 | static void do_fd_request(struct request_queue * q) |
2985 | { | 2985 | { |
2986 | if (max_buffer_sectors == 0) { | 2986 | if (max_buffer_sectors == 0) { |
2987 | printk("VFS: do_fd_request called on non-open device\n"); | 2987 | printk("VFS: do_fd_request called on non-open device\n"); |
diff --git a/drivers/block/lguest_blk.c b/drivers/block/lguest_blk.c index 1634c2dd25ec..93e3c4001bf5 100644 --- a/drivers/block/lguest_blk.c +++ b/drivers/block/lguest_blk.c | |||
@@ -1,6 +1,12 @@ | |||
1 | /* A simple block driver for lguest. | 1 | /*D:400 |
2 | * The Guest block driver | ||
2 | * | 3 | * |
3 | * Copyright 2006 Rusty Russell <rusty@rustcorp.com.au> IBM Corporation | 4 | * This is a simple block driver, which appears as /dev/lgba, lgbb, lgbc etc. |
5 | * The mechanism is simple: we place the information about the request in the | ||
6 | * device page, then use SEND_DMA (containing the data for a write, or an empty | ||
7 | * "ping" DMA for a read). | ||
8 | :*/ | ||
9 | /* Copyright 2006 Rusty Russell <rusty@rustcorp.com.au> IBM Corporation | ||
4 | * | 10 | * |
5 | * This program is free software; you can redistribute it and/or modify | 11 | * This program is free software; you can redistribute it and/or modify |
6 | * it under the terms of the GNU General Public License as published by | 12 | * it under the terms of the GNU General Public License as published by |
@@ -25,27 +31,50 @@ | |||
25 | 31 | ||
26 | static char next_block_index = 'a'; | 32 | static char next_block_index = 'a'; |
27 | 33 | ||
34 | /*D:420 Here is the structure which holds all the information we need about | ||
35 | * each Guest block device. | ||
36 | * | ||
37 | * I'm sure at this stage, you're wondering "hey, where was the adventure I was | ||
38 | * promised?" and thinking "Rusty sucks, I shall say nasty things about him on | ||
39 | * my blog". I think Real adventures have boring bits, too, and you're in the | ||
40 | * middle of one. But it gets better. Just not quite yet. */ | ||
28 | struct blockdev | 41 | struct blockdev |
29 | { | 42 | { |
43 | /* The block queue infrastructure wants a spinlock: it is held while it | ||
44 | * calls our block request function. We grab it in our interrupt | ||
45 | * handler so the responses don't mess with new requests. */ | ||
30 | spinlock_t lock; | 46 | spinlock_t lock; |
31 | 47 | ||
32 | /* The disk structure for the kernel. */ | 48 | /* The disk structure registered with kernel. */ |
33 | struct gendisk *disk; | 49 | struct gendisk *disk; |
34 | 50 | ||
35 | /* The major number for this disk. */ | 51 | /* The major device number for this disk, and the interrupt. We only |
52 | * really keep them here for completeness; we'd need them if we | ||
53 | * supported device unplugging. */ | ||
36 | int major; | 54 | int major; |
37 | int irq; | 55 | int irq; |
38 | 56 | ||
57 | /* The physical address of this device's memory page */ | ||
39 | unsigned long phys_addr; | 58 | unsigned long phys_addr; |
40 | /* The mapped block page. */ | 59 | /* The mapped memory page for convenient acces. */ |
41 | struct lguest_block_page *lb_page; | 60 | struct lguest_block_page *lb_page; |
42 | 61 | ||
43 | /* We only have a single request outstanding at a time. */ | 62 | /* We only have a single request outstanding at a time: this is it. */ |
44 | struct lguest_dma dma; | 63 | struct lguest_dma dma; |
45 | struct request *req; | 64 | struct request *req; |
46 | }; | 65 | }; |
47 | 66 | ||
48 | /* Jens gave me this nice helper to end all chunks of a request. */ | 67 | /*D:495 We originally used end_request() throughout the driver, but it turns |
68 | * out that end_request() is deprecated, and doesn't actually end the request | ||
69 | * (which seems like a good reason to deprecate it!). It simply ends the first | ||
70 | * bio. So if we had 3 bios in a "struct request" we would do all 3, | ||
71 | * end_request(), do 2, end_request(), do 1 and end_request(): twice as much | ||
72 | * work as we needed to do. | ||
73 | * | ||
74 | * This reinforced to me that I do not understand the block layer. | ||
75 | * | ||
76 | * Nonetheless, Jens Axboe gave me this nice helper to end all chunks of a | ||
77 | * request. This improved disk speed by 130%. */ | ||
49 | static void end_entire_request(struct request *req, int uptodate) | 78 | static void end_entire_request(struct request *req, int uptodate) |
50 | { | 79 | { |
51 | if (end_that_request_first(req, uptodate, req->hard_nr_sectors)) | 80 | if (end_that_request_first(req, uptodate, req->hard_nr_sectors)) |
@@ -55,30 +84,62 @@ static void end_entire_request(struct request *req, int uptodate) | |||
55 | end_that_request_last(req, uptodate); | 84 | end_that_request_last(req, uptodate); |
56 | } | 85 | } |
57 | 86 | ||
87 | /* I'm told there are only two stories in the world worth telling: love and | ||
88 | * hate. So there used to be a love scene here like this: | ||
89 | * | ||
90 | * Launcher: We could make beautiful I/O together, you and I. | ||
91 | * Guest: My, that's a big disk! | ||
92 | * | ||
93 | * Unfortunately, it was just too raunchy for our otherwise-gentle tale. */ | ||
94 | |||
95 | /*D:490 This is the interrupt handler, called when a block read or write has | ||
96 | * been completed for us. */ | ||
58 | static irqreturn_t lgb_irq(int irq, void *_bd) | 97 | static irqreturn_t lgb_irq(int irq, void *_bd) |
59 | { | 98 | { |
99 | /* We handed our "struct blockdev" as the argument to request_irq(), so | ||
100 | * it is passed through to us here. This tells us which device we're | ||
101 | * dealing with in case we have more than one. */ | ||
60 | struct blockdev *bd = _bd; | 102 | struct blockdev *bd = _bd; |
61 | unsigned long flags; | 103 | unsigned long flags; |
62 | 104 | ||
105 | /* We weren't doing anything? Strange, but could happen if we shared | ||
106 | * interrupts (we don't!). */ | ||
63 | if (!bd->req) { | 107 | if (!bd->req) { |
64 | pr_debug("No work!\n"); | 108 | pr_debug("No work!\n"); |
65 | return IRQ_NONE; | 109 | return IRQ_NONE; |
66 | } | 110 | } |
67 | 111 | ||
112 | /* Not done yet? That's equally strange. */ | ||
68 | if (!bd->lb_page->result) { | 113 | if (!bd->lb_page->result) { |
69 | pr_debug("No result!\n"); | 114 | pr_debug("No result!\n"); |
70 | return IRQ_NONE; | 115 | return IRQ_NONE; |
71 | } | 116 | } |
72 | 117 | ||
118 | /* We have to grab the lock before ending the request. */ | ||
73 | spin_lock_irqsave(&bd->lock, flags); | 119 | spin_lock_irqsave(&bd->lock, flags); |
120 | /* "result" is 1 for success, 2 for failure: end_entire_request() wants | ||
121 | * to know whether this succeeded or not. */ | ||
74 | end_entire_request(bd->req, bd->lb_page->result == 1); | 122 | end_entire_request(bd->req, bd->lb_page->result == 1); |
123 | /* Clear out request, it's done. */ | ||
75 | bd->req = NULL; | 124 | bd->req = NULL; |
125 | /* Reset incoming DMA for next time. */ | ||
76 | bd->dma.used_len = 0; | 126 | bd->dma.used_len = 0; |
127 | /* Ready for more reads or writes */ | ||
77 | blk_start_queue(bd->disk->queue); | 128 | blk_start_queue(bd->disk->queue); |
78 | spin_unlock_irqrestore(&bd->lock, flags); | 129 | spin_unlock_irqrestore(&bd->lock, flags); |
130 | |||
131 | /* The interrupt was for us, we dealt with it. */ | ||
79 | return IRQ_HANDLED; | 132 | return IRQ_HANDLED; |
80 | } | 133 | } |
81 | 134 | ||
135 | /*D:480 The block layer's "struct request" contains a number of "struct bio"s, | ||
136 | * each of which contains "struct bio_vec"s, each of which contains a page, an | ||
137 | * offset and a length. | ||
138 | * | ||
139 | * Fortunately there are iterators to help us walk through the "struct | ||
140 | * request". Even more fortunately, there were plenty of places to steal the | ||
141 | * code from. We pack the "struct request" into our "struct lguest_dma" and | ||
142 | * return the total length. */ | ||
82 | static unsigned int req_to_dma(struct request *req, struct lguest_dma *dma) | 143 | static unsigned int req_to_dma(struct request *req, struct lguest_dma *dma) |
83 | { | 144 | { |
84 | unsigned int i = 0, idx, len = 0; | 145 | unsigned int i = 0, idx, len = 0; |
@@ -87,8 +148,13 @@ static unsigned int req_to_dma(struct request *req, struct lguest_dma *dma) | |||
87 | rq_for_each_bio(bio, req) { | 148 | rq_for_each_bio(bio, req) { |
88 | struct bio_vec *bvec; | 149 | struct bio_vec *bvec; |
89 | bio_for_each_segment(bvec, bio, idx) { | 150 | bio_for_each_segment(bvec, bio, idx) { |
151 | /* We told the block layer not to give us too many. */ | ||
90 | BUG_ON(i == LGUEST_MAX_DMA_SECTIONS); | 152 | BUG_ON(i == LGUEST_MAX_DMA_SECTIONS); |
153 | /* If we had a zero-length segment, it would look like | ||
154 | * the end of the data referred to by the "struct | ||
155 | * lguest_dma", so make sure that doesn't happen. */ | ||
91 | BUG_ON(!bvec->bv_len); | 156 | BUG_ON(!bvec->bv_len); |
157 | /* Convert page & offset to a physical address */ | ||
92 | dma->addr[i] = page_to_phys(bvec->bv_page) | 158 | dma->addr[i] = page_to_phys(bvec->bv_page) |
93 | + bvec->bv_offset; | 159 | + bvec->bv_offset; |
94 | dma->len[i] = bvec->bv_len; | 160 | dma->len[i] = bvec->bv_len; |
@@ -96,26 +162,39 @@ static unsigned int req_to_dma(struct request *req, struct lguest_dma *dma) | |||
96 | i++; | 162 | i++; |
97 | } | 163 | } |
98 | } | 164 | } |
165 | /* If the array isn't full, we mark the end with a 0 length */ | ||
99 | if (i < LGUEST_MAX_DMA_SECTIONS) | 166 | if (i < LGUEST_MAX_DMA_SECTIONS) |
100 | dma->len[i] = 0; | 167 | dma->len[i] = 0; |
101 | return len; | 168 | return len; |
102 | } | 169 | } |
103 | 170 | ||
171 | /* This creates an empty DMA, useful for prodding the Host without sending data | ||
172 | * (ie. when we want to do a read) */ | ||
104 | static void empty_dma(struct lguest_dma *dma) | 173 | static void empty_dma(struct lguest_dma *dma) |
105 | { | 174 | { |
106 | dma->len[0] = 0; | 175 | dma->len[0] = 0; |
107 | } | 176 | } |
108 | 177 | ||
178 | /*D:470 Setting up a request is fairly easy: */ | ||
109 | static void setup_req(struct blockdev *bd, | 179 | static void setup_req(struct blockdev *bd, |
110 | int type, struct request *req, struct lguest_dma *dma) | 180 | int type, struct request *req, struct lguest_dma *dma) |
111 | { | 181 | { |
182 | /* The type is 1 (write) or 0 (read). */ | ||
112 | bd->lb_page->type = type; | 183 | bd->lb_page->type = type; |
184 | /* The sector on disk where the read or write starts. */ | ||
113 | bd->lb_page->sector = req->sector; | 185 | bd->lb_page->sector = req->sector; |
186 | /* The result is initialized to 0 (unfinished). */ | ||
114 | bd->lb_page->result = 0; | 187 | bd->lb_page->result = 0; |
188 | /* The current request (so we can end it in the interrupt handler). */ | ||
115 | bd->req = req; | 189 | bd->req = req; |
190 | /* The number of bytes: returned as a side-effect of req_to_dma(), | ||
191 | * which packs the block layer's "struct request" into our "struct | ||
192 | * lguest_dma" */ | ||
116 | bd->lb_page->bytes = req_to_dma(req, dma); | 193 | bd->lb_page->bytes = req_to_dma(req, dma); |
117 | } | 194 | } |
118 | 195 | ||
196 | /*D:450 Write is pretty straightforward: we pack the request into a "struct | ||
197 | * lguest_dma", then use SEND_DMA to send the request. */ | ||
119 | static void do_write(struct blockdev *bd, struct request *req) | 198 | static void do_write(struct blockdev *bd, struct request *req) |
120 | { | 199 | { |
121 | struct lguest_dma send; | 200 | struct lguest_dma send; |
@@ -126,6 +205,9 @@ static void do_write(struct blockdev *bd, struct request *req) | |||
126 | lguest_send_dma(bd->phys_addr, &send); | 205 | lguest_send_dma(bd->phys_addr, &send); |
127 | } | 206 | } |
128 | 207 | ||
208 | /* Read is similar to write, except we pack the request into our receive | ||
209 | * "struct lguest_dma" and send through an empty DMA just to tell the Host that | ||
210 | * there's a request pending. */ | ||
129 | static void do_read(struct blockdev *bd, struct request *req) | 211 | static void do_read(struct blockdev *bd, struct request *req) |
130 | { | 212 | { |
131 | struct lguest_dma ping; | 213 | struct lguest_dma ping; |
@@ -137,21 +219,30 @@ static void do_read(struct blockdev *bd, struct request *req) | |||
137 | lguest_send_dma(bd->phys_addr, &ping); | 219 | lguest_send_dma(bd->phys_addr, &ping); |
138 | } | 220 | } |
139 | 221 | ||
140 | static void do_lgb_request(request_queue_t *q) | 222 | /*D:440 This where requests come in: we get handed the request queue and are |
223 | * expected to pull a "struct request" off it until we've finished them or | ||
224 | * we're waiting for a reply: */ | ||
225 | static void do_lgb_request(struct request_queue *q) | ||
141 | { | 226 | { |
142 | struct blockdev *bd; | 227 | struct blockdev *bd; |
143 | struct request *req; | 228 | struct request *req; |
144 | 229 | ||
145 | again: | 230 | again: |
231 | /* This sometimes returns NULL even on the very first time around. I | ||
232 | * wonder if it's something to do with letting elves handle the request | ||
233 | * queue... */ | ||
146 | req = elv_next_request(q); | 234 | req = elv_next_request(q); |
147 | if (!req) | 235 | if (!req) |
148 | return; | 236 | return; |
149 | 237 | ||
238 | /* We attached the struct blockdev to the disk: get it back */ | ||
150 | bd = req->rq_disk->private_data; | 239 | bd = req->rq_disk->private_data; |
151 | /* Sometimes we get repeated requests after blk_stop_queue. */ | 240 | /* Sometimes we get repeated requests after blk_stop_queue(), but we |
241 | * can only handle one at a time. */ | ||
152 | if (bd->req) | 242 | if (bd->req) |
153 | return; | 243 | return; |
154 | 244 | ||
245 | /* We only do reads and writes: no tricky business! */ | ||
155 | if (!blk_fs_request(req)) { | 246 | if (!blk_fs_request(req)) { |
156 | pr_debug("Got non-command 0x%08x\n", req->cmd_type); | 247 | pr_debug("Got non-command 0x%08x\n", req->cmd_type); |
157 | req->errors++; | 248 | req->errors++; |
@@ -164,20 +255,31 @@ again: | |||
164 | else | 255 | else |
165 | do_read(bd, req); | 256 | do_read(bd, req); |
166 | 257 | ||
167 | /* Wait for interrupt to tell us it's done. */ | 258 | /* We've put out the request, so stop any more coming in until we get |
259 | * an interrupt, which takes us to lgb_irq() to re-enable the queue. */ | ||
168 | blk_stop_queue(q); | 260 | blk_stop_queue(q); |
169 | } | 261 | } |
170 | 262 | ||
263 | /*D:430 This is the "struct block_device_operations" we attach to the disk at | ||
264 | * the end of lguestblk_probe(). It doesn't seem to want much. */ | ||
171 | static struct block_device_operations lguestblk_fops = { | 265 | static struct block_device_operations lguestblk_fops = { |
172 | .owner = THIS_MODULE, | 266 | .owner = THIS_MODULE, |
173 | }; | 267 | }; |
174 | 268 | ||
269 | /*D:425 Setting up a disk device seems to involve a lot of code. I'm not sure | ||
270 | * quite why. I do know that the IDE code sent two or three of the maintainers | ||
271 | * insane, perhaps this is the fringe of the same disease? | ||
272 | * | ||
273 | * As in the console code, the probe function gets handed the generic | ||
274 | * lguest_device from lguest_bus.c: */ | ||
175 | static int lguestblk_probe(struct lguest_device *lgdev) | 275 | static int lguestblk_probe(struct lguest_device *lgdev) |
176 | { | 276 | { |
177 | struct blockdev *bd; | 277 | struct blockdev *bd; |
178 | int err; | 278 | int err; |
179 | int irqflags = IRQF_SHARED; | 279 | int irqflags = IRQF_SHARED; |
180 | 280 | ||
281 | /* First we allocate our own "struct blockdev" and initialize the easy | ||
282 | * fields. */ | ||
181 | bd = kmalloc(sizeof(*bd), GFP_KERNEL); | 283 | bd = kmalloc(sizeof(*bd), GFP_KERNEL); |
182 | if (!bd) | 284 | if (!bd) |
183 | return -ENOMEM; | 285 | return -ENOMEM; |
@@ -187,59 +289,100 @@ static int lguestblk_probe(struct lguest_device *lgdev) | |||
187 | bd->req = NULL; | 289 | bd->req = NULL; |
188 | bd->dma.used_len = 0; | 290 | bd->dma.used_len = 0; |
189 | bd->dma.len[0] = 0; | 291 | bd->dma.len[0] = 0; |
292 | /* The descriptor in the lguest_devices array provided by the Host | ||
293 | * gives the Guest the physical page number of the device's page. */ | ||
190 | bd->phys_addr = (lguest_devices[lgdev->index].pfn << PAGE_SHIFT); | 294 | bd->phys_addr = (lguest_devices[lgdev->index].pfn << PAGE_SHIFT); |
191 | 295 | ||
296 | /* We use lguest_map() to get a pointer to the device page */ | ||
192 | bd->lb_page = lguest_map(bd->phys_addr, 1); | 297 | bd->lb_page = lguest_map(bd->phys_addr, 1); |
193 | if (!bd->lb_page) { | 298 | if (!bd->lb_page) { |
194 | err = -ENOMEM; | 299 | err = -ENOMEM; |
195 | goto out_free_bd; | 300 | goto out_free_bd; |
196 | } | 301 | } |
197 | 302 | ||
303 | /* We need a major device number: 0 means "assign one dynamically". */ | ||
198 | bd->major = register_blkdev(0, "lguestblk"); | 304 | bd->major = register_blkdev(0, "lguestblk"); |
199 | if (bd->major < 0) { | 305 | if (bd->major < 0) { |
200 | err = bd->major; | 306 | err = bd->major; |
201 | goto out_unmap; | 307 | goto out_unmap; |
202 | } | 308 | } |
203 | 309 | ||
310 | /* This allocates a "struct gendisk" where we pack all the information | ||
311 | * about the disk which the rest of Linux sees. We ask for one minor | ||
312 | * number; I do wonder if we should be asking for more. */ | ||
204 | bd->disk = alloc_disk(1); | 313 | bd->disk = alloc_disk(1); |
205 | if (!bd->disk) { | 314 | if (!bd->disk) { |
206 | err = -ENOMEM; | 315 | err = -ENOMEM; |
207 | goto out_unregister_blkdev; | 316 | goto out_unregister_blkdev; |
208 | } | 317 | } |
209 | 318 | ||
319 | /* Every disk needs a queue for requests to come in: we set up the | ||
320 | * queue with a callback function (the core of our driver) and the lock | ||
321 | * to use. */ | ||
210 | bd->disk->queue = blk_init_queue(do_lgb_request, &bd->lock); | 322 | bd->disk->queue = blk_init_queue(do_lgb_request, &bd->lock); |
211 | if (!bd->disk->queue) { | 323 | if (!bd->disk->queue) { |
212 | err = -ENOMEM; | 324 | err = -ENOMEM; |
213 | goto out_put_disk; | 325 | goto out_put_disk; |
214 | } | 326 | } |
215 | 327 | ||
216 | /* We can only handle a certain number of sg entries */ | 328 | /* We can only handle a certain number of pointers in our SEND_DMA |
329 | * call, so we set that with blk_queue_max_hw_segments(). This is not | ||
330 | * to be confused with blk_queue_max_phys_segments() of course! I | ||
331 | * know, who could possibly confuse the two? | ||
332 | * | ||
333 | * Well, it's simple to tell them apart: this one seems to work and the | ||
334 | * other one didn't. */ | ||
217 | blk_queue_max_hw_segments(bd->disk->queue, LGUEST_MAX_DMA_SECTIONS); | 335 | blk_queue_max_hw_segments(bd->disk->queue, LGUEST_MAX_DMA_SECTIONS); |
218 | /* Buffers must not cross page boundaries */ | 336 | |
337 | /* Due to technical limitations of our Host (and simple coding) we | ||
338 | * can't have a single buffer which crosses a page boundary. Tell it | ||
339 | * here. This means that our maximum request size is 16 | ||
340 | * (LGUEST_MAX_DMA_SECTIONS) pages. */ | ||
219 | blk_queue_segment_boundary(bd->disk->queue, PAGE_SIZE-1); | 341 | blk_queue_segment_boundary(bd->disk->queue, PAGE_SIZE-1); |
220 | 342 | ||
343 | /* We name our disk: this becomes the device name when udev does its | ||
344 | * magic thing and creates the device node, such as /dev/lgba. | ||
345 | * next_block_index is a global which starts at 'a'. Unfortunately | ||
346 | * this simple increment logic means that the 27th disk will be called | ||
347 | * "/dev/lgb{". In that case, I recommend having at least 29 disks, so | ||
348 | * your /dev directory will be balanced. */ | ||
221 | sprintf(bd->disk->disk_name, "lgb%c", next_block_index++); | 349 | sprintf(bd->disk->disk_name, "lgb%c", next_block_index++); |
350 | |||
351 | /* We look to the device descriptor again to see if this device's | ||
352 | * interrupts are expected to be random. If they are, we tell the irq | ||
353 | * subsystem. At the moment this bit is always set. */ | ||
222 | if (lguest_devices[lgdev->index].features & LGUEST_DEVICE_F_RANDOMNESS) | 354 | if (lguest_devices[lgdev->index].features & LGUEST_DEVICE_F_RANDOMNESS) |
223 | irqflags |= IRQF_SAMPLE_RANDOM; | 355 | irqflags |= IRQF_SAMPLE_RANDOM; |
356 | |||
357 | /* Now we have the name and irqflags, we can request the interrupt; we | ||
358 | * give it the "struct blockdev" we have set up to pass to lgb_irq() | ||
359 | * when there is an interrupt. */ | ||
224 | err = request_irq(bd->irq, lgb_irq, irqflags, bd->disk->disk_name, bd); | 360 | err = request_irq(bd->irq, lgb_irq, irqflags, bd->disk->disk_name, bd); |
225 | if (err) | 361 | if (err) |
226 | goto out_cleanup_queue; | 362 | goto out_cleanup_queue; |
227 | 363 | ||
364 | /* We bind our one-entry DMA pool to the key for this block device so | ||
365 | * the Host can reply to our requests. The key is equal to the | ||
366 | * physical address of the device's page, which is conveniently | ||
367 | * unique. */ | ||
228 | err = lguest_bind_dma(bd->phys_addr, &bd->dma, 1, bd->irq); | 368 | err = lguest_bind_dma(bd->phys_addr, &bd->dma, 1, bd->irq); |
229 | if (err) | 369 | if (err) |
230 | goto out_free_irq; | 370 | goto out_free_irq; |
231 | 371 | ||
372 | /* We finish our disk initialization and add the disk to the system. */ | ||
232 | bd->disk->major = bd->major; | 373 | bd->disk->major = bd->major; |
233 | bd->disk->first_minor = 0; | 374 | bd->disk->first_minor = 0; |
234 | bd->disk->private_data = bd; | 375 | bd->disk->private_data = bd; |
235 | bd->disk->fops = &lguestblk_fops; | 376 | bd->disk->fops = &lguestblk_fops; |
236 | /* This is initialized to the disk size by the other end. */ | 377 | /* This is initialized to the disk size by the Launcher. */ |
237 | set_capacity(bd->disk, bd->lb_page->num_sectors); | 378 | set_capacity(bd->disk, bd->lb_page->num_sectors); |
238 | add_disk(bd->disk); | 379 | add_disk(bd->disk); |
239 | 380 | ||
240 | printk(KERN_INFO "%s: device %i at major %d\n", | 381 | printk(KERN_INFO "%s: device %i at major %d\n", |
241 | bd->disk->disk_name, lgdev->index, bd->major); | 382 | bd->disk->disk_name, lgdev->index, bd->major); |
242 | 383 | ||
384 | /* We don't need to keep the "struct blockdev" around, but if we ever | ||
385 | * implemented device removal, we'd need this. */ | ||
243 | lgdev->private = bd; | 386 | lgdev->private = bd; |
244 | return 0; | 387 | return 0; |
245 | 388 | ||
@@ -258,6 +401,8 @@ out_free_bd: | |||
258 | return err; | 401 | return err; |
259 | } | 402 | } |
260 | 403 | ||
404 | /*D:410 The boilerplate code for registering the lguest block driver is just | ||
405 | * like the console: */ | ||
261 | static struct lguest_driver lguestblk_drv = { | 406 | static struct lguest_driver lguestblk_drv = { |
262 | .name = "lguestblk", | 407 | .name = "lguestblk", |
263 | .owner = THIS_MODULE, | 408 | .owner = THIS_MODULE, |
diff --git a/drivers/block/loop.c b/drivers/block/loop.c index e425daa1eac3..9f015fce4135 100644 --- a/drivers/block/loop.c +++ b/drivers/block/loop.c | |||
@@ -529,7 +529,7 @@ static struct bio *loop_get_bio(struct loop_device *lo) | |||
529 | return bio; | 529 | return bio; |
530 | } | 530 | } |
531 | 531 | ||
532 | static int loop_make_request(request_queue_t *q, struct bio *old_bio) | 532 | static int loop_make_request(struct request_queue *q, struct bio *old_bio) |
533 | { | 533 | { |
534 | struct loop_device *lo = q->queuedata; | 534 | struct loop_device *lo = q->queuedata; |
535 | int rw = bio_rw(old_bio); | 535 | int rw = bio_rw(old_bio); |
@@ -558,7 +558,7 @@ out: | |||
558 | /* | 558 | /* |
559 | * kick off io on the underlying address space | 559 | * kick off io on the underlying address space |
560 | */ | 560 | */ |
561 | static void loop_unplug(request_queue_t *q) | 561 | static void loop_unplug(struct request_queue *q) |
562 | { | 562 | { |
563 | struct loop_device *lo = q->queuedata; | 563 | struct loop_device *lo = q->queuedata; |
564 | 564 | ||
diff --git a/drivers/block/nbd.c b/drivers/block/nbd.c index c12951024090..be92c658f06e 100644 --- a/drivers/block/nbd.c +++ b/drivers/block/nbd.c | |||
@@ -100,7 +100,7 @@ static const char *nbdcmd_to_ascii(int cmd) | |||
100 | static void nbd_end_request(struct request *req) | 100 | static void nbd_end_request(struct request *req) |
101 | { | 101 | { |
102 | int uptodate = (req->errors == 0) ? 1 : 0; | 102 | int uptodate = (req->errors == 0) ? 1 : 0; |
103 | request_queue_t *q = req->q; | 103 | struct request_queue *q = req->q; |
104 | unsigned long flags; | 104 | unsigned long flags; |
105 | 105 | ||
106 | dprintk(DBG_BLKDEV, "%s: request %p: %s\n", req->rq_disk->disk_name, | 106 | dprintk(DBG_BLKDEV, "%s: request %p: %s\n", req->rq_disk->disk_name, |
@@ -410,7 +410,7 @@ static void nbd_clear_que(struct nbd_device *lo) | |||
410 | * { printk( "Warning: Ignoring result!\n"); nbd_end_request( req ); } | 410 | * { printk( "Warning: Ignoring result!\n"); nbd_end_request( req ); } |
411 | */ | 411 | */ |
412 | 412 | ||
413 | static void do_nbd_request(request_queue_t * q) | 413 | static void do_nbd_request(struct request_queue * q) |
414 | { | 414 | { |
415 | struct request *req; | 415 | struct request *req; |
416 | 416 | ||
diff --git a/drivers/block/paride/pcd.c b/drivers/block/paride/pcd.c index 1eeb8f2cde71..b8a994a2b013 100644 --- a/drivers/block/paride/pcd.c +++ b/drivers/block/paride/pcd.c | |||
@@ -183,7 +183,7 @@ static int pcd_packet(struct cdrom_device_info *cdi, | |||
183 | static int pcd_detect(void); | 183 | static int pcd_detect(void); |
184 | static void pcd_probe_capabilities(void); | 184 | static void pcd_probe_capabilities(void); |
185 | static void do_pcd_read_drq(void); | 185 | static void do_pcd_read_drq(void); |
186 | static void do_pcd_request(request_queue_t * q); | 186 | static void do_pcd_request(struct request_queue * q); |
187 | static void do_pcd_read(void); | 187 | static void do_pcd_read(void); |
188 | 188 | ||
189 | struct pcd_unit { | 189 | struct pcd_unit { |
@@ -713,7 +713,7 @@ static int pcd_detect(void) | |||
713 | /* I/O request processing */ | 713 | /* I/O request processing */ |
714 | static struct request_queue *pcd_queue; | 714 | static struct request_queue *pcd_queue; |
715 | 715 | ||
716 | static void do_pcd_request(request_queue_t * q) | 716 | static void do_pcd_request(struct request_queue * q) |
717 | { | 717 | { |
718 | if (pcd_busy) | 718 | if (pcd_busy) |
719 | return; | 719 | return; |
diff --git a/drivers/block/paride/pd.c b/drivers/block/paride/pd.c index 31e01488eb51..df819f8a95a6 100644 --- a/drivers/block/paride/pd.c +++ b/drivers/block/paride/pd.c | |||
@@ -698,7 +698,7 @@ static enum action pd_identify(struct pd_unit *disk) | |||
698 | 698 | ||
699 | /* end of io request engine */ | 699 | /* end of io request engine */ |
700 | 700 | ||
701 | static void do_pd_request(request_queue_t * q) | 701 | static void do_pd_request(struct request_queue * q) |
702 | { | 702 | { |
703 | if (pd_req) | 703 | if (pd_req) |
704 | return; | 704 | return; |
diff --git a/drivers/block/paride/pf.c b/drivers/block/paride/pf.c index 5826508f6731..ceffa6034e20 100644 --- a/drivers/block/paride/pf.c +++ b/drivers/block/paride/pf.c | |||
@@ -202,7 +202,7 @@ module_param_array(drive3, int, NULL, 0); | |||
202 | #define ATAPI_WRITE_10 0x2a | 202 | #define ATAPI_WRITE_10 0x2a |
203 | 203 | ||
204 | static int pf_open(struct inode *inode, struct file *file); | 204 | static int pf_open(struct inode *inode, struct file *file); |
205 | static void do_pf_request(request_queue_t * q); | 205 | static void do_pf_request(struct request_queue * q); |
206 | static int pf_ioctl(struct inode *inode, struct file *file, | 206 | static int pf_ioctl(struct inode *inode, struct file *file, |
207 | unsigned int cmd, unsigned long arg); | 207 | unsigned int cmd, unsigned long arg); |
208 | static int pf_getgeo(struct block_device *bdev, struct hd_geometry *geo); | 208 | static int pf_getgeo(struct block_device *bdev, struct hd_geometry *geo); |
@@ -760,7 +760,7 @@ static void pf_end_request(int uptodate) | |||
760 | } | 760 | } |
761 | } | 761 | } |
762 | 762 | ||
763 | static void do_pf_request(request_queue_t * q) | 763 | static void do_pf_request(struct request_queue * q) |
764 | { | 764 | { |
765 | if (pf_busy) | 765 | if (pf_busy) |
766 | return; | 766 | return; |
diff --git a/drivers/block/pktcdvd.c b/drivers/block/pktcdvd.c index 31be33e4f119..fadbfd880bab 100644 --- a/drivers/block/pktcdvd.c +++ b/drivers/block/pktcdvd.c | |||
@@ -752,7 +752,7 @@ static inline struct bio *pkt_get_list_first(struct bio **list_head, struct bio | |||
752 | */ | 752 | */ |
753 | static int pkt_generic_packet(struct pktcdvd_device *pd, struct packet_command *cgc) | 753 | static int pkt_generic_packet(struct pktcdvd_device *pd, struct packet_command *cgc) |
754 | { | 754 | { |
755 | request_queue_t *q = bdev_get_queue(pd->bdev); | 755 | struct request_queue *q = bdev_get_queue(pd->bdev); |
756 | struct request *rq; | 756 | struct request *rq; |
757 | int ret = 0; | 757 | int ret = 0; |
758 | 758 | ||
@@ -979,7 +979,7 @@ static void pkt_iosched_process_queue(struct pktcdvd_device *pd) | |||
979 | * Special care is needed if the underlying block device has a small | 979 | * Special care is needed if the underlying block device has a small |
980 | * max_phys_segments value. | 980 | * max_phys_segments value. |
981 | */ | 981 | */ |
982 | static int pkt_set_segment_merging(struct pktcdvd_device *pd, request_queue_t *q) | 982 | static int pkt_set_segment_merging(struct pktcdvd_device *pd, struct request_queue *q) |
983 | { | 983 | { |
984 | if ((pd->settings.size << 9) / CD_FRAMESIZE <= q->max_phys_segments) { | 984 | if ((pd->settings.size << 9) / CD_FRAMESIZE <= q->max_phys_segments) { |
985 | /* | 985 | /* |
@@ -2314,7 +2314,7 @@ static int pkt_open_dev(struct pktcdvd_device *pd, int write) | |||
2314 | { | 2314 | { |
2315 | int ret; | 2315 | int ret; |
2316 | long lba; | 2316 | long lba; |
2317 | request_queue_t *q; | 2317 | struct request_queue *q; |
2318 | 2318 | ||
2319 | /* | 2319 | /* |
2320 | * We need to re-open the cdrom device without O_NONBLOCK to be able | 2320 | * We need to re-open the cdrom device without O_NONBLOCK to be able |
@@ -2477,7 +2477,7 @@ static int pkt_end_io_read_cloned(struct bio *bio, unsigned int bytes_done, int | |||
2477 | return 0; | 2477 | return 0; |
2478 | } | 2478 | } |
2479 | 2479 | ||
2480 | static int pkt_make_request(request_queue_t *q, struct bio *bio) | 2480 | static int pkt_make_request(struct request_queue *q, struct bio *bio) |
2481 | { | 2481 | { |
2482 | struct pktcdvd_device *pd; | 2482 | struct pktcdvd_device *pd; |
2483 | char b[BDEVNAME_SIZE]; | 2483 | char b[BDEVNAME_SIZE]; |
@@ -2626,7 +2626,7 @@ end_io: | |||
2626 | 2626 | ||
2627 | 2627 | ||
2628 | 2628 | ||
2629 | static int pkt_merge_bvec(request_queue_t *q, struct bio *bio, struct bio_vec *bvec) | 2629 | static int pkt_merge_bvec(struct request_queue *q, struct bio *bio, struct bio_vec *bvec) |
2630 | { | 2630 | { |
2631 | struct pktcdvd_device *pd = q->queuedata; | 2631 | struct pktcdvd_device *pd = q->queuedata; |
2632 | sector_t zone = ZONE(bio->bi_sector, pd); | 2632 | sector_t zone = ZONE(bio->bi_sector, pd); |
@@ -2647,7 +2647,7 @@ static int pkt_merge_bvec(request_queue_t *q, struct bio *bio, struct bio_vec *b | |||
2647 | 2647 | ||
2648 | static void pkt_init_queue(struct pktcdvd_device *pd) | 2648 | static void pkt_init_queue(struct pktcdvd_device *pd) |
2649 | { | 2649 | { |
2650 | request_queue_t *q = pd->disk->queue; | 2650 | struct request_queue *q = pd->disk->queue; |
2651 | 2651 | ||
2652 | blk_queue_make_request(q, pkt_make_request); | 2652 | blk_queue_make_request(q, pkt_make_request); |
2653 | blk_queue_hardsect_size(q, CD_FRAMESIZE); | 2653 | blk_queue_hardsect_size(q, CD_FRAMESIZE); |
diff --git a/drivers/block/ps2esdi.c b/drivers/block/ps2esdi.c index 688a4fb0dc99..3c796e236253 100644 --- a/drivers/block/ps2esdi.c +++ b/drivers/block/ps2esdi.c | |||
@@ -64,7 +64,7 @@ static void reset_ctrl(void); | |||
64 | 64 | ||
65 | static int ps2esdi_geninit(void); | 65 | static int ps2esdi_geninit(void); |
66 | 66 | ||
67 | static void do_ps2esdi_request(request_queue_t * q); | 67 | static void do_ps2esdi_request(struct request_queue * q); |
68 | 68 | ||
69 | static void ps2esdi_readwrite(int cmd, struct request *req); | 69 | static void ps2esdi_readwrite(int cmd, struct request *req); |
70 | 70 | ||
@@ -473,7 +473,7 @@ static void __init ps2esdi_get_device_cfg(void) | |||
473 | } | 473 | } |
474 | 474 | ||
475 | /* strategy routine that handles most of the IO requests */ | 475 | /* strategy routine that handles most of the IO requests */ |
476 | static void do_ps2esdi_request(request_queue_t * q) | 476 | static void do_ps2esdi_request(struct request_queue * q) |
477 | { | 477 | { |
478 | struct request *req; | 478 | struct request *req; |
479 | /* since, this routine is called with interrupts cleared - they | 479 | /* since, this routine is called with interrupts cleared - they |
diff --git a/drivers/block/ps3disk.c b/drivers/block/ps3disk.c index 170fb33dba97..aa8b890c80d7 100644 --- a/drivers/block/ps3disk.c +++ b/drivers/block/ps3disk.c | |||
@@ -190,7 +190,7 @@ static int ps3disk_submit_flush_request(struct ps3_storage_device *dev, | |||
190 | } | 190 | } |
191 | 191 | ||
192 | static void ps3disk_do_request(struct ps3_storage_device *dev, | 192 | static void ps3disk_do_request(struct ps3_storage_device *dev, |
193 | request_queue_t *q) | 193 | struct request_queue *q) |
194 | { | 194 | { |
195 | struct request *req; | 195 | struct request *req; |
196 | 196 | ||
@@ -211,7 +211,7 @@ static void ps3disk_do_request(struct ps3_storage_device *dev, | |||
211 | } | 211 | } |
212 | } | 212 | } |
213 | 213 | ||
214 | static void ps3disk_request(request_queue_t *q) | 214 | static void ps3disk_request(struct request_queue *q) |
215 | { | 215 | { |
216 | struct ps3_storage_device *dev = q->queuedata; | 216 | struct ps3_storage_device *dev = q->queuedata; |
217 | struct ps3disk_private *priv = dev->sbd.core.driver_data; | 217 | struct ps3disk_private *priv = dev->sbd.core.driver_data; |
@@ -404,7 +404,7 @@ static int ps3disk_identify(struct ps3_storage_device *dev) | |||
404 | return 0; | 404 | return 0; |
405 | } | 405 | } |
406 | 406 | ||
407 | static void ps3disk_prepare_flush(request_queue_t *q, struct request *req) | 407 | static void ps3disk_prepare_flush(struct request_queue *q, struct request *req) |
408 | { | 408 | { |
409 | struct ps3_storage_device *dev = q->queuedata; | 409 | struct ps3_storage_device *dev = q->queuedata; |
410 | 410 | ||
@@ -414,7 +414,7 @@ static void ps3disk_prepare_flush(request_queue_t *q, struct request *req) | |||
414 | req->cmd_type = REQ_TYPE_FLUSH; | 414 | req->cmd_type = REQ_TYPE_FLUSH; |
415 | } | 415 | } |
416 | 416 | ||
417 | static int ps3disk_issue_flush(request_queue_t *q, struct gendisk *gendisk, | 417 | static int ps3disk_issue_flush(struct request_queue *q, struct gendisk *gendisk, |
418 | sector_t *sector) | 418 | sector_t *sector) |
419 | { | 419 | { |
420 | struct ps3_storage_device *dev = q->queuedata; | 420 | struct ps3_storage_device *dev = q->queuedata; |
diff --git a/drivers/block/rd.c b/drivers/block/rd.c index a1512da32410..65150b548f3a 100644 --- a/drivers/block/rd.c +++ b/drivers/block/rd.c | |||
@@ -264,7 +264,7 @@ static int rd_blkdev_pagecache_IO(int rw, struct bio_vec *vec, sector_t sector, | |||
264 | * 19-JAN-1998 Richard Gooch <rgooch@atnf.csiro.au> Added devfs support | 264 | * 19-JAN-1998 Richard Gooch <rgooch@atnf.csiro.au> Added devfs support |
265 | * | 265 | * |
266 | */ | 266 | */ |
267 | static int rd_make_request(request_queue_t *q, struct bio *bio) | 267 | static int rd_make_request(struct request_queue *q, struct bio *bio) |
268 | { | 268 | { |
269 | struct block_device *bdev = bio->bi_bdev; | 269 | struct block_device *bdev = bio->bi_bdev; |
270 | struct address_space * mapping = bdev->bd_inode->i_mapping; | 270 | struct address_space * mapping = bdev->bd_inode->i_mapping; |
diff --git a/drivers/block/sunvdc.c b/drivers/block/sunvdc.c index d50b82381155..4dff49256ac2 100644 --- a/drivers/block/sunvdc.c +++ b/drivers/block/sunvdc.c | |||
@@ -444,7 +444,7 @@ out: | |||
444 | return err; | 444 | return err; |
445 | } | 445 | } |
446 | 446 | ||
447 | static void do_vdc_request(request_queue_t *q) | 447 | static void do_vdc_request(struct request_queue *q) |
448 | { | 448 | { |
449 | while (1) { | 449 | while (1) { |
450 | struct request *req = elv_next_request(q); | 450 | struct request *req = elv_next_request(q); |
diff --git a/drivers/block/swim3.c b/drivers/block/swim3.c index 1a65979f1f0f..b4e462f154ea 100644 --- a/drivers/block/swim3.c +++ b/drivers/block/swim3.c | |||
@@ -225,7 +225,7 @@ static unsigned short write_postamble[] = { | |||
225 | static void swim3_select(struct floppy_state *fs, int sel); | 225 | static void swim3_select(struct floppy_state *fs, int sel); |
226 | static void swim3_action(struct floppy_state *fs, int action); | 226 | static void swim3_action(struct floppy_state *fs, int action); |
227 | static int swim3_readbit(struct floppy_state *fs, int bit); | 227 | static int swim3_readbit(struct floppy_state *fs, int bit); |
228 | static void do_fd_request(request_queue_t * q); | 228 | static void do_fd_request(struct request_queue * q); |
229 | static void start_request(struct floppy_state *fs); | 229 | static void start_request(struct floppy_state *fs); |
230 | static void set_timeout(struct floppy_state *fs, int nticks, | 230 | static void set_timeout(struct floppy_state *fs, int nticks, |
231 | void (*proc)(unsigned long)); | 231 | void (*proc)(unsigned long)); |
@@ -290,7 +290,7 @@ static int swim3_readbit(struct floppy_state *fs, int bit) | |||
290 | return (stat & DATA) == 0; | 290 | return (stat & DATA) == 0; |
291 | } | 291 | } |
292 | 292 | ||
293 | static void do_fd_request(request_queue_t * q) | 293 | static void do_fd_request(struct request_queue * q) |
294 | { | 294 | { |
295 | int i; | 295 | int i; |
296 | for(i=0;i<floppy_count;i++) | 296 | for(i=0;i<floppy_count;i++) |
diff --git a/drivers/block/sx8.c b/drivers/block/sx8.c index 949ae93499e5..402209fec59a 100644 --- a/drivers/block/sx8.c +++ b/drivers/block/sx8.c | |||
@@ -278,7 +278,7 @@ struct carm_host { | |||
278 | unsigned int state; | 278 | unsigned int state; |
279 | u32 fw_ver; | 279 | u32 fw_ver; |
280 | 280 | ||
281 | request_queue_t *oob_q; | 281 | struct request_queue *oob_q; |
282 | unsigned int n_oob; | 282 | unsigned int n_oob; |
283 | 283 | ||
284 | unsigned int hw_sg_used; | 284 | unsigned int hw_sg_used; |
@@ -287,7 +287,7 @@ struct carm_host { | |||
287 | 287 | ||
288 | unsigned int wait_q_prod; | 288 | unsigned int wait_q_prod; |
289 | unsigned int wait_q_cons; | 289 | unsigned int wait_q_cons; |
290 | request_queue_t *wait_q[CARM_MAX_WAIT_Q]; | 290 | struct request_queue *wait_q[CARM_MAX_WAIT_Q]; |
291 | 291 | ||
292 | unsigned int n_msgs; | 292 | unsigned int n_msgs; |
293 | u64 msg_alloc; | 293 | u64 msg_alloc; |
@@ -756,7 +756,7 @@ static inline void carm_end_request_queued(struct carm_host *host, | |||
756 | assert(rc == 0); | 756 | assert(rc == 0); |
757 | } | 757 | } |
758 | 758 | ||
759 | static inline void carm_push_q (struct carm_host *host, request_queue_t *q) | 759 | static inline void carm_push_q (struct carm_host *host, struct request_queue *q) |
760 | { | 760 | { |
761 | unsigned int idx = host->wait_q_prod % CARM_MAX_WAIT_Q; | 761 | unsigned int idx = host->wait_q_prod % CARM_MAX_WAIT_Q; |
762 | 762 | ||
@@ -768,7 +768,7 @@ static inline void carm_push_q (struct carm_host *host, request_queue_t *q) | |||
768 | BUG_ON(host->wait_q_prod == host->wait_q_cons); /* overrun */ | 768 | BUG_ON(host->wait_q_prod == host->wait_q_cons); /* overrun */ |
769 | } | 769 | } |
770 | 770 | ||
771 | static inline request_queue_t *carm_pop_q(struct carm_host *host) | 771 | static inline struct request_queue *carm_pop_q(struct carm_host *host) |
772 | { | 772 | { |
773 | unsigned int idx; | 773 | unsigned int idx; |
774 | 774 | ||
@@ -783,7 +783,7 @@ static inline request_queue_t *carm_pop_q(struct carm_host *host) | |||
783 | 783 | ||
784 | static inline void carm_round_robin(struct carm_host *host) | 784 | static inline void carm_round_robin(struct carm_host *host) |
785 | { | 785 | { |
786 | request_queue_t *q = carm_pop_q(host); | 786 | struct request_queue *q = carm_pop_q(host); |
787 | if (q) { | 787 | if (q) { |
788 | blk_start_queue(q); | 788 | blk_start_queue(q); |
789 | VPRINTK("STARTED QUEUE %p\n", q); | 789 | VPRINTK("STARTED QUEUE %p\n", q); |
@@ -802,7 +802,7 @@ static inline void carm_end_rq(struct carm_host *host, struct carm_request *crq, | |||
802 | } | 802 | } |
803 | } | 803 | } |
804 | 804 | ||
805 | static void carm_oob_rq_fn(request_queue_t *q) | 805 | static void carm_oob_rq_fn(struct request_queue *q) |
806 | { | 806 | { |
807 | struct carm_host *host = q->queuedata; | 807 | struct carm_host *host = q->queuedata; |
808 | struct carm_request *crq; | 808 | struct carm_request *crq; |
@@ -833,7 +833,7 @@ static void carm_oob_rq_fn(request_queue_t *q) | |||
833 | } | 833 | } |
834 | } | 834 | } |
835 | 835 | ||
836 | static void carm_rq_fn(request_queue_t *q) | 836 | static void carm_rq_fn(struct request_queue *q) |
837 | { | 837 | { |
838 | struct carm_port *port = q->queuedata; | 838 | struct carm_port *port = q->queuedata; |
839 | struct carm_host *host = port->host; | 839 | struct carm_host *host = port->host; |
@@ -1494,7 +1494,7 @@ static int carm_init_disks(struct carm_host *host) | |||
1494 | 1494 | ||
1495 | for (i = 0; i < CARM_MAX_PORTS; i++) { | 1495 | for (i = 0; i < CARM_MAX_PORTS; i++) { |
1496 | struct gendisk *disk; | 1496 | struct gendisk *disk; |
1497 | request_queue_t *q; | 1497 | struct request_queue *q; |
1498 | struct carm_port *port; | 1498 | struct carm_port *port; |
1499 | 1499 | ||
1500 | port = &host->port[i]; | 1500 | port = &host->port[i]; |
@@ -1538,7 +1538,7 @@ static void carm_free_disks(struct carm_host *host) | |||
1538 | for (i = 0; i < CARM_MAX_PORTS; i++) { | 1538 | for (i = 0; i < CARM_MAX_PORTS; i++) { |
1539 | struct gendisk *disk = host->port[i].disk; | 1539 | struct gendisk *disk = host->port[i].disk; |
1540 | if (disk) { | 1540 | if (disk) { |
1541 | request_queue_t *q = disk->queue; | 1541 | struct request_queue *q = disk->queue; |
1542 | 1542 | ||
1543 | if (disk->flags & GENHD_FL_UP) | 1543 | if (disk->flags & GENHD_FL_UP) |
1544 | del_gendisk(disk); | 1544 | del_gendisk(disk); |
@@ -1571,7 +1571,7 @@ static int carm_init_one (struct pci_dev *pdev, const struct pci_device_id *ent) | |||
1571 | struct carm_host *host; | 1571 | struct carm_host *host; |
1572 | unsigned int pci_dac; | 1572 | unsigned int pci_dac; |
1573 | int rc; | 1573 | int rc; |
1574 | request_queue_t *q; | 1574 | struct request_queue *q; |
1575 | unsigned int i; | 1575 | unsigned int i; |
1576 | 1576 | ||
1577 | if (!printed_version++) | 1577 | if (!printed_version++) |
diff --git a/drivers/block/ub.c b/drivers/block/ub.c index 8b13d7d2cb63..c57dd2b3a0c8 100644 --- a/drivers/block/ub.c +++ b/drivers/block/ub.c | |||
@@ -503,7 +503,7 @@ static void ub_cleanup(struct ub_dev *sc) | |||
503 | { | 503 | { |
504 | struct list_head *p; | 504 | struct list_head *p; |
505 | struct ub_lun *lun; | 505 | struct ub_lun *lun; |
506 | request_queue_t *q; | 506 | struct request_queue *q; |
507 | 507 | ||
508 | while (!list_empty(&sc->luns)) { | 508 | while (!list_empty(&sc->luns)) { |
509 | p = sc->luns.next; | 509 | p = sc->luns.next; |
@@ -619,7 +619,7 @@ static struct ub_scsi_cmd *ub_cmdq_pop(struct ub_dev *sc) | |||
619 | * The request function is our main entry point | 619 | * The request function is our main entry point |
620 | */ | 620 | */ |
621 | 621 | ||
622 | static void ub_request_fn(request_queue_t *q) | 622 | static void ub_request_fn(struct request_queue *q) |
623 | { | 623 | { |
624 | struct ub_lun *lun = q->queuedata; | 624 | struct ub_lun *lun = q->queuedata; |
625 | struct request *rq; | 625 | struct request *rq; |
@@ -2273,7 +2273,7 @@ err_core: | |||
2273 | static int ub_probe_lun(struct ub_dev *sc, int lnum) | 2273 | static int ub_probe_lun(struct ub_dev *sc, int lnum) |
2274 | { | 2274 | { |
2275 | struct ub_lun *lun; | 2275 | struct ub_lun *lun; |
2276 | request_queue_t *q; | 2276 | struct request_queue *q; |
2277 | struct gendisk *disk; | 2277 | struct gendisk *disk; |
2278 | int rc; | 2278 | int rc; |
2279 | 2279 | ||
diff --git a/drivers/block/umem.c b/drivers/block/umem.c index dec74bd23496..6b7c02d6360d 100644 --- a/drivers/block/umem.c +++ b/drivers/block/umem.c | |||
@@ -114,7 +114,7 @@ struct cardinfo { | |||
114 | */ | 114 | */ |
115 | struct bio *bio, *currentbio, **biotail; | 115 | struct bio *bio, *currentbio, **biotail; |
116 | 116 | ||
117 | request_queue_t *queue; | 117 | struct request_queue *queue; |
118 | 118 | ||
119 | struct mm_page { | 119 | struct mm_page { |
120 | dma_addr_t page_dma; | 120 | dma_addr_t page_dma; |
@@ -357,7 +357,7 @@ static inline void reset_page(struct mm_page *page) | |||
357 | page->biotail = & page->bio; | 357 | page->biotail = & page->bio; |
358 | } | 358 | } |
359 | 359 | ||
360 | static void mm_unplug_device(request_queue_t *q) | 360 | static void mm_unplug_device(struct request_queue *q) |
361 | { | 361 | { |
362 | struct cardinfo *card = q->queuedata; | 362 | struct cardinfo *card = q->queuedata; |
363 | unsigned long flags; | 363 | unsigned long flags; |
@@ -541,7 +541,7 @@ static void process_page(unsigned long data) | |||
541 | -- mm_make_request | 541 | -- mm_make_request |
542 | ----------------------------------------------------------------------------------- | 542 | ----------------------------------------------------------------------------------- |
543 | */ | 543 | */ |
544 | static int mm_make_request(request_queue_t *q, struct bio *bio) | 544 | static int mm_make_request(struct request_queue *q, struct bio *bio) |
545 | { | 545 | { |
546 | struct cardinfo *card = q->queuedata; | 546 | struct cardinfo *card = q->queuedata; |
547 | pr_debug("mm_make_request %llu %u\n", | 547 | pr_debug("mm_make_request %llu %u\n", |
diff --git a/drivers/block/viodasd.c b/drivers/block/viodasd.c index dae39911a11d..85916e2665d4 100644 --- a/drivers/block/viodasd.c +++ b/drivers/block/viodasd.c | |||
@@ -400,7 +400,7 @@ error_ret: | |||
400 | /* | 400 | /* |
401 | * This is the external request processing routine | 401 | * This is the external request processing routine |
402 | */ | 402 | */ |
403 | static void do_viodasd_request(request_queue_t *q) | 403 | static void do_viodasd_request(struct request_queue *q) |
404 | { | 404 | { |
405 | struct request *req; | 405 | struct request *req; |
406 | 406 | ||
diff --git a/drivers/block/xd.c b/drivers/block/xd.c index 0d97b7eb818a..624d30f7da3f 100644 --- a/drivers/block/xd.c +++ b/drivers/block/xd.c | |||
@@ -298,7 +298,7 @@ static u_char __init xd_detect (u_char *controller, unsigned int *address) | |||
298 | } | 298 | } |
299 | 299 | ||
300 | /* do_xd_request: handle an incoming request */ | 300 | /* do_xd_request: handle an incoming request */ |
301 | static void do_xd_request (request_queue_t * q) | 301 | static void do_xd_request (struct request_queue * q) |
302 | { | 302 | { |
303 | struct request *req; | 303 | struct request *req; |
304 | 304 | ||
diff --git a/drivers/block/xd.h b/drivers/block/xd.h index 82e090fea957..cffd44a20383 100644 --- a/drivers/block/xd.h +++ b/drivers/block/xd.h | |||
@@ -104,7 +104,7 @@ static int xd_manual_geo_init (char *command); | |||
104 | static u_char xd_detect (u_char *controller, unsigned int *address); | 104 | static u_char xd_detect (u_char *controller, unsigned int *address); |
105 | static u_char xd_initdrives (void (*init_drive)(u_char drive)); | 105 | static u_char xd_initdrives (void (*init_drive)(u_char drive)); |
106 | 106 | ||
107 | static void do_xd_request (request_queue_t * q); | 107 | static void do_xd_request (struct request_queue * q); |
108 | static int xd_ioctl (struct inode *inode,struct file *file,unsigned int cmd,unsigned long arg); | 108 | static int xd_ioctl (struct inode *inode,struct file *file,unsigned int cmd,unsigned long arg); |
109 | static int xd_readwrite (u_char operation,XD_INFO *disk,char *buffer,u_int block,u_int count); | 109 | static int xd_readwrite (u_char operation,XD_INFO *disk,char *buffer,u_int block,u_int count); |
110 | static void xd_recalibrate (u_char drive); | 110 | static void xd_recalibrate (u_char drive); |
diff --git a/drivers/block/xen-blkfront.c b/drivers/block/xen-blkfront.c index 6746c29181f8..964e51634f2d 100644 --- a/drivers/block/xen-blkfront.c +++ b/drivers/block/xen-blkfront.c | |||
@@ -241,7 +241,7 @@ static inline void flush_requests(struct blkfront_info *info) | |||
241 | * do_blkif_request | 241 | * do_blkif_request |
242 | * read a block; request is in a request queue | 242 | * read a block; request is in a request queue |
243 | */ | 243 | */ |
244 | static void do_blkif_request(request_queue_t *rq) | 244 | static void do_blkif_request(struct request_queue *rq) |
245 | { | 245 | { |
246 | struct blkfront_info *info = NULL; | 246 | struct blkfront_info *info = NULL; |
247 | struct request *req; | 247 | struct request *req; |
@@ -287,7 +287,7 @@ wait: | |||
287 | 287 | ||
288 | static int xlvbd_init_blk_queue(struct gendisk *gd, u16 sector_size) | 288 | static int xlvbd_init_blk_queue(struct gendisk *gd, u16 sector_size) |
289 | { | 289 | { |
290 | request_queue_t *rq; | 290 | struct request_queue *rq; |
291 | 291 | ||
292 | rq = blk_init_queue(do_blkif_request, &blkif_io_lock); | 292 | rq = blk_init_queue(do_blkif_request, &blkif_io_lock); |
293 | if (rq == NULL) | 293 | if (rq == NULL) |
diff --git a/drivers/block/xsysace.c b/drivers/block/xsysace.c index 732ec63b6e9c..cb27e8863d7c 100644 --- a/drivers/block/xsysace.c +++ b/drivers/block/xsysace.c | |||
@@ -458,7 +458,7 @@ static inline void ace_fsm_yieldirq(struct ace_device *ace) | |||
458 | } | 458 | } |
459 | 459 | ||
460 | /* Get the next read/write request; ending requests that we don't handle */ | 460 | /* Get the next read/write request; ending requests that we don't handle */ |
461 | struct request *ace_get_next_request(request_queue_t * q) | 461 | struct request *ace_get_next_request(struct request_queue * q) |
462 | { | 462 | { |
463 | struct request *req; | 463 | struct request *req; |
464 | 464 | ||
@@ -825,7 +825,7 @@ static irqreturn_t ace_interrupt(int irq, void *dev_id) | |||
825 | /* --------------------------------------------------------------------- | 825 | /* --------------------------------------------------------------------- |
826 | * Block ops | 826 | * Block ops |
827 | */ | 827 | */ |
828 | static void ace_request(request_queue_t * q) | 828 | static void ace_request(struct request_queue * q) |
829 | { | 829 | { |
830 | struct request *req; | 830 | struct request *req; |
831 | struct ace_device *ace; | 831 | struct ace_device *ace; |
diff --git a/drivers/block/z2ram.c b/drivers/block/z2ram.c index e40fa98842e5..2d5853cbd4b0 100644 --- a/drivers/block/z2ram.c +++ b/drivers/block/z2ram.c | |||
@@ -67,7 +67,7 @@ static DEFINE_SPINLOCK(z2ram_lock); | |||
67 | static struct block_device_operations z2_fops; | 67 | static struct block_device_operations z2_fops; |
68 | static struct gendisk *z2ram_gendisk; | 68 | static struct gendisk *z2ram_gendisk; |
69 | 69 | ||
70 | static void do_z2_request(request_queue_t *q) | 70 | static void do_z2_request(struct request_queue *q) |
71 | { | 71 | { |
72 | struct request *req; | 72 | struct request *req; |
73 | while ((req = elv_next_request(q)) != NULL) { | 73 | while ((req = elv_next_request(q)) != NULL) { |
diff --git a/drivers/cdrom/cdrom.c b/drivers/cdrom/cdrom.c index 499019bf8f40..67ee3d4b2878 100644 --- a/drivers/cdrom/cdrom.c +++ b/drivers/cdrom/cdrom.c | |||
@@ -2094,7 +2094,7 @@ out: | |||
2094 | static int cdrom_read_cdda_bpc(struct cdrom_device_info *cdi, __u8 __user *ubuf, | 2094 | static int cdrom_read_cdda_bpc(struct cdrom_device_info *cdi, __u8 __user *ubuf, |
2095 | int lba, int nframes) | 2095 | int lba, int nframes) |
2096 | { | 2096 | { |
2097 | request_queue_t *q = cdi->disk->queue; | 2097 | struct request_queue *q = cdi->disk->queue; |
2098 | struct request *rq; | 2098 | struct request *rq; |
2099 | struct bio *bio; | 2099 | struct bio *bio; |
2100 | unsigned int len; | 2100 | unsigned int len; |
diff --git a/drivers/cdrom/viocd.c b/drivers/cdrom/viocd.c index 44cd7b2ddf09..e51550db1575 100644 --- a/drivers/cdrom/viocd.c +++ b/drivers/cdrom/viocd.c | |||
@@ -398,7 +398,7 @@ static void viocd_end_request(struct request *req, int uptodate) | |||
398 | 398 | ||
399 | static int rwreq; | 399 | static int rwreq; |
400 | 400 | ||
401 | static void do_viocd_request(request_queue_t *q) | 401 | static void do_viocd_request(struct request_queue *q) |
402 | { | 402 | { |
403 | struct request *req; | 403 | struct request *req; |
404 | 404 | ||
diff --git a/drivers/char/Kconfig b/drivers/char/Kconfig index c8dfd18bea44..b391776e5bf3 100644 --- a/drivers/char/Kconfig +++ b/drivers/char/Kconfig | |||
@@ -130,6 +130,7 @@ config ROCKETPORT | |||
130 | config CYCLADES | 130 | config CYCLADES |
131 | tristate "Cyclades async mux support" | 131 | tristate "Cyclades async mux support" |
132 | depends on SERIAL_NONSTANDARD && (PCI || ISA) | 132 | depends on SERIAL_NONSTANDARD && (PCI || ISA) |
133 | select FW_LOADER | ||
133 | ---help--- | 134 | ---help--- |
134 | This driver supports Cyclades Z and Y multiserial boards. | 135 | This driver supports Cyclades Z and Y multiserial boards. |
135 | You would need something like this to connect more than two modems to | 136 | You would need something like this to connect more than two modems to |
@@ -726,7 +727,7 @@ config NVRAM | |||
726 | 727 | ||
727 | config RTC | 728 | config RTC |
728 | tristate "Enhanced Real Time Clock Support" | 729 | tristate "Enhanced Real Time Clock Support" |
729 | depends on !PPC && !PARISC && !IA64 && !M68K && !SPARC64 && (!SPARC32 || PCI) && !FRV && !ARM && !SUPERH && !S390 | 730 | depends on !PPC && !PARISC && !IA64 && !M68K && !SPARC && !FRV && !ARM && !SUPERH && !S390 |
730 | ---help--- | 731 | ---help--- |
731 | If you say Y here and create a character special file /dev/rtc with | 732 | If you say Y here and create a character special file /dev/rtc with |
732 | major number 10 and minor number 135 using mknod ("man mknod"), you | 733 | major number 10 and minor number 135 using mknod ("man mknod"), you |
@@ -750,6 +751,28 @@ config RTC | |||
750 | To compile this driver as a module, choose M here: the | 751 | To compile this driver as a module, choose M here: the |
751 | module will be called rtc. | 752 | module will be called rtc. |
752 | 753 | ||
754 | config JS_RTC | ||
755 | tristate "Enhanced Real Time Clock Support" | ||
756 | depends on SPARC32 && PCI | ||
757 | ---help--- | ||
758 | If you say Y here and create a character special file /dev/rtc with | ||
759 | major number 10 and minor number 135 using mknod ("man mknod"), you | ||
760 | will get access to the real time clock (or hardware clock) built | ||
761 | into your computer. | ||
762 | |||
763 | Every PC has such a clock built in. It can be used to generate | ||
764 | signals from as low as 1Hz up to 8192Hz, and can also be used | ||
765 | as a 24 hour alarm. It reports status information via the file | ||
766 | /proc/driver/rtc and its behaviour is set by various ioctls on | ||
767 | /dev/rtc. | ||
768 | |||
769 | If you think you have a use for such a device (such as periodic data | ||
770 | sampling), then say Y here, and read <file:Documentation/rtc.txt> | ||
771 | for details. | ||
772 | |||
773 | To compile this driver as a module, choose M here: the | ||
774 | module will be called js-rtc. | ||
775 | |||
753 | config SGI_DS1286 | 776 | config SGI_DS1286 |
754 | tristate "SGI DS1286 RTC support" | 777 | tristate "SGI DS1286 RTC support" |
755 | depends on SGI_IP22 | 778 | depends on SGI_IP22 |
diff --git a/drivers/char/Makefile b/drivers/char/Makefile index 8fecaf4010b1..23b26b87cc32 100644 --- a/drivers/char/Makefile +++ b/drivers/char/Makefile | |||
@@ -109,6 +109,9 @@ obj-$(CONFIG_TCG_TPM) += tpm/ | |||
109 | 109 | ||
110 | obj-$(CONFIG_PS3_FLASH) += ps3flash.o | 110 | obj-$(CONFIG_PS3_FLASH) += ps3flash.o |
111 | 111 | ||
112 | obj-$(CONFIG_JS_RTC) += js-rtc.o | ||
113 | js-rtc-y = rtc.o | ||
114 | |||
112 | # Files generated that shall be removed upon make clean | 115 | # Files generated that shall be removed upon make clean |
113 | clean-files := consolemap_deftbl.c defkeymap.c | 116 | clean-files := consolemap_deftbl.c defkeymap.c |
114 | 117 | ||
diff --git a/drivers/char/hpet.c b/drivers/char/hpet.c index ba0e74ad74bb..77bf4aa217a8 100644 --- a/drivers/char/hpet.c +++ b/drivers/char/hpet.c | |||
@@ -73,7 +73,7 @@ static struct clocksource clocksource_hpet = { | |||
73 | .name = "hpet", | 73 | .name = "hpet", |
74 | .rating = 250, | 74 | .rating = 250, |
75 | .read = read_hpet, | 75 | .read = read_hpet, |
76 | .mask = 0xffffffffffffffff, | 76 | .mask = CLOCKSOURCE_MASK(64), |
77 | .mult = 0, /*to be caluclated*/ | 77 | .mult = 0, /*to be caluclated*/ |
78 | .shift = 10, | 78 | .shift = 10, |
79 | .flags = CLOCK_SOURCE_IS_CONTINUOUS, | 79 | .flags = CLOCK_SOURCE_IS_CONTINUOUS, |
@@ -1007,9 +1007,15 @@ static int hpet_acpi_remove(struct acpi_device *device, int type) | |||
1007 | return -EINVAL; | 1007 | return -EINVAL; |
1008 | } | 1008 | } |
1009 | 1009 | ||
1010 | static const struct acpi_device_id hpet_device_ids[] = { | ||
1011 | {"PNP0103", 0}, | ||
1012 | {"", 0}, | ||
1013 | }; | ||
1014 | MODULE_DEVICE_TABLE(acpi, hpet_device_ids); | ||
1015 | |||
1010 | static struct acpi_driver hpet_acpi_driver = { | 1016 | static struct acpi_driver hpet_acpi_driver = { |
1011 | .name = "hpet", | 1017 | .name = "hpet", |
1012 | .ids = "PNP0103", | 1018 | .ids = hpet_device_ids, |
1013 | .ops = { | 1019 | .ops = { |
1014 | .add = hpet_acpi_add, | 1020 | .add = hpet_acpi_add, |
1015 | .remove = hpet_acpi_remove, | 1021 | .remove = hpet_acpi_remove, |
diff --git a/drivers/char/hvc_lguest.c b/drivers/char/hvc_lguest.c index e7b889e404a7..feeccbaec438 100644 --- a/drivers/char/hvc_lguest.c +++ b/drivers/char/hvc_lguest.c | |||
@@ -1,6 +1,22 @@ | |||
1 | /* Simple console for lguest. | 1 | /*D:300 |
2 | * The Guest console driver | ||
2 | * | 3 | * |
3 | * Copyright (C) 2006 Rusty Russell, IBM Corporation | 4 | * This is a trivial console driver: we use lguest's DMA mechanism to send |
5 | * bytes out, and register a DMA buffer to receive bytes in. It is assumed to | ||
6 | * be present and available from the very beginning of boot. | ||
7 | * | ||
8 | * Writing console drivers is one of the few remaining Dark Arts in Linux. | ||
9 | * Fortunately for us, the path of virtual consoles has been well-trodden by | ||
10 | * the PowerPC folks, who wrote "hvc_console.c" to generically support any | ||
11 | * virtual console. We use that infrastructure which only requires us to write | ||
12 | * the basic put_chars and get_chars functions and call the right register | ||
13 | * functions. | ||
14 | :*/ | ||
15 | |||
16 | /*M:002 The console can be flooded: while the Guest is processing input the | ||
17 | * Host can send more. Buffering in the Host could alleviate this, but it is a | ||
18 | * difficult problem in general. :*/ | ||
19 | /* Copyright (C) 2006 Rusty Russell, IBM Corporation | ||
4 | * | 20 | * |
5 | * This program is free software; you can redistribute it and/or modify | 21 | * This program is free software; you can redistribute it and/or modify |
6 | * it under the terms of the GNU General Public License as published by | 22 | * it under the terms of the GNU General Public License as published by |
@@ -21,49 +37,81 @@ | |||
21 | #include <linux/lguest_bus.h> | 37 | #include <linux/lguest_bus.h> |
22 | #include "hvc_console.h" | 38 | #include "hvc_console.h" |
23 | 39 | ||
40 | /*D:340 This is our single console input buffer, with associated "struct | ||
41 | * lguest_dma" referring to it. Note the 0-terminated length array, and the | ||
42 | * use of physical address for the buffer itself. */ | ||
24 | static char inbuf[256]; | 43 | static char inbuf[256]; |
25 | static struct lguest_dma cons_input = { .used_len = 0, | 44 | static struct lguest_dma cons_input = { .used_len = 0, |
26 | .addr[0] = __pa(inbuf), | 45 | .addr[0] = __pa(inbuf), |
27 | .len[0] = sizeof(inbuf), | 46 | .len[0] = sizeof(inbuf), |
28 | .len[1] = 0 }; | 47 | .len[1] = 0 }; |
29 | 48 | ||
49 | /*D:310 The put_chars() callback is pretty straightforward. | ||
50 | * | ||
51 | * First we put the pointer and length in a "struct lguest_dma": we only have | ||
52 | * one pointer, so we set the second length to 0. Then we use SEND_DMA to send | ||
53 | * the data to (Host) buffers attached to the console key. Usually a device's | ||
54 | * key is a physical address within the device's memory, but because the | ||
55 | * console device doesn't have any associated physical memory, we use the | ||
56 | * LGUEST_CONSOLE_DMA_KEY constant (aka 0). */ | ||
30 | static int put_chars(u32 vtermno, const char *buf, int count) | 57 | static int put_chars(u32 vtermno, const char *buf, int count) |
31 | { | 58 | { |
32 | struct lguest_dma dma; | 59 | struct lguest_dma dma; |
33 | 60 | ||
34 | /* FIXME: what if it's over a page boundary? */ | 61 | /* FIXME: DMA buffers in a "struct lguest_dma" are not allowed |
62 | * to go over page boundaries. This never seems to happen, | ||
63 | * but if it did we'd need to fix this code. */ | ||
35 | dma.len[0] = count; | 64 | dma.len[0] = count; |
36 | dma.len[1] = 0; | 65 | dma.len[1] = 0; |
37 | dma.addr[0] = __pa(buf); | 66 | dma.addr[0] = __pa(buf); |
38 | 67 | ||
39 | lguest_send_dma(LGUEST_CONSOLE_DMA_KEY, &dma); | 68 | lguest_send_dma(LGUEST_CONSOLE_DMA_KEY, &dma); |
69 | /* We're expected to return the amount of data we wrote: all of it. */ | ||
40 | return count; | 70 | return count; |
41 | } | 71 | } |
42 | 72 | ||
73 | /*D:350 get_chars() is the callback from the hvc_console infrastructure when | ||
74 | * an interrupt is received. | ||
75 | * | ||
76 | * Firstly we see if our buffer has been filled: if not, we return. The rest | ||
77 | * of the code deals with the fact that the hvc_console() infrastructure only | ||
78 | * asks us for 16 bytes at a time. We keep a "cons_offset" variable for | ||
79 | * partially-read buffers. */ | ||
43 | static int get_chars(u32 vtermno, char *buf, int count) | 80 | static int get_chars(u32 vtermno, char *buf, int count) |
44 | { | 81 | { |
45 | static int cons_offset; | 82 | static int cons_offset; |
46 | 83 | ||
84 | /* Nothing left to see here... */ | ||
47 | if (!cons_input.used_len) | 85 | if (!cons_input.used_len) |
48 | return 0; | 86 | return 0; |
49 | 87 | ||
88 | /* You want more than we have to give? Well, try wanting less! */ | ||
50 | if (cons_input.used_len - cons_offset < count) | 89 | if (cons_input.used_len - cons_offset < count) |
51 | count = cons_input.used_len - cons_offset; | 90 | count = cons_input.used_len - cons_offset; |
52 | 91 | ||
92 | /* Copy across to their buffer and increment offset. */ | ||
53 | memcpy(buf, inbuf + cons_offset, count); | 93 | memcpy(buf, inbuf + cons_offset, count); |
54 | cons_offset += count; | 94 | cons_offset += count; |
95 | |||
96 | /* Finished? Zero offset, and reset cons_input so Host will use it | ||
97 | * again. */ | ||
55 | if (cons_offset == cons_input.used_len) { | 98 | if (cons_offset == cons_input.used_len) { |
56 | cons_offset = 0; | 99 | cons_offset = 0; |
57 | cons_input.used_len = 0; | 100 | cons_input.used_len = 0; |
58 | } | 101 | } |
59 | return count; | 102 | return count; |
60 | } | 103 | } |
104 | /*:*/ | ||
61 | 105 | ||
62 | static struct hv_ops lguest_cons = { | 106 | static struct hv_ops lguest_cons = { |
63 | .get_chars = get_chars, | 107 | .get_chars = get_chars, |
64 | .put_chars = put_chars, | 108 | .put_chars = put_chars, |
65 | }; | 109 | }; |
66 | 110 | ||
111 | /*D:320 Console drivers are initialized very early so boot messages can go | ||
112 | * out. At this stage, the console is output-only. Our driver checks we're a | ||
113 | * Guest, and if so hands hvc_instantiate() the console number (0), priority | ||
114 | * (0), and the struct hv_ops containing the put_chars() function. */ | ||
67 | static int __init cons_init(void) | 115 | static int __init cons_init(void) |
68 | { | 116 | { |
69 | if (strcmp(paravirt_ops.name, "lguest") != 0) | 117 | if (strcmp(paravirt_ops.name, "lguest") != 0) |
@@ -73,21 +121,46 @@ static int __init cons_init(void) | |||
73 | } | 121 | } |
74 | console_initcall(cons_init); | 122 | console_initcall(cons_init); |
75 | 123 | ||
124 | /*D:370 To set up and manage our virtual console, we call hvc_alloc() and | ||
125 | * stash the result in the private pointer of the "struct lguest_device". | ||
126 | * Since we never remove the console device we never need this pointer again, | ||
127 | * but using ->private is considered good form, and you never know who's going | ||
128 | * to copy your driver. | ||
129 | * | ||
130 | * Once the console is set up, we bind our input buffer ready for input. */ | ||
76 | static int lguestcons_probe(struct lguest_device *lgdev) | 131 | static int lguestcons_probe(struct lguest_device *lgdev) |
77 | { | 132 | { |
78 | int err; | 133 | int err; |
79 | 134 | ||
135 | /* The first argument of hvc_alloc() is the virtual console number, so | ||
136 | * we use zero. The second argument is the interrupt number. | ||
137 | * | ||
138 | * The third argument is a "struct hv_ops" containing the put_chars() | ||
139 | * and get_chars() pointers. The final argument is the output buffer | ||
140 | * size: we use 256 and expect the Host to have room for us to send | ||
141 | * that much. */ | ||
80 | lgdev->private = hvc_alloc(0, lgdev_irq(lgdev), &lguest_cons, 256); | 142 | lgdev->private = hvc_alloc(0, lgdev_irq(lgdev), &lguest_cons, 256); |
81 | if (IS_ERR(lgdev->private)) | 143 | if (IS_ERR(lgdev->private)) |
82 | return PTR_ERR(lgdev->private); | 144 | return PTR_ERR(lgdev->private); |
83 | 145 | ||
146 | /* We bind a single DMA buffer at key LGUEST_CONSOLE_DMA_KEY. | ||
147 | * "cons_input" is that statically-initialized global DMA buffer we saw | ||
148 | * above, and we also give the interrupt we want. */ | ||
84 | err = lguest_bind_dma(LGUEST_CONSOLE_DMA_KEY, &cons_input, 1, | 149 | err = lguest_bind_dma(LGUEST_CONSOLE_DMA_KEY, &cons_input, 1, |
85 | lgdev_irq(lgdev)); | 150 | lgdev_irq(lgdev)); |
86 | if (err) | 151 | if (err) |
87 | printk("lguest console: failed to bind buffer.\n"); | 152 | printk("lguest console: failed to bind buffer.\n"); |
88 | return err; | 153 | return err; |
89 | } | 154 | } |
155 | /* Note the use of lgdev_irq() for the interrupt number. We tell hvc_alloc() | ||
156 | * to expect input when this interrupt is triggered, and then tell | ||
157 | * lguest_bind_dma() that is the interrupt to send us when input comes in. */ | ||
90 | 158 | ||
159 | /*D:360 From now on the console driver follows standard Guest driver form: | ||
160 | * register_lguest_driver() registers the device type and probe function, and | ||
161 | * the probe function sets up the device. | ||
162 | * | ||
163 | * The standard "struct lguest_driver": */ | ||
91 | static struct lguest_driver lguestcons_drv = { | 164 | static struct lguest_driver lguestcons_drv = { |
92 | .name = "lguestcons", | 165 | .name = "lguestcons", |
93 | .owner = THIS_MODULE, | 166 | .owner = THIS_MODULE, |
@@ -95,6 +168,7 @@ static struct lguest_driver lguestcons_drv = { | |||
95 | .probe = lguestcons_probe, | 168 | .probe = lguestcons_probe, |
96 | }; | 169 | }; |
97 | 170 | ||
171 | /* The standard init function */ | ||
98 | static int __init hvc_lguest_init(void) | 172 | static int __init hvc_lguest_init(void) |
99 | { | 173 | { |
100 | return register_lguest_driver(&lguestcons_drv); | 174 | return register_lguest_driver(&lguestcons_drv); |
diff --git a/drivers/edac/Kconfig b/drivers/edac/Kconfig index 1724c41d2414..98b6b4fb4257 100644 --- a/drivers/edac/Kconfig +++ b/drivers/edac/Kconfig | |||
@@ -8,7 +8,7 @@ menuconfig EDAC | |||
8 | bool "EDAC - error detection and reporting (EXPERIMENTAL)" | 8 | bool "EDAC - error detection and reporting (EXPERIMENTAL)" |
9 | depends on HAS_IOMEM | 9 | depends on HAS_IOMEM |
10 | depends on EXPERIMENTAL | 10 | depends on EXPERIMENTAL |
11 | depends on X86 || MIPS || PPC | 11 | depends on X86 || PPC |
12 | help | 12 | help |
13 | EDAC is designed to report errors in the core system. | 13 | EDAC is designed to report errors in the core system. |
14 | These are low-level errors that are reported in the CPU or | 14 | These are low-level errors that are reported in the CPU or |
@@ -126,7 +126,7 @@ config EDAC_I5000 | |||
126 | config EDAC_PASEMI | 126 | config EDAC_PASEMI |
127 | tristate "PA Semi PWRficient" | 127 | tristate "PA Semi PWRficient" |
128 | depends on EDAC_MM_EDAC && PCI | 128 | depends on EDAC_MM_EDAC && PCI |
129 | depends on PPC | 129 | depends on PPC_PASEMI |
130 | help | 130 | help |
131 | Support for error detection and correction on PA Semi | 131 | Support for error detection and correction on PA Semi |
132 | PWRficient. | 132 | PWRficient. |
diff --git a/drivers/edac/edac_mc.c b/drivers/edac/edac_mc.c index 4471be362599..063a1bffe38b 100644 --- a/drivers/edac/edac_mc.c +++ b/drivers/edac/edac_mc.c | |||
@@ -214,6 +214,13 @@ void edac_mc_free(struct mem_ctl_info *mci) | |||
214 | } | 214 | } |
215 | EXPORT_SYMBOL_GPL(edac_mc_free); | 215 | EXPORT_SYMBOL_GPL(edac_mc_free); |
216 | 216 | ||
217 | |||
218 | /* | ||
219 | * find_mci_by_dev | ||
220 | * | ||
221 | * scan list of controllers looking for the one that manages | ||
222 | * the 'dev' device | ||
223 | */ | ||
217 | static struct mem_ctl_info *find_mci_by_dev(struct device *dev) | 224 | static struct mem_ctl_info *find_mci_by_dev(struct device *dev) |
218 | { | 225 | { |
219 | struct mem_ctl_info *mci; | 226 | struct mem_ctl_info *mci; |
@@ -268,12 +275,6 @@ static void edac_mc_workq_function(struct work_struct *work_req) | |||
268 | if (edac_mc_assert_error_check_and_clear() && (mci->edac_check != NULL)) | 275 | if (edac_mc_assert_error_check_and_clear() && (mci->edac_check != NULL)) |
269 | mci->edac_check(mci); | 276 | mci->edac_check(mci); |
270 | 277 | ||
271 | /* | ||
272 | * FIXME: temp place holder for PCI checks, | ||
273 | * goes away when we break out PCI | ||
274 | */ | ||
275 | edac_pci_do_parity_check(); | ||
276 | |||
277 | mutex_unlock(&mem_ctls_mutex); | 278 | mutex_unlock(&mem_ctls_mutex); |
278 | 279 | ||
279 | /* Reschedule */ | 280 | /* Reschedule */ |
@@ -314,36 +315,55 @@ static void edac_mc_workq_teardown(struct mem_ctl_info *mci) | |||
314 | { | 315 | { |
315 | int status; | 316 | int status; |
316 | 317 | ||
317 | /* if not running POLL, leave now */ | 318 | status = cancel_delayed_work(&mci->work); |
318 | if (mci->op_state == OP_RUNNING_POLL) { | 319 | if (status == 0) { |
319 | status = cancel_delayed_work(&mci->work); | 320 | debugf0("%s() not canceled, flush the queue\n", |
320 | if (status == 0) { | 321 | __func__); |
321 | debugf0("%s() not canceled, flush the queue\n", | ||
322 | __func__); | ||
323 | 322 | ||
324 | /* workq instance might be running, wait for it */ | 323 | /* workq instance might be running, wait for it */ |
325 | flush_workqueue(edac_workqueue); | 324 | flush_workqueue(edac_workqueue); |
326 | } | ||
327 | } | 325 | } |
328 | } | 326 | } |
329 | 327 | ||
330 | /* | 328 | /* |
331 | * edac_reset_delay_period | 329 | * edac_mc_reset_delay_period(unsigned long value) |
330 | * | ||
331 | * user space has updated our poll period value, need to | ||
332 | * reset our workq delays | ||
332 | */ | 333 | */ |
333 | static void edac_reset_delay_period(struct mem_ctl_info *mci, unsigned long value) | 334 | void edac_mc_reset_delay_period(int value) |
334 | { | 335 | { |
335 | /* cancel the current workq request */ | 336 | struct mem_ctl_info *mci; |
336 | edac_mc_workq_teardown(mci); | 337 | struct list_head *item; |
337 | 338 | ||
338 | /* lock the list of devices for the new setup */ | ||
339 | mutex_lock(&mem_ctls_mutex); | 339 | mutex_lock(&mem_ctls_mutex); |
340 | 340 | ||
341 | /* restart the workq request, with new delay value */ | 341 | /* scan the list and turn off all workq timers, doing so under lock |
342 | edac_mc_workq_setup(mci, value); | 342 | */ |
343 | list_for_each(item, &mc_devices) { | ||
344 | mci = list_entry(item, struct mem_ctl_info, link); | ||
345 | |||
346 | if (mci->op_state == OP_RUNNING_POLL) | ||
347 | cancel_delayed_work(&mci->work); | ||
348 | } | ||
349 | |||
350 | mutex_unlock(&mem_ctls_mutex); | ||
351 | |||
352 | |||
353 | /* re-walk the list, and reset the poll delay */ | ||
354 | mutex_lock(&mem_ctls_mutex); | ||
355 | |||
356 | list_for_each(item, &mc_devices) { | ||
357 | mci = list_entry(item, struct mem_ctl_info, link); | ||
358 | |||
359 | edac_mc_workq_setup(mci, (unsigned long) value); | ||
360 | } | ||
343 | 361 | ||
344 | mutex_unlock(&mem_ctls_mutex); | 362 | mutex_unlock(&mem_ctls_mutex); |
345 | } | 363 | } |
346 | 364 | ||
365 | |||
366 | |||
347 | /* Return 0 on success, 1 on failure. | 367 | /* Return 0 on success, 1 on failure. |
348 | * Before calling this function, caller must | 368 | * Before calling this function, caller must |
349 | * assign a unique value to mci->mc_idx. | 369 | * assign a unique value to mci->mc_idx. |
diff --git a/drivers/edac/edac_mc_sysfs.c b/drivers/edac/edac_mc_sysfs.c index cd090b0677a7..4a0576bd06fc 100644 --- a/drivers/edac/edac_mc_sysfs.c +++ b/drivers/edac/edac_mc_sysfs.c | |||
@@ -122,6 +122,23 @@ static ssize_t memctrl_int_store(void *ptr, const char *buffer, size_t count) | |||
122 | return count; | 122 | return count; |
123 | } | 123 | } |
124 | 124 | ||
125 | /* | ||
126 | * mc poll_msec time value | ||
127 | */ | ||
128 | static ssize_t poll_msec_int_store(void *ptr, const char *buffer, size_t count) | ||
129 | { | ||
130 | int *value = (int *)ptr; | ||
131 | |||
132 | if (isdigit(*buffer)) { | ||
133 | *value = simple_strtoul(buffer, NULL, 0); | ||
134 | |||
135 | /* notify edac_mc engine to reset the poll period */ | ||
136 | edac_mc_reset_delay_period(*value); | ||
137 | } | ||
138 | |||
139 | return count; | ||
140 | } | ||
141 | |||
125 | 142 | ||
126 | /* EDAC sysfs CSROW data structures and methods | 143 | /* EDAC sysfs CSROW data structures and methods |
127 | */ | 144 | */ |
@@ -704,7 +721,7 @@ MEMCTRL_ATTR(edac_mc_log_ce, | |||
704 | S_IRUGO | S_IWUSR, memctrl_int_show, memctrl_int_store); | 721 | S_IRUGO | S_IWUSR, memctrl_int_show, memctrl_int_store); |
705 | 722 | ||
706 | MEMCTRL_ATTR(edac_mc_poll_msec, | 723 | MEMCTRL_ATTR(edac_mc_poll_msec, |
707 | S_IRUGO | S_IWUSR, memctrl_int_show, memctrl_int_store); | 724 | S_IRUGO | S_IWUSR, memctrl_int_show, poll_msec_int_store); |
708 | 725 | ||
709 | /* Base Attributes of the memory ECC object */ | 726 | /* Base Attributes of the memory ECC object */ |
710 | static struct memctrl_dev_attribute *memctrl_attr[] = { | 727 | static struct memctrl_dev_attribute *memctrl_attr[] = { |
diff --git a/drivers/edac/edac_module.h b/drivers/edac/edac_module.h index a2134dfc3cc6..cbc419c8ebc1 100644 --- a/drivers/edac/edac_module.h +++ b/drivers/edac/edac_module.h | |||
@@ -52,6 +52,8 @@ extern void edac_device_workq_setup(struct edac_device_ctl_info *edac_dev, | |||
52 | extern void edac_device_workq_teardown(struct edac_device_ctl_info *edac_dev); | 52 | extern void edac_device_workq_teardown(struct edac_device_ctl_info *edac_dev); |
53 | extern void edac_device_reset_delay_period(struct edac_device_ctl_info | 53 | extern void edac_device_reset_delay_period(struct edac_device_ctl_info |
54 | *edac_dev, unsigned long value); | 54 | *edac_dev, unsigned long value); |
55 | extern void edac_mc_reset_delay_period(int value); | ||
56 | |||
55 | extern void *edac_align_ptr(void *ptr, unsigned size); | 57 | extern void *edac_align_ptr(void *ptr, unsigned size); |
56 | 58 | ||
57 | /* | 59 | /* |
@@ -64,6 +66,10 @@ extern int edac_sysfs_pci_setup(void); | |||
64 | extern void edac_sysfs_pci_teardown(void); | 66 | extern void edac_sysfs_pci_teardown(void); |
65 | extern int edac_pci_get_check_errors(void); | 67 | extern int edac_pci_get_check_errors(void); |
66 | extern int edac_pci_get_poll_msec(void); | 68 | extern int edac_pci_get_poll_msec(void); |
69 | extern void edac_pci_remove_sysfs(struct edac_pci_ctl_info *pci); | ||
70 | extern void edac_pci_handle_pe(struct edac_pci_ctl_info *pci, const char *msg); | ||
71 | extern void edac_pci_handle_npe(struct edac_pci_ctl_info *pci, | ||
72 | const char *msg); | ||
67 | #else /* CONFIG_PCI */ | 73 | #else /* CONFIG_PCI */ |
68 | /* pre-process these away */ | 74 | /* pre-process these away */ |
69 | #define edac_pci_do_parity_check() | 75 | #define edac_pci_do_parity_check() |
@@ -72,6 +78,8 @@ extern int edac_pci_get_poll_msec(void); | |||
72 | #define edac_sysfs_pci_teardown() | 78 | #define edac_sysfs_pci_teardown() |
73 | #define edac_pci_get_check_errors() | 79 | #define edac_pci_get_check_errors() |
74 | #define edac_pci_get_poll_msec() | 80 | #define edac_pci_get_poll_msec() |
81 | #define edac_pci_handle_pe() | ||
82 | #define edac_pci_handle_npe() | ||
75 | #endif /* CONFIG_PCI */ | 83 | #endif /* CONFIG_PCI */ |
76 | 84 | ||
77 | #endif /* __EDAC_MODULE_H__ */ | 85 | #endif /* __EDAC_MODULE_H__ */ |
diff --git a/drivers/edac/edac_pci.c b/drivers/edac/edac_pci.c index d9cd5e048cee..5dee9f50414b 100644 --- a/drivers/edac/edac_pci.c +++ b/drivers/edac/edac_pci.c | |||
@@ -31,20 +31,12 @@ | |||
31 | static DEFINE_MUTEX(edac_pci_ctls_mutex); | 31 | static DEFINE_MUTEX(edac_pci_ctls_mutex); |
32 | static struct list_head edac_pci_list = LIST_HEAD_INIT(edac_pci_list); | 32 | static struct list_head edac_pci_list = LIST_HEAD_INIT(edac_pci_list); |
33 | 33 | ||
34 | static inline void edac_lock_pci_list(void) | ||
35 | { | ||
36 | mutex_lock(&edac_pci_ctls_mutex); | ||
37 | } | ||
38 | |||
39 | static inline void edac_unlock_pci_list(void) | ||
40 | { | ||
41 | mutex_unlock(&edac_pci_ctls_mutex); | ||
42 | } | ||
43 | |||
44 | /* | 34 | /* |
45 | * The alloc() and free() functions for the 'edac_pci' control info | 35 | * edac_pci_alloc_ctl_info |
46 | * structure. The chip driver will allocate one of these for each | 36 | * |
47 | * edac_pci it is going to control/register with the EDAC CORE. | 37 | * The alloc() function for the 'edac_pci' control info |
38 | * structure. The chip driver will allocate one of these for each | ||
39 | * edac_pci it is going to control/register with the EDAC CORE. | ||
48 | */ | 40 | */ |
49 | struct edac_pci_ctl_info *edac_pci_alloc_ctl_info(unsigned int sz_pvt, | 41 | struct edac_pci_ctl_info *edac_pci_alloc_ctl_info(unsigned int sz_pvt, |
50 | const char *edac_pci_name) | 42 | const char *edac_pci_name) |
@@ -53,47 +45,59 @@ struct edac_pci_ctl_info *edac_pci_alloc_ctl_info(unsigned int sz_pvt, | |||
53 | void *pvt; | 45 | void *pvt; |
54 | unsigned int size; | 46 | unsigned int size; |
55 | 47 | ||
48 | debugf1("%s()\n", __func__); | ||
49 | |||
56 | pci = (struct edac_pci_ctl_info *)0; | 50 | pci = (struct edac_pci_ctl_info *)0; |
57 | pvt = edac_align_ptr(&pci[1], sz_pvt); | 51 | pvt = edac_align_ptr(&pci[1], sz_pvt); |
58 | size = ((unsigned long)pvt) + sz_pvt; | 52 | size = ((unsigned long)pvt) + sz_pvt; |
59 | 53 | ||
60 | if ((pci = kzalloc(size, GFP_KERNEL)) == NULL) | 54 | /* Alloc the needed control struct memory */ |
55 | pci = kzalloc(size, GFP_KERNEL); | ||
56 | if (pci == NULL) | ||
61 | return NULL; | 57 | return NULL; |
62 | 58 | ||
59 | /* Now much private space */ | ||
63 | pvt = sz_pvt ? ((char *)pci) + ((unsigned long)pvt) : NULL; | 60 | pvt = sz_pvt ? ((char *)pci) + ((unsigned long)pvt) : NULL; |
64 | 61 | ||
65 | pci->pvt_info = pvt; | 62 | pci->pvt_info = pvt; |
66 | |||
67 | pci->op_state = OP_ALLOC; | 63 | pci->op_state = OP_ALLOC; |
68 | 64 | ||
69 | snprintf(pci->name, strlen(edac_pci_name) + 1, "%s", edac_pci_name); | 65 | snprintf(pci->name, strlen(edac_pci_name) + 1, "%s", edac_pci_name); |
70 | 66 | ||
71 | return pci; | 67 | return pci; |
72 | } | 68 | } |
73 | |||
74 | EXPORT_SYMBOL_GPL(edac_pci_alloc_ctl_info); | 69 | EXPORT_SYMBOL_GPL(edac_pci_alloc_ctl_info); |
75 | 70 | ||
76 | /* | 71 | /* |
77 | * edac_pci_free_ctl_info() | 72 | * edac_pci_free_ctl_info() |
78 | * frees the memory allocated by edac_pci_alloc_ctl_info() function | 73 | * |
74 | * Last action on the pci control structure. | ||
75 | * | ||
76 | * call the remove sysfs informaton, which will unregister | ||
77 | * this control struct's kobj. When that kobj's ref count | ||
78 | * goes to zero, its release function will be call and then | ||
79 | * kfree() the memory. | ||
79 | */ | 80 | */ |
80 | void edac_pci_free_ctl_info(struct edac_pci_ctl_info *pci) | 81 | void edac_pci_free_ctl_info(struct edac_pci_ctl_info *pci) |
81 | { | 82 | { |
82 | kfree(pci); | 83 | debugf1("%s()\n", __func__); |
83 | } | ||
84 | 84 | ||
85 | edac_pci_remove_sysfs(pci); | ||
86 | } | ||
85 | EXPORT_SYMBOL_GPL(edac_pci_free_ctl_info); | 87 | EXPORT_SYMBOL_GPL(edac_pci_free_ctl_info); |
86 | 88 | ||
87 | /* | 89 | /* |
88 | * find_edac_pci_by_dev() | 90 | * find_edac_pci_by_dev() |
89 | * scans the edac_pci list for a specific 'struct device *' | 91 | * scans the edac_pci list for a specific 'struct device *' |
92 | * | ||
93 | * return NULL if not found, or return control struct pointer | ||
90 | */ | 94 | */ |
91 | static struct edac_pci_ctl_info *find_edac_pci_by_dev(struct device *dev) | 95 | static struct edac_pci_ctl_info *find_edac_pci_by_dev(struct device *dev) |
92 | { | 96 | { |
93 | struct edac_pci_ctl_info *pci; | 97 | struct edac_pci_ctl_info *pci; |
94 | struct list_head *item; | 98 | struct list_head *item; |
95 | 99 | ||
96 | debugf3("%s()\n", __func__); | 100 | debugf1("%s()\n", __func__); |
97 | 101 | ||
98 | list_for_each(item, &edac_pci_list) { | 102 | list_for_each(item, &edac_pci_list) { |
99 | pci = list_entry(item, struct edac_pci_ctl_info, link); | 103 | pci = list_entry(item, struct edac_pci_ctl_info, link); |
@@ -118,10 +122,13 @@ static int add_edac_pci_to_global_list(struct edac_pci_ctl_info *pci) | |||
118 | struct list_head *item, *insert_before; | 122 | struct list_head *item, *insert_before; |
119 | struct edac_pci_ctl_info *rover; | 123 | struct edac_pci_ctl_info *rover; |
120 | 124 | ||
125 | debugf1("%s()\n", __func__); | ||
126 | |||
121 | insert_before = &edac_pci_list; | 127 | insert_before = &edac_pci_list; |
122 | 128 | ||
123 | /* Determine if already on the list */ | 129 | /* Determine if already on the list */ |
124 | if (unlikely((rover = find_edac_pci_by_dev(pci->dev)) != NULL)) | 130 | rover = find_edac_pci_by_dev(pci->dev); |
131 | if (unlikely(rover != NULL)) | ||
125 | goto fail0; | 132 | goto fail0; |
126 | 133 | ||
127 | /* Insert in ascending order by 'pci_idx', so find position */ | 134 | /* Insert in ascending order by 'pci_idx', so find position */ |
@@ -157,6 +164,8 @@ fail1: | |||
157 | 164 | ||
158 | /* | 165 | /* |
159 | * complete_edac_pci_list_del | 166 | * complete_edac_pci_list_del |
167 | * | ||
168 | * RCU completion callback to indicate item is deleted | ||
160 | */ | 169 | */ |
161 | static void complete_edac_pci_list_del(struct rcu_head *head) | 170 | static void complete_edac_pci_list_del(struct rcu_head *head) |
162 | { | 171 | { |
@@ -169,6 +178,8 @@ static void complete_edac_pci_list_del(struct rcu_head *head) | |||
169 | 178 | ||
170 | /* | 179 | /* |
171 | * del_edac_pci_from_global_list | 180 | * del_edac_pci_from_global_list |
181 | * | ||
182 | * remove the PCI control struct from the global list | ||
172 | */ | 183 | */ |
173 | static void del_edac_pci_from_global_list(struct edac_pci_ctl_info *pci) | 184 | static void del_edac_pci_from_global_list(struct edac_pci_ctl_info *pci) |
174 | { | 185 | { |
@@ -207,35 +218,52 @@ struct edac_pci_ctl_info *edac_pci_find(int idx) | |||
207 | 218 | ||
208 | return NULL; | 219 | return NULL; |
209 | } | 220 | } |
210 | |||
211 | EXPORT_SYMBOL_GPL(edac_pci_find); | 221 | EXPORT_SYMBOL_GPL(edac_pci_find); |
212 | 222 | ||
213 | /* | 223 | /* |
214 | * edac_pci_workq_function() | 224 | * edac_pci_workq_function() |
215 | * performs the operation scheduled by a workq request | 225 | * |
226 | * periodic function that performs the operation | ||
227 | * scheduled by a workq request, for a given PCI control struct | ||
216 | */ | 228 | */ |
217 | static void edac_pci_workq_function(struct work_struct *work_req) | 229 | static void edac_pci_workq_function(struct work_struct *work_req) |
218 | { | 230 | { |
219 | struct delayed_work *d_work = (struct delayed_work *)work_req; | 231 | struct delayed_work *d_work = (struct delayed_work *)work_req; |
220 | struct edac_pci_ctl_info *pci = to_edac_pci_ctl_work(d_work); | 232 | struct edac_pci_ctl_info *pci = to_edac_pci_ctl_work(d_work); |
233 | int msec; | ||
234 | unsigned long delay; | ||
221 | 235 | ||
222 | edac_lock_pci_list(); | 236 | debugf3("%s() checking\n", __func__); |
223 | 237 | ||
224 | if ((pci->op_state == OP_RUNNING_POLL) && | 238 | mutex_lock(&edac_pci_ctls_mutex); |
225 | (pci->edac_check != NULL) && (edac_pci_get_check_errors())) | ||
226 | pci->edac_check(pci); | ||
227 | 239 | ||
228 | edac_unlock_pci_list(); | 240 | if (pci->op_state == OP_RUNNING_POLL) { |
241 | /* we might be in POLL mode, but there may NOT be a poll func | ||
242 | */ | ||
243 | if ((pci->edac_check != NULL) && edac_pci_get_check_errors()) | ||
244 | pci->edac_check(pci); | ||
245 | |||
246 | /* if we are on a one second period, then use round */ | ||
247 | msec = edac_pci_get_poll_msec(); | ||
248 | if (msec == 1000) | ||
249 | delay = round_jiffies(msecs_to_jiffies(msec)); | ||
250 | else | ||
251 | delay = msecs_to_jiffies(msec); | ||
252 | |||
253 | /* Reschedule only if we are in POLL mode */ | ||
254 | queue_delayed_work(edac_workqueue, &pci->work, delay); | ||
255 | } | ||
229 | 256 | ||
230 | /* Reschedule */ | 257 | mutex_unlock(&edac_pci_ctls_mutex); |
231 | queue_delayed_work(edac_workqueue, &pci->work, | ||
232 | msecs_to_jiffies(edac_pci_get_poll_msec())); | ||
233 | } | 258 | } |
234 | 259 | ||
235 | /* | 260 | /* |
236 | * edac_pci_workq_setup() | 261 | * edac_pci_workq_setup() |
237 | * initialize a workq item for this edac_pci instance | 262 | * initialize a workq item for this edac_pci instance |
238 | * passing in the new delay period in msec | 263 | * passing in the new delay period in msec |
264 | * | ||
265 | * locking model: | ||
266 | * called when 'edac_pci_ctls_mutex' is locked | ||
239 | */ | 267 | */ |
240 | static void edac_pci_workq_setup(struct edac_pci_ctl_info *pci, | 268 | static void edac_pci_workq_setup(struct edac_pci_ctl_info *pci, |
241 | unsigned int msec) | 269 | unsigned int msec) |
@@ -255,6 +283,8 @@ static void edac_pci_workq_teardown(struct edac_pci_ctl_info *pci) | |||
255 | { | 283 | { |
256 | int status; | 284 | int status; |
257 | 285 | ||
286 | debugf0("%s()\n", __func__); | ||
287 | |||
258 | status = cancel_delayed_work(&pci->work); | 288 | status = cancel_delayed_work(&pci->work); |
259 | if (status == 0) | 289 | if (status == 0) |
260 | flush_workqueue(edac_workqueue); | 290 | flush_workqueue(edac_workqueue); |
@@ -262,19 +292,25 @@ static void edac_pci_workq_teardown(struct edac_pci_ctl_info *pci) | |||
262 | 292 | ||
263 | /* | 293 | /* |
264 | * edac_pci_reset_delay_period | 294 | * edac_pci_reset_delay_period |
295 | * | ||
296 | * called with a new period value for the workq period | ||
297 | * a) stop current workq timer | ||
298 | * b) restart workq timer with new value | ||
265 | */ | 299 | */ |
266 | void edac_pci_reset_delay_period(struct edac_pci_ctl_info *pci, | 300 | void edac_pci_reset_delay_period(struct edac_pci_ctl_info *pci, |
267 | unsigned long value) | 301 | unsigned long value) |
268 | { | 302 | { |
269 | edac_lock_pci_list(); | 303 | debugf0("%s()\n", __func__); |
270 | 304 | ||
271 | edac_pci_workq_teardown(pci); | 305 | edac_pci_workq_teardown(pci); |
272 | 306 | ||
307 | /* need to lock for the setup */ | ||
308 | mutex_lock(&edac_pci_ctls_mutex); | ||
309 | |||
273 | edac_pci_workq_setup(pci, value); | 310 | edac_pci_workq_setup(pci, value); |
274 | 311 | ||
275 | edac_unlock_pci_list(); | 312 | mutex_unlock(&edac_pci_ctls_mutex); |
276 | } | 313 | } |
277 | |||
278 | EXPORT_SYMBOL_GPL(edac_pci_reset_delay_period); | 314 | EXPORT_SYMBOL_GPL(edac_pci_reset_delay_period); |
279 | 315 | ||
280 | /* | 316 | /* |
@@ -294,14 +330,13 @@ int edac_pci_add_device(struct edac_pci_ctl_info *pci, int edac_idx) | |||
294 | debugf0("%s()\n", __func__); | 330 | debugf0("%s()\n", __func__); |
295 | 331 | ||
296 | pci->pci_idx = edac_idx; | 332 | pci->pci_idx = edac_idx; |
333 | pci->start_time = jiffies; | ||
297 | 334 | ||
298 | edac_lock_pci_list(); | 335 | mutex_lock(&edac_pci_ctls_mutex); |
299 | 336 | ||
300 | if (add_edac_pci_to_global_list(pci)) | 337 | if (add_edac_pci_to_global_list(pci)) |
301 | goto fail0; | 338 | goto fail0; |
302 | 339 | ||
303 | pci->start_time = jiffies; | ||
304 | |||
305 | if (edac_pci_create_sysfs(pci)) { | 340 | if (edac_pci_create_sysfs(pci)) { |
306 | edac_pci_printk(pci, KERN_WARNING, | 341 | edac_pci_printk(pci, KERN_WARNING, |
307 | "failed to create sysfs pci\n"); | 342 | "failed to create sysfs pci\n"); |
@@ -323,16 +358,16 @@ int edac_pci_add_device(struct edac_pci_ctl_info *pci, int edac_idx) | |||
323 | pci->ctl_name, | 358 | pci->ctl_name, |
324 | dev_name(pci), edac_op_state_to_string(pci->op_state)); | 359 | dev_name(pci), edac_op_state_to_string(pci->op_state)); |
325 | 360 | ||
326 | edac_unlock_pci_list(); | 361 | mutex_unlock(&edac_pci_ctls_mutex); |
327 | return 0; | 362 | return 0; |
328 | 363 | ||
364 | /* error unwind stack */ | ||
329 | fail1: | 365 | fail1: |
330 | del_edac_pci_from_global_list(pci); | 366 | del_edac_pci_from_global_list(pci); |
331 | fail0: | 367 | fail0: |
332 | edac_unlock_pci_list(); | 368 | mutex_unlock(&edac_pci_ctls_mutex); |
333 | return 1; | 369 | return 1; |
334 | } | 370 | } |
335 | |||
336 | EXPORT_SYMBOL_GPL(edac_pci_add_device); | 371 | EXPORT_SYMBOL_GPL(edac_pci_add_device); |
337 | 372 | ||
338 | /* | 373 | /* |
@@ -354,22 +389,25 @@ struct edac_pci_ctl_info *edac_pci_del_device(struct device *dev) | |||
354 | 389 | ||
355 | debugf0("%s()\n", __func__); | 390 | debugf0("%s()\n", __func__); |
356 | 391 | ||
357 | edac_lock_pci_list(); | 392 | mutex_lock(&edac_pci_ctls_mutex); |
358 | 393 | ||
359 | if ((pci = find_edac_pci_by_dev(dev)) == NULL) { | 394 | /* ensure the control struct is on the global list |
360 | edac_unlock_pci_list(); | 395 | * if not, then leave |
396 | */ | ||
397 | pci = find_edac_pci_by_dev(dev); | ||
398 | if (pci == NULL) { | ||
399 | mutex_unlock(&edac_pci_ctls_mutex); | ||
361 | return NULL; | 400 | return NULL; |
362 | } | 401 | } |
363 | 402 | ||
364 | pci->op_state = OP_OFFLINE; | 403 | pci->op_state = OP_OFFLINE; |
365 | 404 | ||
366 | edac_pci_workq_teardown(pci); | ||
367 | |||
368 | edac_pci_remove_sysfs(pci); | ||
369 | |||
370 | del_edac_pci_from_global_list(pci); | 405 | del_edac_pci_from_global_list(pci); |
371 | 406 | ||
372 | edac_unlock_pci_list(); | 407 | mutex_unlock(&edac_pci_ctls_mutex); |
408 | |||
409 | /* stop the workq timer */ | ||
410 | edac_pci_workq_teardown(pci); | ||
373 | 411 | ||
374 | edac_printk(KERN_INFO, EDAC_PCI, | 412 | edac_printk(KERN_INFO, EDAC_PCI, |
375 | "Removed device %d for %s %s: DEV %s\n", | 413 | "Removed device %d for %s %s: DEV %s\n", |
@@ -377,14 +415,20 @@ struct edac_pci_ctl_info *edac_pci_del_device(struct device *dev) | |||
377 | 415 | ||
378 | return pci; | 416 | return pci; |
379 | } | 417 | } |
380 | |||
381 | EXPORT_SYMBOL_GPL(edac_pci_del_device); | 418 | EXPORT_SYMBOL_GPL(edac_pci_del_device); |
382 | 419 | ||
420 | /* | ||
421 | * edac_pci_generic_check | ||
422 | * | ||
423 | * a Generic parity check API | ||
424 | */ | ||
383 | void edac_pci_generic_check(struct edac_pci_ctl_info *pci) | 425 | void edac_pci_generic_check(struct edac_pci_ctl_info *pci) |
384 | { | 426 | { |
427 | debugf4("%s()\n", __func__); | ||
385 | edac_pci_do_parity_check(); | 428 | edac_pci_do_parity_check(); |
386 | } | 429 | } |
387 | 430 | ||
431 | /* free running instance index counter */ | ||
388 | static int edac_pci_idx; | 432 | static int edac_pci_idx; |
389 | #define EDAC_PCI_GENCTL_NAME "EDAC PCI controller" | 433 | #define EDAC_PCI_GENCTL_NAME "EDAC PCI controller" |
390 | 434 | ||
@@ -392,6 +436,17 @@ struct edac_pci_gen_data { | |||
392 | int edac_idx; | 436 | int edac_idx; |
393 | }; | 437 | }; |
394 | 438 | ||
439 | /* | ||
440 | * edac_pci_create_generic_ctl | ||
441 | * | ||
442 | * A generic constructor for a PCI parity polling device | ||
443 | * Some systems have more than one domain of PCI busses. | ||
444 | * For systems with one domain, then this API will | ||
445 | * provide for a generic poller. | ||
446 | * | ||
447 | * This routine calls the edac_pci_alloc_ctl_info() for | ||
448 | * the generic device, with default values | ||
449 | */ | ||
395 | struct edac_pci_ctl_info *edac_pci_create_generic_ctl(struct device *dev, | 450 | struct edac_pci_ctl_info *edac_pci_create_generic_ctl(struct device *dev, |
396 | const char *mod_name) | 451 | const char *mod_name) |
397 | { | 452 | { |
@@ -421,13 +476,18 @@ struct edac_pci_ctl_info *edac_pci_create_generic_ctl(struct device *dev, | |||
421 | 476 | ||
422 | return pci; | 477 | return pci; |
423 | } | 478 | } |
424 | |||
425 | EXPORT_SYMBOL_GPL(edac_pci_create_generic_ctl); | 479 | EXPORT_SYMBOL_GPL(edac_pci_create_generic_ctl); |
426 | 480 | ||
481 | /* | ||
482 | * edac_pci_release_generic_ctl | ||
483 | * | ||
484 | * The release function of a generic EDAC PCI polling device | ||
485 | */ | ||
427 | void edac_pci_release_generic_ctl(struct edac_pci_ctl_info *pci) | 486 | void edac_pci_release_generic_ctl(struct edac_pci_ctl_info *pci) |
428 | { | 487 | { |
488 | debugf0("%s() pci mod=%s\n", __func__, pci->mod_name); | ||
489 | |||
429 | edac_pci_del_device(pci->dev); | 490 | edac_pci_del_device(pci->dev); |
430 | edac_pci_free_ctl_info(pci); | 491 | edac_pci_free_ctl_info(pci); |
431 | } | 492 | } |
432 | |||
433 | EXPORT_SYMBOL_GPL(edac_pci_release_generic_ctl); | 493 | EXPORT_SYMBOL_GPL(edac_pci_release_generic_ctl); |
diff --git a/drivers/edac/edac_pci_sysfs.c b/drivers/edac/edac_pci_sysfs.c index fac94cae2c3d..69f5dddabddf 100644 --- a/drivers/edac/edac_pci_sysfs.c +++ b/drivers/edac/edac_pci_sysfs.c | |||
@@ -13,22 +13,25 @@ | |||
13 | #include "edac_core.h" | 13 | #include "edac_core.h" |
14 | #include "edac_module.h" | 14 | #include "edac_module.h" |
15 | 15 | ||
16 | /* Turn off this whole feature if PCI is not configured */ | ||
16 | #ifdef CONFIG_PCI | 17 | #ifdef CONFIG_PCI |
17 | 18 | ||
18 | #define EDAC_PCI_SYMLINK "device" | 19 | #define EDAC_PCI_SYMLINK "device" |
19 | 20 | ||
20 | static int check_pci_errors; /* default YES check PCI parity */ | 21 | /* data variables exported via sysfs */ |
21 | static int edac_pci_panic_on_pe; /* default no panic on PCI Parity */ | 22 | static int check_pci_errors; /* default NO check PCI parity */ |
22 | static int edac_pci_log_pe = 1; /* log PCI parity errors */ | 23 | static int edac_pci_panic_on_pe; /* default NO panic on PCI Parity */ |
24 | static int edac_pci_log_pe = 1; /* log PCI parity errors */ | ||
23 | static int edac_pci_log_npe = 1; /* log PCI non-parity error errors */ | 25 | static int edac_pci_log_npe = 1; /* log PCI non-parity error errors */ |
26 | static int edac_pci_poll_msec = 1000; /* one second workq period */ | ||
27 | |||
24 | static atomic_t pci_parity_count = ATOMIC_INIT(0); | 28 | static atomic_t pci_parity_count = ATOMIC_INIT(0); |
25 | static atomic_t pci_nonparity_count = ATOMIC_INIT(0); | 29 | static atomic_t pci_nonparity_count = ATOMIC_INIT(0); |
26 | static int edac_pci_poll_msec = 1000; | ||
27 | 30 | ||
28 | static struct kobject edac_pci_kobj; /* /sys/devices/system/edac/pci */ | 31 | static struct kobject edac_pci_top_main_kobj; |
29 | static struct completion edac_pci_kobj_complete; | ||
30 | static atomic_t edac_pci_sysfs_refcount = ATOMIC_INIT(0); | 32 | static atomic_t edac_pci_sysfs_refcount = ATOMIC_INIT(0); |
31 | 33 | ||
34 | /* getter functions for the data variables */ | ||
32 | int edac_pci_get_check_errors(void) | 35 | int edac_pci_get_check_errors(void) |
33 | { | 36 | { |
34 | return check_pci_errors; | 37 | return check_pci_errors; |
@@ -74,17 +77,22 @@ static void edac_pci_instance_release(struct kobject *kobj) | |||
74 | { | 77 | { |
75 | struct edac_pci_ctl_info *pci; | 78 | struct edac_pci_ctl_info *pci; |
76 | 79 | ||
77 | debugf1("%s()\n", __func__); | 80 | debugf0("%s()\n", __func__); |
78 | 81 | ||
82 | /* Form pointer to containing struct, the pci control struct */ | ||
79 | pci = to_instance(kobj); | 83 | pci = to_instance(kobj); |
80 | complete(&pci->kobj_complete); | 84 | |
85 | /* decrement reference count on top main kobj */ | ||
86 | kobject_put(&edac_pci_top_main_kobj); | ||
87 | |||
88 | kfree(pci); /* Free the control struct */ | ||
81 | } | 89 | } |
82 | 90 | ||
83 | /* instance specific attribute structure */ | 91 | /* instance specific attribute structure */ |
84 | struct instance_attribute { | 92 | struct instance_attribute { |
85 | struct attribute attr; | 93 | struct attribute attr; |
86 | ssize_t(*show) (struct edac_pci_ctl_info *, char *); | 94 | ssize_t(*show) (struct edac_pci_ctl_info *, char *); |
87 | ssize_t(*store) (struct edac_pci_ctl_info *, const char *, size_t); | 95 | ssize_t(*store) (struct edac_pci_ctl_info *, const char *, size_t); |
88 | }; | 96 | }; |
89 | 97 | ||
90 | /* Function to 'show' fields from the edac_pci 'instance' structure */ | 98 | /* Function to 'show' fields from the edac_pci 'instance' structure */ |
@@ -112,6 +120,7 @@ static ssize_t edac_pci_instance_store(struct kobject *kobj, | |||
112 | return -EIO; | 120 | return -EIO; |
113 | } | 121 | } |
114 | 122 | ||
123 | /* fs_ops table */ | ||
115 | static struct sysfs_ops pci_instance_ops = { | 124 | static struct sysfs_ops pci_instance_ops = { |
116 | .show = edac_pci_instance_show, | 125 | .show = edac_pci_instance_show, |
117 | .store = edac_pci_instance_store | 126 | .store = edac_pci_instance_store |
@@ -134,48 +143,82 @@ static struct instance_attribute *pci_instance_attr[] = { | |||
134 | NULL | 143 | NULL |
135 | }; | 144 | }; |
136 | 145 | ||
137 | /* the ktype for pci instance */ | 146 | /* the ktype for a pci instance */ |
138 | static struct kobj_type ktype_pci_instance = { | 147 | static struct kobj_type ktype_pci_instance = { |
139 | .release = edac_pci_instance_release, | 148 | .release = edac_pci_instance_release, |
140 | .sysfs_ops = &pci_instance_ops, | 149 | .sysfs_ops = &pci_instance_ops, |
141 | .default_attrs = (struct attribute **)pci_instance_attr, | 150 | .default_attrs = (struct attribute **)pci_instance_attr, |
142 | }; | 151 | }; |
143 | 152 | ||
153 | /* | ||
154 | * edac_pci_create_instance_kobj | ||
155 | * | ||
156 | * construct one EDAC PCI instance's kobject for use | ||
157 | */ | ||
144 | static int edac_pci_create_instance_kobj(struct edac_pci_ctl_info *pci, int idx) | 158 | static int edac_pci_create_instance_kobj(struct edac_pci_ctl_info *pci, int idx) |
145 | { | 159 | { |
160 | struct kobject *main_kobj; | ||
146 | int err; | 161 | int err; |
147 | 162 | ||
148 | pci->kobj.parent = &edac_pci_kobj; | 163 | debugf0("%s()\n", __func__); |
164 | |||
165 | /* Set the parent and the instance's ktype */ | ||
166 | pci->kobj.parent = &edac_pci_top_main_kobj; | ||
149 | pci->kobj.ktype = &ktype_pci_instance; | 167 | pci->kobj.ktype = &ktype_pci_instance; |
150 | 168 | ||
151 | err = kobject_set_name(&pci->kobj, "pci%d", idx); | 169 | err = kobject_set_name(&pci->kobj, "pci%d", idx); |
152 | if (err) | 170 | if (err) |
153 | return err; | 171 | return err; |
154 | 172 | ||
173 | /* First bump the ref count on the top main kobj, which will | ||
174 | * track the number of PCI instances we have, and thus nest | ||
175 | * properly on keeping the module loaded | ||
176 | */ | ||
177 | main_kobj = kobject_get(&edac_pci_top_main_kobj); | ||
178 | if (!main_kobj) { | ||
179 | err = -ENODEV; | ||
180 | goto error_out; | ||
181 | } | ||
182 | |||
183 | /* And now register this new kobject under the main kobj */ | ||
155 | err = kobject_register(&pci->kobj); | 184 | err = kobject_register(&pci->kobj); |
156 | if (err != 0) { | 185 | if (err != 0) { |
157 | debugf2("%s() failed to register instance pci%d\n", | 186 | debugf2("%s() failed to register instance pci%d\n", |
158 | __func__, idx); | 187 | __func__, idx); |
159 | return err; | 188 | kobject_put(&edac_pci_top_main_kobj); |
189 | goto error_out; | ||
160 | } | 190 | } |
161 | 191 | ||
162 | debugf1("%s() Register instance 'pci%d' kobject\n", __func__, idx); | 192 | debugf1("%s() Register instance 'pci%d' kobject\n", __func__, idx); |
163 | 193 | ||
164 | return 0; | 194 | return 0; |
195 | |||
196 | /* Error unwind statck */ | ||
197 | error_out: | ||
198 | return err; | ||
165 | } | 199 | } |
166 | 200 | ||
167 | static void | 201 | /* |
168 | edac_pci_delete_instance_kobj(struct edac_pci_ctl_info *pci, int idx) | 202 | * edac_pci_unregister_sysfs_instance_kobj |
203 | * | ||
204 | * unregister the kobj for the EDAC PCI instance | ||
205 | */ | ||
206 | void edac_pci_unregister_sysfs_instance_kobj(struct edac_pci_ctl_info *pci) | ||
169 | { | 207 | { |
170 | init_completion(&pci->kobj_complete); | 208 | debugf0("%s()\n", __func__); |
209 | |||
210 | /* Unregister the instance kobject and allow its release | ||
211 | * function release the main reference count and then | ||
212 | * kfree the memory | ||
213 | */ | ||
171 | kobject_unregister(&pci->kobj); | 214 | kobject_unregister(&pci->kobj); |
172 | wait_for_completion(&pci->kobj_complete); | ||
173 | } | 215 | } |
174 | 216 | ||
175 | /***************************** EDAC PCI sysfs root **********************/ | 217 | /***************************** EDAC PCI sysfs root **********************/ |
176 | #define to_edacpci(k) container_of(k, struct edac_pci_ctl_info, kobj) | 218 | #define to_edacpci(k) container_of(k, struct edac_pci_ctl_info, kobj) |
177 | #define to_edacpci_attr(a) container_of(a, struct edac_pci_attr, attr) | 219 | #define to_edacpci_attr(a) container_of(a, struct edac_pci_attr, attr) |
178 | 220 | ||
221 | /* simple show/store functions for attributes */ | ||
179 | static ssize_t edac_pci_int_show(void *ptr, char *buffer) | 222 | static ssize_t edac_pci_int_show(void *ptr, char *buffer) |
180 | { | 223 | { |
181 | int *value = ptr; | 224 | int *value = ptr; |
@@ -267,118 +310,189 @@ static struct edac_pci_dev_attribute *edac_pci_attr[] = { | |||
267 | NULL, | 310 | NULL, |
268 | }; | 311 | }; |
269 | 312 | ||
270 | /* No memory to release */ | 313 | /* |
271 | static void edac_pci_release(struct kobject *kobj) | 314 | * edac_pci_release_main_kobj |
315 | * | ||
316 | * This release function is called when the reference count to the | ||
317 | * passed kobj goes to zero. | ||
318 | * | ||
319 | * This kobj is the 'main' kobject that EDAC PCI instances | ||
320 | * link to, and thus provide for proper nesting counts | ||
321 | */ | ||
322 | static void edac_pci_release_main_kobj(struct kobject *kobj) | ||
272 | { | 323 | { |
273 | struct edac_pci_ctl_info *pci; | ||
274 | 324 | ||
275 | pci = to_edacpci(kobj); | 325 | debugf0("%s() here to module_put(THIS_MODULE)\n", __func__); |
276 | 326 | ||
277 | debugf1("%s()\n", __func__); | 327 | /* last reference to top EDAC PCI kobject has been removed, |
278 | complete(&pci->kobj_complete); | 328 | * NOW release our ref count on the core module |
329 | */ | ||
330 | module_put(THIS_MODULE); | ||
279 | } | 331 | } |
280 | 332 | ||
281 | static struct kobj_type ktype_edac_pci = { | 333 | /* ktype struct for the EDAC PCI main kobj */ |
282 | .release = edac_pci_release, | 334 | static struct kobj_type ktype_edac_pci_main_kobj = { |
335 | .release = edac_pci_release_main_kobj, | ||
283 | .sysfs_ops = &edac_pci_sysfs_ops, | 336 | .sysfs_ops = &edac_pci_sysfs_ops, |
284 | .default_attrs = (struct attribute **)edac_pci_attr, | 337 | .default_attrs = (struct attribute **)edac_pci_attr, |
285 | }; | 338 | }; |
286 | 339 | ||
287 | /** | 340 | /** |
288 | * edac_sysfs_pci_setup() | 341 | * edac_pci_main_kobj_setup() |
289 | * | 342 | * |
290 | * setup the sysfs for EDAC PCI attributes | 343 | * setup the sysfs for EDAC PCI attributes |
291 | * assumes edac_class has already been initialized | 344 | * assumes edac_class has already been initialized |
292 | */ | 345 | */ |
293 | int edac_pci_register_main_kobj(void) | 346 | int edac_pci_main_kobj_setup(void) |
294 | { | 347 | { |
295 | int err; | 348 | int err; |
296 | struct sysdev_class *edac_class; | 349 | struct sysdev_class *edac_class; |
297 | 350 | ||
298 | debugf1("%s()\n", __func__); | 351 | debugf0("%s()\n", __func__); |
352 | |||
353 | /* check and count if we have already created the main kobject */ | ||
354 | if (atomic_inc_return(&edac_pci_sysfs_refcount) != 1) | ||
355 | return 0; | ||
299 | 356 | ||
357 | /* First time, so create the main kobject and its | ||
358 | * controls and atributes | ||
359 | */ | ||
300 | edac_class = edac_get_edac_class(); | 360 | edac_class = edac_get_edac_class(); |
301 | if (edac_class == NULL) { | 361 | if (edac_class == NULL) { |
302 | debugf1("%s() no edac_class\n", __func__); | 362 | debugf1("%s() no edac_class\n", __func__); |
303 | return -ENODEV; | 363 | err = -ENODEV; |
364 | goto decrement_count_fail; | ||
304 | } | 365 | } |
305 | 366 | ||
306 | edac_pci_kobj.ktype = &ktype_edac_pci; | 367 | /* Need the kobject hook ups, and name setting */ |
368 | edac_pci_top_main_kobj.ktype = &ktype_edac_pci_main_kobj; | ||
369 | edac_pci_top_main_kobj.parent = &edac_class->kset.kobj; | ||
307 | 370 | ||
308 | edac_pci_kobj.parent = &edac_class->kset.kobj; | 371 | err = kobject_set_name(&edac_pci_top_main_kobj, "pci"); |
309 | |||
310 | err = kobject_set_name(&edac_pci_kobj, "pci"); | ||
311 | if (err) | 372 | if (err) |
312 | return err; | 373 | goto decrement_count_fail; |
374 | |||
375 | /* Bump the reference count on this module to ensure the | ||
376 | * modules isn't unloaded until we deconstruct the top | ||
377 | * level main kobj for EDAC PCI | ||
378 | */ | ||
379 | if (!try_module_get(THIS_MODULE)) { | ||
380 | debugf1("%s() try_module_get() failed\n", __func__); | ||
381 | err = -ENODEV; | ||
382 | goto decrement_count_fail; | ||
383 | } | ||
313 | 384 | ||
314 | /* Instanstiate the pci object */ | 385 | /* Instanstiate the pci object */ |
315 | /* FIXME: maybe new sysdev_create_subdir() */ | 386 | /* FIXME: maybe new sysdev_create_subdir() */ |
316 | err = kobject_register(&edac_pci_kobj); | 387 | err = kobject_register(&edac_pci_top_main_kobj); |
317 | |||
318 | if (err) { | 388 | if (err) { |
319 | debugf1("Failed to register '.../edac/pci'\n"); | 389 | debugf1("Failed to register '.../edac/pci'\n"); |
320 | return err; | 390 | goto kobject_register_fail; |
321 | } | 391 | } |
322 | 392 | ||
393 | /* At this point, to 'release' the top level kobject | ||
394 | * for EDAC PCI, then edac_pci_main_kobj_teardown() | ||
395 | * must be used, for resources to be cleaned up properly | ||
396 | */ | ||
323 | debugf1("Registered '.../edac/pci' kobject\n"); | 397 | debugf1("Registered '.../edac/pci' kobject\n"); |
324 | 398 | ||
325 | return 0; | 399 | return 0; |
400 | |||
401 | /* Error unwind statck */ | ||
402 | kobject_register_fail: | ||
403 | module_put(THIS_MODULE); | ||
404 | |||
405 | decrement_count_fail: | ||
406 | /* if are on this error exit, nothing to tear down */ | ||
407 | atomic_dec(&edac_pci_sysfs_refcount); | ||
408 | |||
409 | return err; | ||
326 | } | 410 | } |
327 | 411 | ||
328 | /* | 412 | /* |
329 | * edac_pci_unregister_main_kobj() | 413 | * edac_pci_main_kobj_teardown() |
330 | * | 414 | * |
331 | * perform the sysfs teardown for the PCI attributes | 415 | * if no longer linked (needed) remove the top level EDAC PCI |
416 | * kobject with its controls and attributes | ||
332 | */ | 417 | */ |
333 | void edac_pci_unregister_main_kobj(void) | 418 | static void edac_pci_main_kobj_teardown(void) |
334 | { | 419 | { |
335 | debugf0("%s()\n", __func__); | 420 | debugf0("%s()\n", __func__); |
336 | init_completion(&edac_pci_kobj_complete); | 421 | |
337 | kobject_unregister(&edac_pci_kobj); | 422 | /* Decrement the count and only if no more controller instances |
338 | wait_for_completion(&edac_pci_kobj_complete); | 423 | * are connected perform the unregisteration of the top level |
424 | * main kobj | ||
425 | */ | ||
426 | if (atomic_dec_return(&edac_pci_sysfs_refcount) == 0) { | ||
427 | debugf0("%s() called kobject_unregister on main kobj\n", | ||
428 | __func__); | ||
429 | kobject_unregister(&edac_pci_top_main_kobj); | ||
430 | } | ||
339 | } | 431 | } |
340 | 432 | ||
433 | /* | ||
434 | * | ||
435 | * edac_pci_create_sysfs | ||
436 | * | ||
437 | * Create the controls/attributes for the specified EDAC PCI device | ||
438 | */ | ||
341 | int edac_pci_create_sysfs(struct edac_pci_ctl_info *pci) | 439 | int edac_pci_create_sysfs(struct edac_pci_ctl_info *pci) |
342 | { | 440 | { |
343 | int err; | 441 | int err; |
344 | struct kobject *edac_kobj = &pci->kobj; | 442 | struct kobject *edac_kobj = &pci->kobj; |
345 | 443 | ||
346 | if (atomic_inc_return(&edac_pci_sysfs_refcount) == 1) { | 444 | debugf0("%s() idx=%d\n", __func__, pci->pci_idx); |
347 | err = edac_pci_register_main_kobj(); | ||
348 | if (err) { | ||
349 | atomic_dec(&edac_pci_sysfs_refcount); | ||
350 | return err; | ||
351 | } | ||
352 | } | ||
353 | 445 | ||
354 | err = edac_pci_create_instance_kobj(pci, pci->pci_idx); | 446 | /* create the top main EDAC PCI kobject, IF needed */ |
355 | if (err) { | 447 | err = edac_pci_main_kobj_setup(); |
356 | if (atomic_dec_return(&edac_pci_sysfs_refcount) == 0) | 448 | if (err) |
357 | edac_pci_unregister_main_kobj(); | 449 | return err; |
358 | } | ||
359 | 450 | ||
360 | debugf0("%s() idx=%d\n", __func__, pci->pci_idx); | 451 | /* Create this instance's kobject under the MAIN kobject */ |
452 | err = edac_pci_create_instance_kobj(pci, pci->pci_idx); | ||
453 | if (err) | ||
454 | goto unregister_cleanup; | ||
361 | 455 | ||
362 | err = sysfs_create_link(edac_kobj, &pci->dev->kobj, EDAC_PCI_SYMLINK); | 456 | err = sysfs_create_link(edac_kobj, &pci->dev->kobj, EDAC_PCI_SYMLINK); |
363 | if (err) { | 457 | if (err) { |
364 | debugf0("%s() sysfs_create_link() returned err= %d\n", | 458 | debugf0("%s() sysfs_create_link() returned err= %d\n", |
365 | __func__, err); | 459 | __func__, err); |
366 | return err; | 460 | goto symlink_fail; |
367 | } | 461 | } |
368 | 462 | ||
369 | return 0; | 463 | return 0; |
464 | |||
465 | /* Error unwind stack */ | ||
466 | symlink_fail: | ||
467 | edac_pci_unregister_sysfs_instance_kobj(pci); | ||
468 | |||
469 | unregister_cleanup: | ||
470 | edac_pci_main_kobj_teardown(); | ||
471 | |||
472 | return err; | ||
370 | } | 473 | } |
371 | 474 | ||
475 | /* | ||
476 | * edac_pci_remove_sysfs | ||
477 | * | ||
478 | * remove the controls and attributes for this EDAC PCI device | ||
479 | */ | ||
372 | void edac_pci_remove_sysfs(struct edac_pci_ctl_info *pci) | 480 | void edac_pci_remove_sysfs(struct edac_pci_ctl_info *pci) |
373 | { | 481 | { |
374 | debugf0("%s()\n", __func__); | 482 | debugf0("%s() index=%d\n", __func__, pci->pci_idx); |
375 | |||
376 | edac_pci_delete_instance_kobj(pci, pci->pci_idx); | ||
377 | 483 | ||
484 | /* Remove the symlink */ | ||
378 | sysfs_remove_link(&pci->kobj, EDAC_PCI_SYMLINK); | 485 | sysfs_remove_link(&pci->kobj, EDAC_PCI_SYMLINK); |
379 | 486 | ||
380 | if (atomic_dec_return(&edac_pci_sysfs_refcount) == 0) | 487 | /* remove this PCI instance's sysfs entries */ |
381 | edac_pci_unregister_main_kobj(); | 488 | edac_pci_unregister_sysfs_instance_kobj(pci); |
489 | |||
490 | /* Call the main unregister function, which will determine | ||
491 | * if this 'pci' is the last instance. | ||
492 | * If it is, the main kobject will be unregistered as a result | ||
493 | */ | ||
494 | debugf0("%s() calling edac_pci_main_kobj_teardown()\n", __func__); | ||
495 | edac_pci_main_kobj_teardown(); | ||
382 | } | 496 | } |
383 | 497 | ||
384 | /************************ PCI error handling *************************/ | 498 | /************************ PCI error handling *************************/ |
@@ -414,13 +528,14 @@ static u16 get_pci_parity_status(struct pci_dev *dev, int secondary) | |||
414 | return status; | 528 | return status; |
415 | } | 529 | } |
416 | 530 | ||
417 | typedef void (*pci_parity_check_fn_t) (struct pci_dev * dev); | ||
418 | 531 | ||
419 | /* Clear any PCI parity errors logged by this device. */ | 532 | /* Clear any PCI parity errors logged by this device. */ |
420 | static void edac_pci_dev_parity_clear(struct pci_dev *dev) | 533 | static void edac_pci_dev_parity_clear(struct pci_dev *dev) |
421 | { | 534 | { |
422 | u8 header_type; | 535 | u8 header_type; |
423 | 536 | ||
537 | debugf0("%s()\n", __func__); | ||
538 | |||
424 | get_pci_parity_status(dev, 0); | 539 | get_pci_parity_status(dev, 0); |
425 | 540 | ||
426 | /* read the device TYPE, looking for bridges */ | 541 | /* read the device TYPE, looking for bridges */ |
@@ -433,17 +548,28 @@ static void edac_pci_dev_parity_clear(struct pci_dev *dev) | |||
433 | /* | 548 | /* |
434 | * PCI Parity polling | 549 | * PCI Parity polling |
435 | * | 550 | * |
551 | * Fucntion to retrieve the current parity status | ||
552 | * and decode it | ||
553 | * | ||
436 | */ | 554 | */ |
437 | static void edac_pci_dev_parity_test(struct pci_dev *dev) | 555 | static void edac_pci_dev_parity_test(struct pci_dev *dev) |
438 | { | 556 | { |
557 | unsigned long flags; | ||
439 | u16 status; | 558 | u16 status; |
440 | u8 header_type; | 559 | u8 header_type; |
441 | 560 | ||
442 | /* read the STATUS register on this device | 561 | /* stop any interrupts until we can acquire the status */ |
443 | */ | 562 | local_irq_save(flags); |
563 | |||
564 | /* read the STATUS register on this device */ | ||
444 | status = get_pci_parity_status(dev, 0); | 565 | status = get_pci_parity_status(dev, 0); |
445 | 566 | ||
446 | debugf2("PCI STATUS= 0x%04x %s\n", status, dev->dev.bus_id); | 567 | /* read the device TYPE, looking for bridges */ |
568 | pci_read_config_byte(dev, PCI_HEADER_TYPE, &header_type); | ||
569 | |||
570 | local_irq_restore(flags); | ||
571 | |||
572 | debugf4("PCI STATUS= 0x%04x %s\n", status, dev->dev.bus_id); | ||
447 | 573 | ||
448 | /* check the status reg for errors */ | 574 | /* check the status reg for errors */ |
449 | if (status) { | 575 | if (status) { |
@@ -471,16 +597,14 @@ static void edac_pci_dev_parity_test(struct pci_dev *dev) | |||
471 | } | 597 | } |
472 | } | 598 | } |
473 | 599 | ||
474 | /* read the device TYPE, looking for bridges */ | ||
475 | pci_read_config_byte(dev, PCI_HEADER_TYPE, &header_type); | ||
476 | 600 | ||
477 | debugf2("PCI HEADER TYPE= 0x%02x %s\n", header_type, dev->dev.bus_id); | 601 | debugf4("PCI HEADER TYPE= 0x%02x %s\n", header_type, dev->dev.bus_id); |
478 | 602 | ||
479 | if ((header_type & 0x7F) == PCI_HEADER_TYPE_BRIDGE) { | 603 | if ((header_type & 0x7F) == PCI_HEADER_TYPE_BRIDGE) { |
480 | /* On bridges, need to examine secondary status register */ | 604 | /* On bridges, need to examine secondary status register */ |
481 | status = get_pci_parity_status(dev, 1); | 605 | status = get_pci_parity_status(dev, 1); |
482 | 606 | ||
483 | debugf2("PCI SEC_STATUS= 0x%04x %s\n", status, dev->dev.bus_id); | 607 | debugf4("PCI SEC_STATUS= 0x%04x %s\n", status, dev->dev.bus_id); |
484 | 608 | ||
485 | /* check the secondary status reg for errors */ | 609 | /* check the secondary status reg for errors */ |
486 | if (status) { | 610 | if (status) { |
@@ -510,9 +634,12 @@ static void edac_pci_dev_parity_test(struct pci_dev *dev) | |||
510 | } | 634 | } |
511 | } | 635 | } |
512 | 636 | ||
637 | /* reduce some complexity in definition of the iterator */ | ||
638 | typedef void (*pci_parity_check_fn_t) (struct pci_dev *dev); | ||
639 | |||
513 | /* | 640 | /* |
514 | * pci_dev parity list iterator | 641 | * pci_dev parity list iterator |
515 | * Scan the PCI device list for one iteration, looking for SERRORs | 642 | * Scan the PCI device list for one pass, looking for SERRORs |
516 | * Master Parity ERRORS or Parity ERRORs on primary or secondary devices | 643 | * Master Parity ERRORS or Parity ERRORs on primary or secondary devices |
517 | */ | 644 | */ |
518 | static inline void edac_pci_dev_parity_iterator(pci_parity_check_fn_t fn) | 645 | static inline void edac_pci_dev_parity_iterator(pci_parity_check_fn_t fn) |
@@ -535,22 +662,22 @@ static inline void edac_pci_dev_parity_iterator(pci_parity_check_fn_t fn) | |||
535 | */ | 662 | */ |
536 | void edac_pci_do_parity_check(void) | 663 | void edac_pci_do_parity_check(void) |
537 | { | 664 | { |
538 | unsigned long flags; | ||
539 | int before_count; | 665 | int before_count; |
540 | 666 | ||
541 | debugf3("%s()\n", __func__); | 667 | debugf3("%s()\n", __func__); |
542 | 668 | ||
669 | /* if policy has PCI check off, leave now */ | ||
543 | if (!check_pci_errors) | 670 | if (!check_pci_errors) |
544 | return; | 671 | return; |
545 | 672 | ||
546 | before_count = atomic_read(&pci_parity_count); | 673 | before_count = atomic_read(&pci_parity_count); |
547 | 674 | ||
548 | /* scan all PCI devices looking for a Parity Error on devices and | 675 | /* scan all PCI devices looking for a Parity Error on devices and |
549 | * bridges | 676 | * bridges. |
677 | * The iterator calls pci_get_device() which might sleep, thus | ||
678 | * we cannot disable interrupts in this scan. | ||
550 | */ | 679 | */ |
551 | local_irq_save(flags); | ||
552 | edac_pci_dev_parity_iterator(edac_pci_dev_parity_test); | 680 | edac_pci_dev_parity_iterator(edac_pci_dev_parity_test); |
553 | local_irq_restore(flags); | ||
554 | 681 | ||
555 | /* Only if operator has selected panic on PCI Error */ | 682 | /* Only if operator has selected panic on PCI Error */ |
556 | if (edac_pci_get_panic_on_pe()) { | 683 | if (edac_pci_get_panic_on_pe()) { |
@@ -560,6 +687,12 @@ void edac_pci_do_parity_check(void) | |||
560 | } | 687 | } |
561 | } | 688 | } |
562 | 689 | ||
690 | /* | ||
691 | * edac_pci_clear_parity_errors | ||
692 | * | ||
693 | * function to perform an iteration over the PCI devices | ||
694 | * and clearn their current status | ||
695 | */ | ||
563 | void edac_pci_clear_parity_errors(void) | 696 | void edac_pci_clear_parity_errors(void) |
564 | { | 697 | { |
565 | /* Clear any PCI bus parity errors that devices initially have logged | 698 | /* Clear any PCI bus parity errors that devices initially have logged |
@@ -567,6 +700,12 @@ void edac_pci_clear_parity_errors(void) | |||
567 | */ | 700 | */ |
568 | edac_pci_dev_parity_iterator(edac_pci_dev_parity_clear); | 701 | edac_pci_dev_parity_iterator(edac_pci_dev_parity_clear); |
569 | } | 702 | } |
703 | |||
704 | /* | ||
705 | * edac_pci_handle_pe | ||
706 | * | ||
707 | * Called to handle a PARITY ERROR event | ||
708 | */ | ||
570 | void edac_pci_handle_pe(struct edac_pci_ctl_info *pci, const char *msg) | 709 | void edac_pci_handle_pe(struct edac_pci_ctl_info *pci, const char *msg) |
571 | { | 710 | { |
572 | 711 | ||
@@ -584,9 +723,14 @@ void edac_pci_handle_pe(struct edac_pci_ctl_info *pci, const char *msg) | |||
584 | */ | 723 | */ |
585 | edac_pci_do_parity_check(); | 724 | edac_pci_do_parity_check(); |
586 | } | 725 | } |
587 | |||
588 | EXPORT_SYMBOL_GPL(edac_pci_handle_pe); | 726 | EXPORT_SYMBOL_GPL(edac_pci_handle_pe); |
589 | 727 | ||
728 | |||
729 | /* | ||
730 | * edac_pci_handle_npe | ||
731 | * | ||
732 | * Called to handle a NON-PARITY ERROR event | ||
733 | */ | ||
590 | void edac_pci_handle_npe(struct edac_pci_ctl_info *pci, const char *msg) | 734 | void edac_pci_handle_npe(struct edac_pci_ctl_info *pci, const char *msg) |
591 | { | 735 | { |
592 | 736 | ||
@@ -604,7 +748,6 @@ void edac_pci_handle_npe(struct edac_pci_ctl_info *pci, const char *msg) | |||
604 | */ | 748 | */ |
605 | edac_pci_do_parity_check(); | 749 | edac_pci_do_parity_check(); |
606 | } | 750 | } |
607 | |||
608 | EXPORT_SYMBOL_GPL(edac_pci_handle_npe); | 751 | EXPORT_SYMBOL_GPL(edac_pci_handle_npe); |
609 | 752 | ||
610 | /* | 753 | /* |
diff --git a/drivers/edac/i3000_edac.c b/drivers/edac/i3000_edac.c index 0ecfdc432f87..e895f9f887ab 100644 --- a/drivers/edac/i3000_edac.c +++ b/drivers/edac/i3000_edac.c | |||
@@ -275,7 +275,7 @@ static int i3000_probe1(struct pci_dev *pdev, int dev_idx) | |||
275 | unsigned char *c0dra = dra, *c1dra = &dra[I3000_RANKS_PER_CHANNEL / 2]; | 275 | unsigned char *c0dra = dra, *c1dra = &dra[I3000_RANKS_PER_CHANNEL / 2]; |
276 | unsigned char *c0drb = drb, *c1drb = &drb[I3000_RANKS_PER_CHANNEL]; | 276 | unsigned char *c0drb = drb, *c1drb = &drb[I3000_RANKS_PER_CHANNEL]; |
277 | unsigned long mchbar; | 277 | unsigned long mchbar; |
278 | void *window; | 278 | void __iomem *window; |
279 | 279 | ||
280 | debugf0("MC: %s()\n", __func__); | 280 | debugf0("MC: %s()\n", __func__); |
281 | 281 | ||
diff --git a/drivers/i2c/chips/ds1682.c b/drivers/i2c/chips/ds1682.c index 5879f0f25495..9e94542c18a2 100644 --- a/drivers/i2c/chips/ds1682.c +++ b/drivers/i2c/chips/ds1682.c | |||
@@ -75,7 +75,8 @@ static ssize_t ds1682_show(struct device *dev, struct device_attribute *attr, | |||
75 | /* Special case: the 32 bit regs are time values with 1/4s | 75 | /* Special case: the 32 bit regs are time values with 1/4s |
76 | * resolution, scale them up to milliseconds */ | 76 | * resolution, scale them up to milliseconds */ |
77 | if (sattr->nr == 4) | 77 | if (sattr->nr == 4) |
78 | return sprintf(buf, "%llu\n", ((u64) le32_to_cpu(val)) * 250); | 78 | return sprintf(buf, "%llu\n", |
79 | ((unsigned long long)le32_to_cpu(val)) * 250); | ||
79 | 80 | ||
80 | /* Format the output string and return # of bytes */ | 81 | /* Format the output string and return # of bytes */ |
81 | return sprintf(buf, "%li\n", (long)le32_to_cpu(val)); | 82 | return sprintf(buf, "%li\n", (long)le32_to_cpu(val)); |
diff --git a/drivers/ide/ide-cd.c b/drivers/ide/ide-cd.c index 1486eb212ccc..ca843522f91d 100644 --- a/drivers/ide/ide-cd.c +++ b/drivers/ide/ide-cd.c | |||
@@ -3071,7 +3071,7 @@ static inline void ide_cdrom_add_settings(ide_drive_t *drive) { ; } | |||
3071 | /* | 3071 | /* |
3072 | * standard prep_rq_fn that builds 10 byte cmds | 3072 | * standard prep_rq_fn that builds 10 byte cmds |
3073 | */ | 3073 | */ |
3074 | static int ide_cdrom_prep_fs(request_queue_t *q, struct request *rq) | 3074 | static int ide_cdrom_prep_fs(struct request_queue *q, struct request *rq) |
3075 | { | 3075 | { |
3076 | int hard_sect = queue_hardsect_size(q); | 3076 | int hard_sect = queue_hardsect_size(q); |
3077 | long block = (long)rq->hard_sector / (hard_sect >> 9); | 3077 | long block = (long)rq->hard_sector / (hard_sect >> 9); |
@@ -3137,7 +3137,7 @@ static int ide_cdrom_prep_pc(struct request *rq) | |||
3137 | return BLKPREP_OK; | 3137 | return BLKPREP_OK; |
3138 | } | 3138 | } |
3139 | 3139 | ||
3140 | static int ide_cdrom_prep_fn(request_queue_t *q, struct request *rq) | 3140 | static int ide_cdrom_prep_fn(struct request_queue *q, struct request *rq) |
3141 | { | 3141 | { |
3142 | if (blk_fs_request(rq)) | 3142 | if (blk_fs_request(rq)) |
3143 | return ide_cdrom_prep_fs(q, rq); | 3143 | return ide_cdrom_prep_fs(q, rq); |
diff --git a/drivers/ide/ide-disk.c b/drivers/ide/ide-disk.c index b1304a7f3e0a..5ce4216f72a2 100644 --- a/drivers/ide/ide-disk.c +++ b/drivers/ide/ide-disk.c | |||
@@ -679,7 +679,7 @@ static ide_proc_entry_t idedisk_proc[] = { | |||
679 | }; | 679 | }; |
680 | #endif /* CONFIG_IDE_PROC_FS */ | 680 | #endif /* CONFIG_IDE_PROC_FS */ |
681 | 681 | ||
682 | static void idedisk_prepare_flush(request_queue_t *q, struct request *rq) | 682 | static void idedisk_prepare_flush(struct request_queue *q, struct request *rq) |
683 | { | 683 | { |
684 | ide_drive_t *drive = q->queuedata; | 684 | ide_drive_t *drive = q->queuedata; |
685 | 685 | ||
@@ -697,7 +697,7 @@ static void idedisk_prepare_flush(request_queue_t *q, struct request *rq) | |||
697 | rq->buffer = rq->cmd; | 697 | rq->buffer = rq->cmd; |
698 | } | 698 | } |
699 | 699 | ||
700 | static int idedisk_issue_flush(request_queue_t *q, struct gendisk *disk, | 700 | static int idedisk_issue_flush(struct request_queue *q, struct gendisk *disk, |
701 | sector_t *error_sector) | 701 | sector_t *error_sector) |
702 | { | 702 | { |
703 | ide_drive_t *drive = q->queuedata; | 703 | ide_drive_t *drive = q->queuedata; |
diff --git a/drivers/ide/ide-io.c b/drivers/ide/ide-io.c index 484c50e71446..aa9f5f0b1e67 100644 --- a/drivers/ide/ide-io.c +++ b/drivers/ide/ide-io.c | |||
@@ -1327,7 +1327,7 @@ static void ide_do_request (ide_hwgroup_t *hwgroup, int masked_irq) | |||
1327 | /* | 1327 | /* |
1328 | * Passes the stuff to ide_do_request | 1328 | * Passes the stuff to ide_do_request |
1329 | */ | 1329 | */ |
1330 | void do_ide_request(request_queue_t *q) | 1330 | void do_ide_request(struct request_queue *q) |
1331 | { | 1331 | { |
1332 | ide_drive_t *drive = q->queuedata; | 1332 | ide_drive_t *drive = q->queuedata; |
1333 | 1333 | ||
diff --git a/drivers/ide/ide-probe.c b/drivers/ide/ide-probe.c index 5a4c5ea12f89..3a2a9a338fd9 100644 --- a/drivers/ide/ide-probe.c +++ b/drivers/ide/ide-probe.c | |||
@@ -945,7 +945,7 @@ static void save_match(ide_hwif_t *hwif, ide_hwif_t *new, ide_hwif_t **match) | |||
945 | */ | 945 | */ |
946 | static int ide_init_queue(ide_drive_t *drive) | 946 | static int ide_init_queue(ide_drive_t *drive) |
947 | { | 947 | { |
948 | request_queue_t *q; | 948 | struct request_queue *q; |
949 | ide_hwif_t *hwif = HWIF(drive); | 949 | ide_hwif_t *hwif = HWIF(drive); |
950 | int max_sectors = 256; | 950 | int max_sectors = 256; |
951 | int max_sg_entries = PRD_ENTRIES; | 951 | int max_sg_entries = PRD_ENTRIES; |
diff --git a/drivers/ide/legacy/hd.c b/drivers/ide/legacy/hd.c index 8f2db8dd35f7..8e05d88e81ba 100644 --- a/drivers/ide/legacy/hd.c +++ b/drivers/ide/legacy/hd.c | |||
@@ -652,7 +652,7 @@ repeat: | |||
652 | } | 652 | } |
653 | } | 653 | } |
654 | 654 | ||
655 | static void do_hd_request (request_queue_t * q) | 655 | static void do_hd_request (struct request_queue * q) |
656 | { | 656 | { |
657 | disable_irq(HD_IRQ); | 657 | disable_irq(HD_IRQ); |
658 | hd_request(); | 658 | hd_request(); |
diff --git a/drivers/ide/pci/scc_pata.c b/drivers/ide/pci/scc_pata.c index f668d235e6be..bf19ddfa6cda 100644 --- a/drivers/ide/pci/scc_pata.c +++ b/drivers/ide/pci/scc_pata.c | |||
@@ -551,8 +551,8 @@ static int setup_mmio_scc (struct pci_dev *dev, const char *name) | |||
551 | unsigned long dma_base = pci_resource_start(dev, 1); | 551 | unsigned long dma_base = pci_resource_start(dev, 1); |
552 | unsigned long ctl_size = pci_resource_len(dev, 0); | 552 | unsigned long ctl_size = pci_resource_len(dev, 0); |
553 | unsigned long dma_size = pci_resource_len(dev, 1); | 553 | unsigned long dma_size = pci_resource_len(dev, 1); |
554 | void *ctl_addr; | 554 | void __iomem *ctl_addr; |
555 | void *dma_addr; | 555 | void __iomem *dma_addr; |
556 | int i; | 556 | int i; |
557 | 557 | ||
558 | for (i = 0; i < MAX_HWIFS; i++) { | 558 | for (i = 0; i < MAX_HWIFS; i++) { |
diff --git a/drivers/ieee1394/raw1394.c b/drivers/ieee1394/raw1394.c index 336e5ff4cfcf..cadf0479cce5 100644 --- a/drivers/ieee1394/raw1394.c +++ b/drivers/ieee1394/raw1394.c | |||
@@ -2677,7 +2677,7 @@ static long raw1394_iso_xmit_recv_packets32(struct file *file, unsigned int cmd, | |||
2677 | struct raw1394_iso_packets32 __user *arg) | 2677 | struct raw1394_iso_packets32 __user *arg) |
2678 | { | 2678 | { |
2679 | compat_uptr_t infos32; | 2679 | compat_uptr_t infos32; |
2680 | void *infos; | 2680 | void __user *infos; |
2681 | long err = -EFAULT; | 2681 | long err = -EFAULT; |
2682 | struct raw1394_iso_packets __user *dst = compat_alloc_user_space(sizeof(struct raw1394_iso_packets)); | 2682 | struct raw1394_iso_packets __user *dst = compat_alloc_user_space(sizeof(struct raw1394_iso_packets)); |
2683 | 2683 | ||
diff --git a/drivers/input/misc/atlas_btns.c b/drivers/input/misc/atlas_btns.c index 0acc3a123604..e43e92fd9e23 100644 --- a/drivers/input/misc/atlas_btns.c +++ b/drivers/input/misc/atlas_btns.c | |||
@@ -31,7 +31,6 @@ | |||
31 | 31 | ||
32 | #define ACPI_ATLAS_NAME "Atlas ACPI" | 32 | #define ACPI_ATLAS_NAME "Atlas ACPI" |
33 | #define ACPI_ATLAS_CLASS "Atlas" | 33 | #define ACPI_ATLAS_CLASS "Atlas" |
34 | #define ACPI_ATLAS_BUTTON_HID "ASIM0000" | ||
35 | 34 | ||
36 | static struct input_dev *input_dev; | 35 | static struct input_dev *input_dev; |
37 | 36 | ||
@@ -130,10 +129,16 @@ static int atlas_acpi_button_remove(struct acpi_device *device, int type) | |||
130 | return status; | 129 | return status; |
131 | } | 130 | } |
132 | 131 | ||
132 | static const struct acpi_device_id atlas_device_ids[] = { | ||
133 | {"ASIM0000", 0}, | ||
134 | {"", 0}, | ||
135 | }; | ||
136 | MODULE_DEVICE_TABLE(acpi, atlas_device_ids); | ||
137 | |||
133 | static struct acpi_driver atlas_acpi_driver = { | 138 | static struct acpi_driver atlas_acpi_driver = { |
134 | .name = ACPI_ATLAS_NAME, | 139 | .name = ACPI_ATLAS_NAME, |
135 | .class = ACPI_ATLAS_CLASS, | 140 | .class = ACPI_ATLAS_CLASS, |
136 | .ids = ACPI_ATLAS_BUTTON_HID, | 141 | .ids = atlas_device_ids, |
137 | .ops = { | 142 | .ops = { |
138 | .add = atlas_acpi_button_add, | 143 | .add = atlas_acpi_button_add, |
139 | .remove = atlas_acpi_button_remove, | 144 | .remove = atlas_acpi_button_remove, |
diff --git a/drivers/kvm/kvm_main.c b/drivers/kvm/kvm_main.c index bcbe6835beb4..96856097d15b 100644 --- a/drivers/kvm/kvm_main.c +++ b/drivers/kvm/kvm_main.c | |||
@@ -297,9 +297,6 @@ static struct kvm *kvm_create_vm(void) | |||
297 | kvm_io_bus_init(&kvm->pio_bus); | 297 | kvm_io_bus_init(&kvm->pio_bus); |
298 | spin_lock_init(&kvm->lock); | 298 | spin_lock_init(&kvm->lock); |
299 | INIT_LIST_HEAD(&kvm->active_mmu_pages); | 299 | INIT_LIST_HEAD(&kvm->active_mmu_pages); |
300 | spin_lock(&kvm_lock); | ||
301 | list_add(&kvm->vm_list, &vm_list); | ||
302 | spin_unlock(&kvm_lock); | ||
303 | kvm_io_bus_init(&kvm->mmio_bus); | 300 | kvm_io_bus_init(&kvm->mmio_bus); |
304 | for (i = 0; i < KVM_MAX_VCPUS; ++i) { | 301 | for (i = 0; i < KVM_MAX_VCPUS; ++i) { |
305 | struct kvm_vcpu *vcpu = &kvm->vcpus[i]; | 302 | struct kvm_vcpu *vcpu = &kvm->vcpus[i]; |
@@ -309,6 +306,9 @@ static struct kvm *kvm_create_vm(void) | |||
309 | vcpu->kvm = kvm; | 306 | vcpu->kvm = kvm; |
310 | vcpu->mmu.root_hpa = INVALID_PAGE; | 307 | vcpu->mmu.root_hpa = INVALID_PAGE; |
311 | } | 308 | } |
309 | spin_lock(&kvm_lock); | ||
310 | list_add(&kvm->vm_list, &vm_list); | ||
311 | spin_unlock(&kvm_lock); | ||
312 | return kvm; | 312 | return kvm; |
313 | } | 313 | } |
314 | 314 | ||
@@ -1070,18 +1070,16 @@ static int emulator_write_phys(struct kvm_vcpu *vcpu, gpa_t gpa, | |||
1070 | return 0; | 1070 | return 0; |
1071 | mark_page_dirty(vcpu->kvm, gpa >> PAGE_SHIFT); | 1071 | mark_page_dirty(vcpu->kvm, gpa >> PAGE_SHIFT); |
1072 | virt = kmap_atomic(page, KM_USER0); | 1072 | virt = kmap_atomic(page, KM_USER0); |
1073 | if (memcmp(virt + offset_in_page(gpa), val, bytes)) { | 1073 | kvm_mmu_pte_write(vcpu, gpa, virt + offset, val, bytes); |
1074 | kvm_mmu_pte_write(vcpu, gpa, virt + offset, val, bytes); | 1074 | memcpy(virt + offset_in_page(gpa), val, bytes); |
1075 | memcpy(virt + offset_in_page(gpa), val, bytes); | ||
1076 | } | ||
1077 | kunmap_atomic(virt, KM_USER0); | 1075 | kunmap_atomic(virt, KM_USER0); |
1078 | return 1; | 1076 | return 1; |
1079 | } | 1077 | } |
1080 | 1078 | ||
1081 | static int emulator_write_emulated(unsigned long addr, | 1079 | static int emulator_write_emulated_onepage(unsigned long addr, |
1082 | const void *val, | 1080 | const void *val, |
1083 | unsigned int bytes, | 1081 | unsigned int bytes, |
1084 | struct x86_emulate_ctxt *ctxt) | 1082 | struct x86_emulate_ctxt *ctxt) |
1085 | { | 1083 | { |
1086 | struct kvm_vcpu *vcpu = ctxt->vcpu; | 1084 | struct kvm_vcpu *vcpu = ctxt->vcpu; |
1087 | struct kvm_io_device *mmio_dev; | 1085 | struct kvm_io_device *mmio_dev; |
@@ -1113,6 +1111,26 @@ static int emulator_write_emulated(unsigned long addr, | |||
1113 | return X86EMUL_CONTINUE; | 1111 | return X86EMUL_CONTINUE; |
1114 | } | 1112 | } |
1115 | 1113 | ||
1114 | static int emulator_write_emulated(unsigned long addr, | ||
1115 | const void *val, | ||
1116 | unsigned int bytes, | ||
1117 | struct x86_emulate_ctxt *ctxt) | ||
1118 | { | ||
1119 | /* Crossing a page boundary? */ | ||
1120 | if (((addr + bytes - 1) ^ addr) & PAGE_MASK) { | ||
1121 | int rc, now; | ||
1122 | |||
1123 | now = -addr & ~PAGE_MASK; | ||
1124 | rc = emulator_write_emulated_onepage(addr, val, now, ctxt); | ||
1125 | if (rc != X86EMUL_CONTINUE) | ||
1126 | return rc; | ||
1127 | addr += now; | ||
1128 | val += now; | ||
1129 | bytes -= now; | ||
1130 | } | ||
1131 | return emulator_write_emulated_onepage(addr, val, bytes, ctxt); | ||
1132 | } | ||
1133 | |||
1116 | static int emulator_cmpxchg_emulated(unsigned long addr, | 1134 | static int emulator_cmpxchg_emulated(unsigned long addr, |
1117 | const void *old, | 1135 | const void *old, |
1118 | const void *new, | 1136 | const void *new, |
@@ -2414,9 +2432,9 @@ static void cpuid_fix_nx_cap(struct kvm_vcpu *vcpu) | |||
2414 | break; | 2432 | break; |
2415 | } | 2433 | } |
2416 | } | 2434 | } |
2417 | if (entry && (entry->edx & EFER_NX) && !(efer & EFER_NX)) { | 2435 | if (entry && (entry->edx & (1 << 20)) && !(efer & EFER_NX)) { |
2418 | entry->edx &= ~(1 << 20); | 2436 | entry->edx &= ~(1 << 20); |
2419 | printk(KERN_INFO ": guest NX capability removed\n"); | 2437 | printk(KERN_INFO "kvm: guest NX capability removed\n"); |
2420 | } | 2438 | } |
2421 | } | 2439 | } |
2422 | 2440 | ||
diff --git a/drivers/kvm/x86_emulate.c b/drivers/kvm/x86_emulate.c index 1b800fc00342..1f979cb0df31 100644 --- a/drivers/kvm/x86_emulate.c +++ b/drivers/kvm/x86_emulate.c | |||
@@ -1178,6 +1178,8 @@ pop_instruction: | |||
1178 | twobyte_insn: | 1178 | twobyte_insn: |
1179 | switch (b) { | 1179 | switch (b) { |
1180 | case 0x01: /* lgdt, lidt, lmsw */ | 1180 | case 0x01: /* lgdt, lidt, lmsw */ |
1181 | /* Disable writeback. */ | ||
1182 | no_wb = 1; | ||
1181 | switch (modrm_reg) { | 1183 | switch (modrm_reg) { |
1182 | u16 size; | 1184 | u16 size; |
1183 | unsigned long address; | 1185 | unsigned long address; |
diff --git a/drivers/lguest/Makefile b/drivers/lguest/Makefile index 55382c7d799c..e5047471c334 100644 --- a/drivers/lguest/Makefile +++ b/drivers/lguest/Makefile | |||
@@ -5,3 +5,15 @@ obj-$(CONFIG_LGUEST_GUEST) += lguest.o lguest_asm.o lguest_bus.o | |||
5 | obj-$(CONFIG_LGUEST) += lg.o | 5 | obj-$(CONFIG_LGUEST) += lg.o |
6 | lg-y := core.o hypercalls.o page_tables.o interrupts_and_traps.o \ | 6 | lg-y := core.o hypercalls.o page_tables.o interrupts_and_traps.o \ |
7 | segments.o io.o lguest_user.o switcher.o | 7 | segments.o io.o lguest_user.o switcher.o |
8 | |||
9 | Preparation Preparation!: PREFIX=P | ||
10 | Guest: PREFIX=G | ||
11 | Drivers: PREFIX=D | ||
12 | Launcher: PREFIX=L | ||
13 | Host: PREFIX=H | ||
14 | Switcher: PREFIX=S | ||
15 | Mastery: PREFIX=M | ||
16 | Beer: | ||
17 | @for f in Preparation Guest Drivers Launcher Host Switcher Mastery; do echo "{==- $$f -==}"; make -s $$f; done; echo "{==-==}" | ||
18 | Preparation Preparation! Guest Drivers Launcher Host Switcher Mastery: | ||
19 | @sh ../../Documentation/lguest/extract $(PREFIX) `find ../../* -name '*.[chS]' -wholename '*lguest*'` | ||
diff --git a/drivers/lguest/README b/drivers/lguest/README new file mode 100644 index 000000000000..b7db39a64c66 --- /dev/null +++ b/drivers/lguest/README | |||
@@ -0,0 +1,47 @@ | |||
1 | Welcome, friend reader, to lguest. | ||
2 | |||
3 | Lguest is an adventure, with you, the reader, as Hero. I can't think of many | ||
4 | 5000-line projects which offer both such capability and glimpses of future | ||
5 | potential; it is an exciting time to be delving into the source! | ||
6 | |||
7 | But be warned; this is an arduous journey of several hours or more! And as we | ||
8 | know, all true Heroes are driven by a Noble Goal. Thus I offer a Beer (or | ||
9 | equivalent) to anyone I meet who has completed this documentation. | ||
10 | |||
11 | So get comfortable and keep your wits about you (both quick and humorous). | ||
12 | Along your way to the Noble Goal, you will also gain masterly insight into | ||
13 | lguest, and hypervisors and x86 virtualization in general. | ||
14 | |||
15 | Our Quest is in seven parts: (best read with C highlighting turned on) | ||
16 | |||
17 | I) Preparation | ||
18 | - In which our potential hero is flown quickly over the landscape for a | ||
19 | taste of its scope. Suitable for the armchair coders and other such | ||
20 | persons of faint constitution. | ||
21 | |||
22 | II) Guest | ||
23 | - Where we encounter the first tantalising wisps of code, and come to | ||
24 | understand the details of the life of a Guest kernel. | ||
25 | |||
26 | III) Drivers | ||
27 | - Whereby the Guest finds its voice and become useful, and our | ||
28 | understanding of the Guest is completed. | ||
29 | |||
30 | IV) Launcher | ||
31 | - Where we trace back to the creation of the Guest, and thus begin our | ||
32 | understanding of the Host. | ||
33 | |||
34 | V) Host | ||
35 | - Where we master the Host code, through a long and tortuous journey. | ||
36 | Indeed, it is here that our hero is tested in the Bit of Despair. | ||
37 | |||
38 | VI) Switcher | ||
39 | - Where our understanding of the intertwined nature of Guests and Hosts | ||
40 | is completed. | ||
41 | |||
42 | VII) Mastery | ||
43 | - Where our fully fledged hero grapples with the Great Question: | ||
44 | "What next?" | ||
45 | |||
46 | make Preparation! | ||
47 | Rusty Russell. | ||
diff --git a/drivers/lguest/core.c b/drivers/lguest/core.c index ce909ec57499..0a46e8837d9a 100644 --- a/drivers/lguest/core.c +++ b/drivers/lguest/core.c | |||
@@ -1,5 +1,8 @@ | |||
1 | /* World's simplest hypervisor, to test paravirt_ops and show | 1 | /*P:400 This contains run_guest() which actually calls into the Host<->Guest |
2 | * unbelievers that virtualization is the future. Plus, it's fun! */ | 2 | * Switcher and analyzes the return, such as determining if the Guest wants the |
3 | * Host to do something. This file also contains useful helper routines, and a | ||
4 | * couple of non-obvious setup and teardown pieces which were implemented after | ||
5 | * days of debugging pain. :*/ | ||
3 | #include <linux/module.h> | 6 | #include <linux/module.h> |
4 | #include <linux/stringify.h> | 7 | #include <linux/stringify.h> |
5 | #include <linux/stddef.h> | 8 | #include <linux/stddef.h> |
@@ -61,11 +64,33 @@ static struct lguest_pages *lguest_pages(unsigned int cpu) | |||
61 | (SWITCHER_ADDR + SHARED_SWITCHER_PAGES*PAGE_SIZE))[cpu]); | 64 | (SWITCHER_ADDR + SHARED_SWITCHER_PAGES*PAGE_SIZE))[cpu]); |
62 | } | 65 | } |
63 | 66 | ||
67 | /*H:010 We need to set up the Switcher at a high virtual address. Remember the | ||
68 | * Switcher is a few hundred bytes of assembler code which actually changes the | ||
69 | * CPU to run the Guest, and then changes back to the Host when a trap or | ||
70 | * interrupt happens. | ||
71 | * | ||
72 | * The Switcher code must be at the same virtual address in the Guest as the | ||
73 | * Host since it will be running as the switchover occurs. | ||
74 | * | ||
75 | * Trying to map memory at a particular address is an unusual thing to do, so | ||
76 | * it's not a simple one-liner. We also set up the per-cpu parts of the | ||
77 | * Switcher here. | ||
78 | */ | ||
64 | static __init int map_switcher(void) | 79 | static __init int map_switcher(void) |
65 | { | 80 | { |
66 | int i, err; | 81 | int i, err; |
67 | struct page **pagep; | 82 | struct page **pagep; |
68 | 83 | ||
84 | /* | ||
85 | * Map the Switcher in to high memory. | ||
86 | * | ||
87 | * It turns out that if we choose the address 0xFFC00000 (4MB under the | ||
88 | * top virtual address), it makes setting up the page tables really | ||
89 | * easy. | ||
90 | */ | ||
91 | |||
92 | /* We allocate an array of "struct page"s. map_vm_area() wants the | ||
93 | * pages in this form, rather than just an array of pointers. */ | ||
69 | switcher_page = kmalloc(sizeof(switcher_page[0])*TOTAL_SWITCHER_PAGES, | 94 | switcher_page = kmalloc(sizeof(switcher_page[0])*TOTAL_SWITCHER_PAGES, |
70 | GFP_KERNEL); | 95 | GFP_KERNEL); |
71 | if (!switcher_page) { | 96 | if (!switcher_page) { |
@@ -73,6 +98,8 @@ static __init int map_switcher(void) | |||
73 | goto out; | 98 | goto out; |
74 | } | 99 | } |
75 | 100 | ||
101 | /* Now we actually allocate the pages. The Guest will see these pages, | ||
102 | * so we make sure they're zeroed. */ | ||
76 | for (i = 0; i < TOTAL_SWITCHER_PAGES; i++) { | 103 | for (i = 0; i < TOTAL_SWITCHER_PAGES; i++) { |
77 | unsigned long addr = get_zeroed_page(GFP_KERNEL); | 104 | unsigned long addr = get_zeroed_page(GFP_KERNEL); |
78 | if (!addr) { | 105 | if (!addr) { |
@@ -82,6 +109,9 @@ static __init int map_switcher(void) | |||
82 | switcher_page[i] = virt_to_page(addr); | 109 | switcher_page[i] = virt_to_page(addr); |
83 | } | 110 | } |
84 | 111 | ||
112 | /* Now we reserve the "virtual memory area" we want: 0xFFC00000 | ||
113 | * (SWITCHER_ADDR). We might not get it in theory, but in practice | ||
114 | * it's worked so far. */ | ||
85 | switcher_vma = __get_vm_area(TOTAL_SWITCHER_PAGES * PAGE_SIZE, | 115 | switcher_vma = __get_vm_area(TOTAL_SWITCHER_PAGES * PAGE_SIZE, |
86 | VM_ALLOC, SWITCHER_ADDR, VMALLOC_END); | 116 | VM_ALLOC, SWITCHER_ADDR, VMALLOC_END); |
87 | if (!switcher_vma) { | 117 | if (!switcher_vma) { |
@@ -90,49 +120,105 @@ static __init int map_switcher(void) | |||
90 | goto free_pages; | 120 | goto free_pages; |
91 | } | 121 | } |
92 | 122 | ||
123 | /* This code actually sets up the pages we've allocated to appear at | ||
124 | * SWITCHER_ADDR. map_vm_area() takes the vma we allocated above, the | ||
125 | * kind of pages we're mapping (kernel pages), and a pointer to our | ||
126 | * array of struct pages. It increments that pointer, but we don't | ||
127 | * care. */ | ||
93 | pagep = switcher_page; | 128 | pagep = switcher_page; |
94 | err = map_vm_area(switcher_vma, PAGE_KERNEL, &pagep); | 129 | err = map_vm_area(switcher_vma, PAGE_KERNEL, &pagep); |
95 | if (err) { | 130 | if (err) { |
96 | printk("lguest: map_vm_area failed: %i\n", err); | 131 | printk("lguest: map_vm_area failed: %i\n", err); |
97 | goto free_vma; | 132 | goto free_vma; |
98 | } | 133 | } |
134 | |||
135 | /* Now the switcher is mapped at the right address, we can't fail! | ||
136 | * Copy in the compiled-in Switcher code (from switcher.S). */ | ||
99 | memcpy(switcher_vma->addr, start_switcher_text, | 137 | memcpy(switcher_vma->addr, start_switcher_text, |
100 | end_switcher_text - start_switcher_text); | 138 | end_switcher_text - start_switcher_text); |
101 | 139 | ||
102 | /* Fix up IDT entries to point into copied text. */ | 140 | /* Most of the switcher.S doesn't care that it's been moved; on Intel, |
141 | * jumps are relative, and it doesn't access any references to external | ||
142 | * code or data. | ||
143 | * | ||
144 | * The only exception is the interrupt handlers in switcher.S: their | ||
145 | * addresses are placed in a table (default_idt_entries), so we need to | ||
146 | * update the table with the new addresses. switcher_offset() is a | ||
147 | * convenience function which returns the distance between the builtin | ||
148 | * switcher code and the high-mapped copy we just made. */ | ||
103 | for (i = 0; i < IDT_ENTRIES; i++) | 149 | for (i = 0; i < IDT_ENTRIES; i++) |
104 | default_idt_entries[i] += switcher_offset(); | 150 | default_idt_entries[i] += switcher_offset(); |
105 | 151 | ||
152 | /* | ||
153 | * Set up the Switcher's per-cpu areas. | ||
154 | * | ||
155 | * Each CPU gets two pages of its own within the high-mapped region | ||
156 | * (aka. "struct lguest_pages"). Much of this can be initialized now, | ||
157 | * but some depends on what Guest we are running (which is set up in | ||
158 | * copy_in_guest_info()). | ||
159 | */ | ||
106 | for_each_possible_cpu(i) { | 160 | for_each_possible_cpu(i) { |
161 | /* lguest_pages() returns this CPU's two pages. */ | ||
107 | struct lguest_pages *pages = lguest_pages(i); | 162 | struct lguest_pages *pages = lguest_pages(i); |
163 | /* This is a convenience pointer to make the code fit one | ||
164 | * statement to a line. */ | ||
108 | struct lguest_ro_state *state = &pages->state; | 165 | struct lguest_ro_state *state = &pages->state; |
109 | 166 | ||
110 | /* These fields are static: rest done in copy_in_guest_info */ | 167 | /* The Global Descriptor Table: the Host has a different one |
168 | * for each CPU. We keep a descriptor for the GDT which says | ||
169 | * where it is and how big it is (the size is actually the last | ||
170 | * byte, not the size, hence the "-1"). */ | ||
111 | state->host_gdt_desc.size = GDT_SIZE-1; | 171 | state->host_gdt_desc.size = GDT_SIZE-1; |
112 | state->host_gdt_desc.address = (long)get_cpu_gdt_table(i); | 172 | state->host_gdt_desc.address = (long)get_cpu_gdt_table(i); |
173 | |||
174 | /* All CPUs on the Host use the same Interrupt Descriptor | ||
175 | * Table, so we just use store_idt(), which gets this CPU's IDT | ||
176 | * descriptor. */ | ||
113 | store_idt(&state->host_idt_desc); | 177 | store_idt(&state->host_idt_desc); |
178 | |||
179 | /* The descriptors for the Guest's GDT and IDT can be filled | ||
180 | * out now, too. We copy the GDT & IDT into ->guest_gdt and | ||
181 | * ->guest_idt before actually running the Guest. */ | ||
114 | state->guest_idt_desc.size = sizeof(state->guest_idt)-1; | 182 | state->guest_idt_desc.size = sizeof(state->guest_idt)-1; |
115 | state->guest_idt_desc.address = (long)&state->guest_idt; | 183 | state->guest_idt_desc.address = (long)&state->guest_idt; |
116 | state->guest_gdt_desc.size = sizeof(state->guest_gdt)-1; | 184 | state->guest_gdt_desc.size = sizeof(state->guest_gdt)-1; |
117 | state->guest_gdt_desc.address = (long)&state->guest_gdt; | 185 | state->guest_gdt_desc.address = (long)&state->guest_gdt; |
186 | |||
187 | /* We know where we want the stack to be when the Guest enters | ||
188 | * the switcher: in pages->regs. The stack grows upwards, so | ||
189 | * we start it at the end of that structure. */ | ||
118 | state->guest_tss.esp0 = (long)(&pages->regs + 1); | 190 | state->guest_tss.esp0 = (long)(&pages->regs + 1); |
191 | /* And this is the GDT entry to use for the stack: we keep a | ||
192 | * couple of special LGUEST entries. */ | ||
119 | state->guest_tss.ss0 = LGUEST_DS; | 193 | state->guest_tss.ss0 = LGUEST_DS; |
120 | /* No I/O for you! */ | 194 | |
195 | /* x86 can have a finegrained bitmap which indicates what I/O | ||
196 | * ports the process can use. We set it to the end of our | ||
197 | * structure, meaning "none". */ | ||
121 | state->guest_tss.io_bitmap_base = sizeof(state->guest_tss); | 198 | state->guest_tss.io_bitmap_base = sizeof(state->guest_tss); |
199 | |||
200 | /* Some GDT entries are the same across all Guests, so we can | ||
201 | * set them up now. */ | ||
122 | setup_default_gdt_entries(state); | 202 | setup_default_gdt_entries(state); |
203 | /* Most IDT entries are the same for all Guests, too.*/ | ||
123 | setup_default_idt_entries(state, default_idt_entries); | 204 | setup_default_idt_entries(state, default_idt_entries); |
124 | 205 | ||
125 | /* Setup LGUEST segments on all cpus */ | 206 | /* The Host needs to be able to use the LGUEST segments on this |
207 | * CPU, too, so put them in the Host GDT. */ | ||
126 | get_cpu_gdt_table(i)[GDT_ENTRY_LGUEST_CS] = FULL_EXEC_SEGMENT; | 208 | get_cpu_gdt_table(i)[GDT_ENTRY_LGUEST_CS] = FULL_EXEC_SEGMENT; |
127 | get_cpu_gdt_table(i)[GDT_ENTRY_LGUEST_DS] = FULL_SEGMENT; | 209 | get_cpu_gdt_table(i)[GDT_ENTRY_LGUEST_DS] = FULL_SEGMENT; |
128 | } | 210 | } |
129 | 211 | ||
130 | /* Initialize entry point into switcher. */ | 212 | /* In the Switcher, we want the %cs segment register to use the |
213 | * LGUEST_CS GDT entry: we've put that in the Host and Guest GDTs, so | ||
214 | * it will be undisturbed when we switch. To change %cs and jump we | ||
215 | * need this structure to feed to Intel's "lcall" instruction. */ | ||
131 | lguest_entry.offset = (long)switch_to_guest + switcher_offset(); | 216 | lguest_entry.offset = (long)switch_to_guest + switcher_offset(); |
132 | lguest_entry.segment = LGUEST_CS; | 217 | lguest_entry.segment = LGUEST_CS; |
133 | 218 | ||
134 | printk(KERN_INFO "lguest: mapped switcher at %p\n", | 219 | printk(KERN_INFO "lguest: mapped switcher at %p\n", |
135 | switcher_vma->addr); | 220 | switcher_vma->addr); |
221 | /* And we succeeded... */ | ||
136 | return 0; | 222 | return 0; |
137 | 223 | ||
138 | free_vma: | 224 | free_vma: |
@@ -146,35 +232,58 @@ free_some_pages: | |||
146 | out: | 232 | out: |
147 | return err; | 233 | return err; |
148 | } | 234 | } |
235 | /*:*/ | ||
149 | 236 | ||
237 | /* Cleaning up the mapping when the module is unloaded is almost... | ||
238 | * too easy. */ | ||
150 | static void unmap_switcher(void) | 239 | static void unmap_switcher(void) |
151 | { | 240 | { |
152 | unsigned int i; | 241 | unsigned int i; |
153 | 242 | ||
243 | /* vunmap() undoes *both* map_vm_area() and __get_vm_area(). */ | ||
154 | vunmap(switcher_vma->addr); | 244 | vunmap(switcher_vma->addr); |
245 | /* Now we just need to free the pages we copied the switcher into */ | ||
155 | for (i = 0; i < TOTAL_SWITCHER_PAGES; i++) | 246 | for (i = 0; i < TOTAL_SWITCHER_PAGES; i++) |
156 | __free_pages(switcher_page[i], 0); | 247 | __free_pages(switcher_page[i], 0); |
157 | } | 248 | } |
158 | 249 | ||
159 | /* IN/OUT insns: enough to get us past boot-time probing. */ | 250 | /*H:130 Our Guest is usually so well behaved; it never tries to do things it |
251 | * isn't allowed to. Unfortunately, "struct paravirt_ops" isn't quite | ||
252 | * complete, because it doesn't contain replacements for the Intel I/O | ||
253 | * instructions. As a result, the Guest sometimes fumbles across one during | ||
254 | * the boot process as it probes for various things which are usually attached | ||
255 | * to a PC. | ||
256 | * | ||
257 | * When the Guest uses one of these instructions, we get trap #13 (General | ||
258 | * Protection Fault) and come here. We see if it's one of those troublesome | ||
259 | * instructions and skip over it. We return true if we did. */ | ||
160 | static int emulate_insn(struct lguest *lg) | 260 | static int emulate_insn(struct lguest *lg) |
161 | { | 261 | { |
162 | u8 insn; | 262 | u8 insn; |
163 | unsigned int insnlen = 0, in = 0, shift = 0; | 263 | unsigned int insnlen = 0, in = 0, shift = 0; |
264 | /* The eip contains the *virtual* address of the Guest's instruction: | ||
265 | * guest_pa just subtracts the Guest's page_offset. */ | ||
164 | unsigned long physaddr = guest_pa(lg, lg->regs->eip); | 266 | unsigned long physaddr = guest_pa(lg, lg->regs->eip); |
165 | 267 | ||
166 | /* This only works for addresses in linear mapping... */ | 268 | /* The guest_pa() function only works for Guest kernel addresses, but |
269 | * that's all we're trying to do anyway. */ | ||
167 | if (lg->regs->eip < lg->page_offset) | 270 | if (lg->regs->eip < lg->page_offset) |
168 | return 0; | 271 | return 0; |
272 | |||
273 | /* Decoding x86 instructions is icky. */ | ||
169 | lgread(lg, &insn, physaddr, 1); | 274 | lgread(lg, &insn, physaddr, 1); |
170 | 275 | ||
171 | /* Operand size prefix means it's actually for ax. */ | 276 | /* 0x66 is an "operand prefix". It means it's using the upper 16 bits |
277 | of the eax register. */ | ||
172 | if (insn == 0x66) { | 278 | if (insn == 0x66) { |
173 | shift = 16; | 279 | shift = 16; |
280 | /* The instruction is 1 byte so far, read the next byte. */ | ||
174 | insnlen = 1; | 281 | insnlen = 1; |
175 | lgread(lg, &insn, physaddr + insnlen, 1); | 282 | lgread(lg, &insn, physaddr + insnlen, 1); |
176 | } | 283 | } |
177 | 284 | ||
285 | /* We can ignore the lower bit for the moment and decode the 4 opcodes | ||
286 | * we need to emulate. */ | ||
178 | switch (insn & 0xFE) { | 287 | switch (insn & 0xFE) { |
179 | case 0xE4: /* in <next byte>,%al */ | 288 | case 0xE4: /* in <next byte>,%al */ |
180 | insnlen += 2; | 289 | insnlen += 2; |
@@ -191,9 +300,13 @@ static int emulate_insn(struct lguest *lg) | |||
191 | insnlen += 1; | 300 | insnlen += 1; |
192 | break; | 301 | break; |
193 | default: | 302 | default: |
303 | /* OK, we don't know what this is, can't emulate. */ | ||
194 | return 0; | 304 | return 0; |
195 | } | 305 | } |
196 | 306 | ||
307 | /* If it was an "IN" instruction, they expect the result to be read | ||
308 | * into %eax, so we change %eax. We always return all-ones, which | ||
309 | * traditionally means "there's nothing there". */ | ||
197 | if (in) { | 310 | if (in) { |
198 | /* Lower bit tells is whether it's a 16 or 32 bit access */ | 311 | /* Lower bit tells is whether it's a 16 or 32 bit access */ |
199 | if (insn & 0x1) | 312 | if (insn & 0x1) |
@@ -201,28 +314,46 @@ static int emulate_insn(struct lguest *lg) | |||
201 | else | 314 | else |
202 | lg->regs->eax |= (0xFFFF << shift); | 315 | lg->regs->eax |= (0xFFFF << shift); |
203 | } | 316 | } |
317 | /* Finally, we've "done" the instruction, so move past it. */ | ||
204 | lg->regs->eip += insnlen; | 318 | lg->regs->eip += insnlen; |
319 | /* Success! */ | ||
205 | return 1; | 320 | return 1; |
206 | } | 321 | } |
207 | 322 | /*:*/ | |
323 | |||
324 | /*L:305 | ||
325 | * Dealing With Guest Memory. | ||
326 | * | ||
327 | * When the Guest gives us (what it thinks is) a physical address, we can use | ||
328 | * the normal copy_from_user() & copy_to_user() on that address: remember, | ||
329 | * Guest physical == Launcher virtual. | ||
330 | * | ||
331 | * But we can't trust the Guest: it might be trying to access the Launcher | ||
332 | * code. We have to check that the range is below the pfn_limit the Launcher | ||
333 | * gave us. We have to make sure that addr + len doesn't give us a false | ||
334 | * positive by overflowing, too. */ | ||
208 | int lguest_address_ok(const struct lguest *lg, | 335 | int lguest_address_ok(const struct lguest *lg, |
209 | unsigned long addr, unsigned long len) | 336 | unsigned long addr, unsigned long len) |
210 | { | 337 | { |
211 | return (addr+len) / PAGE_SIZE < lg->pfn_limit && (addr+len >= addr); | 338 | return (addr+len) / PAGE_SIZE < lg->pfn_limit && (addr+len >= addr); |
212 | } | 339 | } |
213 | 340 | ||
214 | /* Just like get_user, but don't let guest access lguest binary. */ | 341 | /* This is a convenient routine to get a 32-bit value from the Guest (a very |
342 | * common operation). Here we can see how useful the kill_lguest() routine we | ||
343 | * met in the Launcher can be: we return a random value (0) instead of needing | ||
344 | * to return an error. */ | ||
215 | u32 lgread_u32(struct lguest *lg, unsigned long addr) | 345 | u32 lgread_u32(struct lguest *lg, unsigned long addr) |
216 | { | 346 | { |
217 | u32 val = 0; | 347 | u32 val = 0; |
218 | 348 | ||
219 | /* Don't let them access lguest binary */ | 349 | /* Don't let them access lguest binary. */ |
220 | if (!lguest_address_ok(lg, addr, sizeof(val)) | 350 | if (!lguest_address_ok(lg, addr, sizeof(val)) |
221 | || get_user(val, (u32 __user *)addr) != 0) | 351 | || get_user(val, (u32 __user *)addr) != 0) |
222 | kill_guest(lg, "bad read address %#lx", addr); | 352 | kill_guest(lg, "bad read address %#lx", addr); |
223 | return val; | 353 | return val; |
224 | } | 354 | } |
225 | 355 | ||
356 | /* Same thing for writing a value. */ | ||
226 | void lgwrite_u32(struct lguest *lg, unsigned long addr, u32 val) | 357 | void lgwrite_u32(struct lguest *lg, unsigned long addr, u32 val) |
227 | { | 358 | { |
228 | if (!lguest_address_ok(lg, addr, sizeof(val)) | 359 | if (!lguest_address_ok(lg, addr, sizeof(val)) |
@@ -230,6 +361,9 @@ void lgwrite_u32(struct lguest *lg, unsigned long addr, u32 val) | |||
230 | kill_guest(lg, "bad write address %#lx", addr); | 361 | kill_guest(lg, "bad write address %#lx", addr); |
231 | } | 362 | } |
232 | 363 | ||
364 | /* This routine is more generic, and copies a range of Guest bytes into a | ||
365 | * buffer. If the copy_from_user() fails, we fill the buffer with zeroes, so | ||
366 | * the caller doesn't end up using uninitialized kernel memory. */ | ||
233 | void lgread(struct lguest *lg, void *b, unsigned long addr, unsigned bytes) | 367 | void lgread(struct lguest *lg, void *b, unsigned long addr, unsigned bytes) |
234 | { | 368 | { |
235 | if (!lguest_address_ok(lg, addr, bytes) | 369 | if (!lguest_address_ok(lg, addr, bytes) |
@@ -240,6 +374,7 @@ void lgread(struct lguest *lg, void *b, unsigned long addr, unsigned bytes) | |||
240 | } | 374 | } |
241 | } | 375 | } |
242 | 376 | ||
377 | /* Similarly, our generic routine to copy into a range of Guest bytes. */ | ||
243 | void lgwrite(struct lguest *lg, unsigned long addr, const void *b, | 378 | void lgwrite(struct lguest *lg, unsigned long addr, const void *b, |
244 | unsigned bytes) | 379 | unsigned bytes) |
245 | { | 380 | { |
@@ -247,6 +382,7 @@ void lgwrite(struct lguest *lg, unsigned long addr, const void *b, | |||
247 | || copy_to_user((void __user *)addr, b, bytes) != 0) | 382 | || copy_to_user((void __user *)addr, b, bytes) != 0) |
248 | kill_guest(lg, "bad write address %#lx len %u", addr, bytes); | 383 | kill_guest(lg, "bad write address %#lx len %u", addr, bytes); |
249 | } | 384 | } |
385 | /* (end of memory access helper routines) :*/ | ||
250 | 386 | ||
251 | static void set_ts(void) | 387 | static void set_ts(void) |
252 | { | 388 | { |
@@ -257,54 +393,108 @@ static void set_ts(void) | |||
257 | write_cr0(cr0|8); | 393 | write_cr0(cr0|8); |
258 | } | 394 | } |
259 | 395 | ||
396 | /*S:010 | ||
397 | * We are getting close to the Switcher. | ||
398 | * | ||
399 | * Remember that each CPU has two pages which are visible to the Guest when it | ||
400 | * runs on that CPU. This has to contain the state for that Guest: we copy the | ||
401 | * state in just before we run the Guest. | ||
402 | * | ||
403 | * Each Guest has "changed" flags which indicate what has changed in the Guest | ||
404 | * since it last ran. We saw this set in interrupts_and_traps.c and | ||
405 | * segments.c. | ||
406 | */ | ||
260 | static void copy_in_guest_info(struct lguest *lg, struct lguest_pages *pages) | 407 | static void copy_in_guest_info(struct lguest *lg, struct lguest_pages *pages) |
261 | { | 408 | { |
409 | /* Copying all this data can be quite expensive. We usually run the | ||
410 | * same Guest we ran last time (and that Guest hasn't run anywhere else | ||
411 | * meanwhile). If that's not the case, we pretend everything in the | ||
412 | * Guest has changed. */ | ||
262 | if (__get_cpu_var(last_guest) != lg || lg->last_pages != pages) { | 413 | if (__get_cpu_var(last_guest) != lg || lg->last_pages != pages) { |
263 | __get_cpu_var(last_guest) = lg; | 414 | __get_cpu_var(last_guest) = lg; |
264 | lg->last_pages = pages; | 415 | lg->last_pages = pages; |
265 | lg->changed = CHANGED_ALL; | 416 | lg->changed = CHANGED_ALL; |
266 | } | 417 | } |
267 | 418 | ||
268 | /* These are pretty cheap, so we do them unconditionally. */ | 419 | /* These copies are pretty cheap, so we do them unconditionally: */ |
420 | /* Save the current Host top-level page directory. */ | ||
269 | pages->state.host_cr3 = __pa(current->mm->pgd); | 421 | pages->state.host_cr3 = __pa(current->mm->pgd); |
422 | /* Set up the Guest's page tables to see this CPU's pages (and no | ||
423 | * other CPU's pages). */ | ||
270 | map_switcher_in_guest(lg, pages); | 424 | map_switcher_in_guest(lg, pages); |
425 | /* Set up the two "TSS" members which tell the CPU what stack to use | ||
426 | * for traps which do directly into the Guest (ie. traps at privilege | ||
427 | * level 1). */ | ||
271 | pages->state.guest_tss.esp1 = lg->esp1; | 428 | pages->state.guest_tss.esp1 = lg->esp1; |
272 | pages->state.guest_tss.ss1 = lg->ss1; | 429 | pages->state.guest_tss.ss1 = lg->ss1; |
273 | 430 | ||
274 | /* Copy direct trap entries. */ | 431 | /* Copy direct-to-Guest trap entries. */ |
275 | if (lg->changed & CHANGED_IDT) | 432 | if (lg->changed & CHANGED_IDT) |
276 | copy_traps(lg, pages->state.guest_idt, default_idt_entries); | 433 | copy_traps(lg, pages->state.guest_idt, default_idt_entries); |
277 | 434 | ||
278 | /* Copy all GDT entries but the TSS. */ | 435 | /* Copy all GDT entries which the Guest can change. */ |
279 | if (lg->changed & CHANGED_GDT) | 436 | if (lg->changed & CHANGED_GDT) |
280 | copy_gdt(lg, pages->state.guest_gdt); | 437 | copy_gdt(lg, pages->state.guest_gdt); |
281 | /* If only the TLS entries have changed, copy them. */ | 438 | /* If only the TLS entries have changed, copy them. */ |
282 | else if (lg->changed & CHANGED_GDT_TLS) | 439 | else if (lg->changed & CHANGED_GDT_TLS) |
283 | copy_gdt_tls(lg, pages->state.guest_gdt); | 440 | copy_gdt_tls(lg, pages->state.guest_gdt); |
284 | 441 | ||
442 | /* Mark the Guest as unchanged for next time. */ | ||
285 | lg->changed = 0; | 443 | lg->changed = 0; |
286 | } | 444 | } |
287 | 445 | ||
446 | /* Finally: the code to actually call into the Switcher to run the Guest. */ | ||
288 | static void run_guest_once(struct lguest *lg, struct lguest_pages *pages) | 447 | static void run_guest_once(struct lguest *lg, struct lguest_pages *pages) |
289 | { | 448 | { |
449 | /* This is a dummy value we need for GCC's sake. */ | ||
290 | unsigned int clobber; | 450 | unsigned int clobber; |
291 | 451 | ||
452 | /* Copy the guest-specific information into this CPU's "struct | ||
453 | * lguest_pages". */ | ||
292 | copy_in_guest_info(lg, pages); | 454 | copy_in_guest_info(lg, pages); |
293 | 455 | ||
294 | /* Put eflags on stack, lcall does rest: suitable for iret return. */ | 456 | /* Now: we push the "eflags" register on the stack, then do an "lcall". |
457 | * This is how we change from using the kernel code segment to using | ||
458 | * the dedicated lguest code segment, as well as jumping into the | ||
459 | * Switcher. | ||
460 | * | ||
461 | * The lcall also pushes the old code segment (KERNEL_CS) onto the | ||
462 | * stack, then the address of this call. This stack layout happens to | ||
463 | * exactly match the stack of an interrupt... */ | ||
295 | asm volatile("pushf; lcall *lguest_entry" | 464 | asm volatile("pushf; lcall *lguest_entry" |
465 | /* This is how we tell GCC that %eax ("a") and %ebx ("b") | ||
466 | * are changed by this routine. The "=" means output. */ | ||
296 | : "=a"(clobber), "=b"(clobber) | 467 | : "=a"(clobber), "=b"(clobber) |
468 | /* %eax contains the pages pointer. ("0" refers to the | ||
469 | * 0-th argument above, ie "a"). %ebx contains the | ||
470 | * physical address of the Guest's top-level page | ||
471 | * directory. */ | ||
297 | : "0"(pages), "1"(__pa(lg->pgdirs[lg->pgdidx].pgdir)) | 472 | : "0"(pages), "1"(__pa(lg->pgdirs[lg->pgdidx].pgdir)) |
473 | /* We tell gcc that all these registers could change, | ||
474 | * which means we don't have to save and restore them in | ||
475 | * the Switcher. */ | ||
298 | : "memory", "%edx", "%ecx", "%edi", "%esi"); | 476 | : "memory", "%edx", "%ecx", "%edi", "%esi"); |
299 | } | 477 | } |
478 | /*:*/ | ||
300 | 479 | ||
480 | /*H:030 Let's jump straight to the the main loop which runs the Guest. | ||
481 | * Remember, this is called by the Launcher reading /dev/lguest, and we keep | ||
482 | * going around and around until something interesting happens. */ | ||
301 | int run_guest(struct lguest *lg, unsigned long __user *user) | 483 | int run_guest(struct lguest *lg, unsigned long __user *user) |
302 | { | 484 | { |
485 | /* We stop running once the Guest is dead. */ | ||
303 | while (!lg->dead) { | 486 | while (!lg->dead) { |
487 | /* We need to initialize this, otherwise gcc complains. It's | ||
488 | * not (yet) clever enough to see that it's initialized when we | ||
489 | * need it. */ | ||
304 | unsigned int cr2 = 0; /* Damn gcc */ | 490 | unsigned int cr2 = 0; /* Damn gcc */ |
305 | 491 | ||
306 | /* Hypercalls first: we might have been out to userspace */ | 492 | /* First we run any hypercalls the Guest wants done: either in |
493 | * the hypercall ring in "struct lguest_data", or directly by | ||
494 | * using int 31 (LGUEST_TRAP_ENTRY). */ | ||
307 | do_hypercalls(lg); | 495 | do_hypercalls(lg); |
496 | /* It's possible the Guest did a SEND_DMA hypercall to the | ||
497 | * Launcher, in which case we return from the read() now. */ | ||
308 | if (lg->dma_is_pending) { | 498 | if (lg->dma_is_pending) { |
309 | if (put_user(lg->pending_dma, user) || | 499 | if (put_user(lg->pending_dma, user) || |
310 | put_user(lg->pending_key, user+1)) | 500 | put_user(lg->pending_key, user+1)) |
@@ -312,6 +502,7 @@ int run_guest(struct lguest *lg, unsigned long __user *user) | |||
312 | return sizeof(unsigned long)*2; | 502 | return sizeof(unsigned long)*2; |
313 | } | 503 | } |
314 | 504 | ||
505 | /* Check for signals */ | ||
315 | if (signal_pending(current)) | 506 | if (signal_pending(current)) |
316 | return -ERESTARTSYS; | 507 | return -ERESTARTSYS; |
317 | 508 | ||
@@ -319,77 +510,154 @@ int run_guest(struct lguest *lg, unsigned long __user *user) | |||
319 | if (lg->break_out) | 510 | if (lg->break_out) |
320 | return -EAGAIN; | 511 | return -EAGAIN; |
321 | 512 | ||
513 | /* Check if there are any interrupts which can be delivered | ||
514 | * now: if so, this sets up the hander to be executed when we | ||
515 | * next run the Guest. */ | ||
322 | maybe_do_interrupt(lg); | 516 | maybe_do_interrupt(lg); |
323 | 517 | ||
518 | /* All long-lived kernel loops need to check with this horrible | ||
519 | * thing called the freezer. If the Host is trying to suspend, | ||
520 | * it stops us. */ | ||
324 | try_to_freeze(); | 521 | try_to_freeze(); |
325 | 522 | ||
523 | /* Just make absolutely sure the Guest is still alive. One of | ||
524 | * those hypercalls could have been fatal, for example. */ | ||
326 | if (lg->dead) | 525 | if (lg->dead) |
327 | break; | 526 | break; |
328 | 527 | ||
528 | /* If the Guest asked to be stopped, we sleep. The Guest's | ||
529 | * clock timer or LHCALL_BREAK from the Waker will wake us. */ | ||
329 | if (lg->halted) { | 530 | if (lg->halted) { |
330 | set_current_state(TASK_INTERRUPTIBLE); | 531 | set_current_state(TASK_INTERRUPTIBLE); |
331 | schedule(); | 532 | schedule(); |
332 | continue; | 533 | continue; |
333 | } | 534 | } |
334 | 535 | ||
536 | /* OK, now we're ready to jump into the Guest. First we put up | ||
537 | * the "Do Not Disturb" sign: */ | ||
335 | local_irq_disable(); | 538 | local_irq_disable(); |
336 | 539 | ||
337 | /* Even if *we* don't want FPU trap, guest might... */ | 540 | /* Remember the awfully-named TS bit? If the Guest has asked |
541 | * to set it we set it now, so we can trap and pass that trap | ||
542 | * to the Guest if it uses the FPU. */ | ||
338 | if (lg->ts) | 543 | if (lg->ts) |
339 | set_ts(); | 544 | set_ts(); |
340 | 545 | ||
341 | /* Don't let Guest do SYSENTER: we can't handle it. */ | 546 | /* SYSENTER is an optimized way of doing system calls. We |
547 | * can't allow it because it always jumps to privilege level 0. | ||
548 | * A normal Guest won't try it because we don't advertise it in | ||
549 | * CPUID, but a malicious Guest (or malicious Guest userspace | ||
550 | * program) could, so we tell the CPU to disable it before | ||
551 | * running the Guest. */ | ||
342 | if (boot_cpu_has(X86_FEATURE_SEP)) | 552 | if (boot_cpu_has(X86_FEATURE_SEP)) |
343 | wrmsr(MSR_IA32_SYSENTER_CS, 0, 0); | 553 | wrmsr(MSR_IA32_SYSENTER_CS, 0, 0); |
344 | 554 | ||
555 | /* Now we actually run the Guest. It will pop back out when | ||
556 | * something interesting happens, and we can examine its | ||
557 | * registers to see what it was doing. */ | ||
345 | run_guest_once(lg, lguest_pages(raw_smp_processor_id())); | 558 | run_guest_once(lg, lguest_pages(raw_smp_processor_id())); |
346 | 559 | ||
347 | /* Save cr2 now if we page-faulted. */ | 560 | /* The "regs" pointer contains two extra entries which are not |
561 | * really registers: a trap number which says what interrupt or | ||
562 | * trap made the switcher code come back, and an error code | ||
563 | * which some traps set. */ | ||
564 | |||
565 | /* If the Guest page faulted, then the cr2 register will tell | ||
566 | * us the bad virtual address. We have to grab this now, | ||
567 | * because once we re-enable interrupts an interrupt could | ||
568 | * fault and thus overwrite cr2, or we could even move off to a | ||
569 | * different CPU. */ | ||
348 | if (lg->regs->trapnum == 14) | 570 | if (lg->regs->trapnum == 14) |
349 | cr2 = read_cr2(); | 571 | cr2 = read_cr2(); |
572 | /* Similarly, if we took a trap because the Guest used the FPU, | ||
573 | * we have to restore the FPU it expects to see. */ | ||
350 | else if (lg->regs->trapnum == 7) | 574 | else if (lg->regs->trapnum == 7) |
351 | math_state_restore(); | 575 | math_state_restore(); |
352 | 576 | ||
577 | /* Restore SYSENTER if it's supposed to be on. */ | ||
353 | if (boot_cpu_has(X86_FEATURE_SEP)) | 578 | if (boot_cpu_has(X86_FEATURE_SEP)) |
354 | wrmsr(MSR_IA32_SYSENTER_CS, __KERNEL_CS, 0); | 579 | wrmsr(MSR_IA32_SYSENTER_CS, __KERNEL_CS, 0); |
580 | |||
581 | /* Now we're ready to be interrupted or moved to other CPUs */ | ||
355 | local_irq_enable(); | 582 | local_irq_enable(); |
356 | 583 | ||
584 | /* OK, so what happened? */ | ||
357 | switch (lg->regs->trapnum) { | 585 | switch (lg->regs->trapnum) { |
358 | case 13: /* We've intercepted a GPF. */ | 586 | case 13: /* We've intercepted a GPF. */ |
587 | /* Check if this was one of those annoying IN or OUT | ||
588 | * instructions which we need to emulate. If so, we | ||
589 | * just go back into the Guest after we've done it. */ | ||
359 | if (lg->regs->errcode == 0) { | 590 | if (lg->regs->errcode == 0) { |
360 | if (emulate_insn(lg)) | 591 | if (emulate_insn(lg)) |
361 | continue; | 592 | continue; |
362 | } | 593 | } |
363 | break; | 594 | break; |
364 | case 14: /* We've intercepted a page fault. */ | 595 | case 14: /* We've intercepted a page fault. */ |
596 | /* The Guest accessed a virtual address that wasn't | ||
597 | * mapped. This happens a lot: we don't actually set | ||
598 | * up most of the page tables for the Guest at all when | ||
599 | * we start: as it runs it asks for more and more, and | ||
600 | * we set them up as required. In this case, we don't | ||
601 | * even tell the Guest that the fault happened. | ||
602 | * | ||
603 | * The errcode tells whether this was a read or a | ||
604 | * write, and whether kernel or userspace code. */ | ||
365 | if (demand_page(lg, cr2, lg->regs->errcode)) | 605 | if (demand_page(lg, cr2, lg->regs->errcode)) |
366 | continue; | 606 | continue; |
367 | 607 | ||
368 | /* If lguest_data is NULL, this won't hurt. */ | 608 | /* OK, it's really not there (or not OK): the Guest |
609 | * needs to know. We write out the cr2 value so it | ||
610 | * knows where the fault occurred. | ||
611 | * | ||
612 | * Note that if the Guest were really messed up, this | ||
613 | * could happen before it's done the INITIALIZE | ||
614 | * hypercall, so lg->lguest_data will be NULL, so | ||
615 | * &lg->lguest_data->cr2 will be address 8. Writing | ||
616 | * into that address won't hurt the Host at all, | ||
617 | * though. */ | ||
369 | if (put_user(cr2, &lg->lguest_data->cr2)) | 618 | if (put_user(cr2, &lg->lguest_data->cr2)) |
370 | kill_guest(lg, "Writing cr2"); | 619 | kill_guest(lg, "Writing cr2"); |
371 | break; | 620 | break; |
372 | case 7: /* We've intercepted a Device Not Available fault. */ | 621 | case 7: /* We've intercepted a Device Not Available fault. */ |
373 | /* If they don't want to know, just absorb it. */ | 622 | /* If the Guest doesn't want to know, we already |
623 | * restored the Floating Point Unit, so we just | ||
624 | * continue without telling it. */ | ||
374 | if (!lg->ts) | 625 | if (!lg->ts) |
375 | continue; | 626 | continue; |
376 | break; | 627 | break; |
377 | case 32 ... 255: /* Real interrupt, fall thru */ | 628 | case 32 ... 255: |
629 | /* These values mean a real interrupt occurred, in | ||
630 | * which case the Host handler has already been run. | ||
631 | * We just do a friendly check if another process | ||
632 | * should now be run, then fall through to loop | ||
633 | * around: */ | ||
378 | cond_resched(); | 634 | cond_resched(); |
379 | case LGUEST_TRAP_ENTRY: /* Handled at top of loop */ | 635 | case LGUEST_TRAP_ENTRY: /* Handled at top of loop */ |
380 | continue; | 636 | continue; |
381 | } | 637 | } |
382 | 638 | ||
639 | /* If we get here, it's a trap the Guest wants to know | ||
640 | * about. */ | ||
383 | if (deliver_trap(lg, lg->regs->trapnum)) | 641 | if (deliver_trap(lg, lg->regs->trapnum)) |
384 | continue; | 642 | continue; |
385 | 643 | ||
644 | /* If the Guest doesn't have a handler (either it hasn't | ||
645 | * registered any yet, or it's one of the faults we don't let | ||
646 | * it handle), it dies with a cryptic error message. */ | ||
386 | kill_guest(lg, "unhandled trap %li at %#lx (%#lx)", | 647 | kill_guest(lg, "unhandled trap %li at %#lx (%#lx)", |
387 | lg->regs->trapnum, lg->regs->eip, | 648 | lg->regs->trapnum, lg->regs->eip, |
388 | lg->regs->trapnum == 14 ? cr2 : lg->regs->errcode); | 649 | lg->regs->trapnum == 14 ? cr2 : lg->regs->errcode); |
389 | } | 650 | } |
651 | /* The Guest is dead => "No such file or directory" */ | ||
390 | return -ENOENT; | 652 | return -ENOENT; |
391 | } | 653 | } |
392 | 654 | ||
655 | /* Now we can look at each of the routines this calls, in increasing order of | ||
656 | * complexity: do_hypercalls(), emulate_insn(), maybe_do_interrupt(), | ||
657 | * deliver_trap() and demand_page(). After all those, we'll be ready to | ||
658 | * examine the Switcher, and our philosophical understanding of the Host/Guest | ||
659 | * duality will be complete. :*/ | ||
660 | |||
393 | int find_free_guest(void) | 661 | int find_free_guest(void) |
394 | { | 662 | { |
395 | unsigned int i; | 663 | unsigned int i; |
@@ -407,55 +675,96 @@ static void adjust_pge(void *on) | |||
407 | write_cr4(read_cr4() & ~X86_CR4_PGE); | 675 | write_cr4(read_cr4() & ~X86_CR4_PGE); |
408 | } | 676 | } |
409 | 677 | ||
678 | /*H:000 | ||
679 | * Welcome to the Host! | ||
680 | * | ||
681 | * By this point your brain has been tickled by the Guest code and numbed by | ||
682 | * the Launcher code; prepare for it to be stretched by the Host code. This is | ||
683 | * the heart. Let's begin at the initialization routine for the Host's lg | ||
684 | * module. | ||
685 | */ | ||
410 | static int __init init(void) | 686 | static int __init init(void) |
411 | { | 687 | { |
412 | int err; | 688 | int err; |
413 | 689 | ||
690 | /* Lguest can't run under Xen, VMI or itself. It does Tricky Stuff. */ | ||
414 | if (paravirt_enabled()) { | 691 | if (paravirt_enabled()) { |
415 | printk("lguest is afraid of %s\n", paravirt_ops.name); | 692 | printk("lguest is afraid of %s\n", paravirt_ops.name); |
416 | return -EPERM; | 693 | return -EPERM; |
417 | } | 694 | } |
418 | 695 | ||
696 | /* First we put the Switcher up in very high virtual memory. */ | ||
419 | err = map_switcher(); | 697 | err = map_switcher(); |
420 | if (err) | 698 | if (err) |
421 | return err; | 699 | return err; |
422 | 700 | ||
701 | /* Now we set up the pagetable implementation for the Guests. */ | ||
423 | err = init_pagetables(switcher_page, SHARED_SWITCHER_PAGES); | 702 | err = init_pagetables(switcher_page, SHARED_SWITCHER_PAGES); |
424 | if (err) { | 703 | if (err) { |
425 | unmap_switcher(); | 704 | unmap_switcher(); |
426 | return err; | 705 | return err; |
427 | } | 706 | } |
707 | |||
708 | /* The I/O subsystem needs some things initialized. */ | ||
428 | lguest_io_init(); | 709 | lguest_io_init(); |
429 | 710 | ||
711 | /* /dev/lguest needs to be registered. */ | ||
430 | err = lguest_device_init(); | 712 | err = lguest_device_init(); |
431 | if (err) { | 713 | if (err) { |
432 | free_pagetables(); | 714 | free_pagetables(); |
433 | unmap_switcher(); | 715 | unmap_switcher(); |
434 | return err; | 716 | return err; |
435 | } | 717 | } |
718 | |||
719 | /* Finally, we need to turn off "Page Global Enable". PGE is an | ||
720 | * optimization where page table entries are specially marked to show | ||
721 | * they never change. The Host kernel marks all the kernel pages this | ||
722 | * way because it's always present, even when userspace is running. | ||
723 | * | ||
724 | * Lguest breaks this: unbeknownst to the rest of the Host kernel, we | ||
725 | * switch to the Guest kernel. If you don't disable this on all CPUs, | ||
726 | * you'll get really weird bugs that you'll chase for two days. | ||
727 | * | ||
728 | * I used to turn PGE off every time we switched to the Guest and back | ||
729 | * on when we return, but that slowed the Switcher down noticibly. */ | ||
730 | |||
731 | /* We don't need the complexity of CPUs coming and going while we're | ||
732 | * doing this. */ | ||
436 | lock_cpu_hotplug(); | 733 | lock_cpu_hotplug(); |
437 | if (cpu_has_pge) { /* We have a broader idea of "global". */ | 734 | if (cpu_has_pge) { /* We have a broader idea of "global". */ |
735 | /* Remember that this was originally set (for cleanup). */ | ||
438 | cpu_had_pge = 1; | 736 | cpu_had_pge = 1; |
737 | /* adjust_pge is a helper function which sets or unsets the PGE | ||
738 | * bit on its CPU, depending on the argument (0 == unset). */ | ||
439 | on_each_cpu(adjust_pge, (void *)0, 0, 1); | 739 | on_each_cpu(adjust_pge, (void *)0, 0, 1); |
740 | /* Turn off the feature in the global feature set. */ | ||
440 | clear_bit(X86_FEATURE_PGE, boot_cpu_data.x86_capability); | 741 | clear_bit(X86_FEATURE_PGE, boot_cpu_data.x86_capability); |
441 | } | 742 | } |
442 | unlock_cpu_hotplug(); | 743 | unlock_cpu_hotplug(); |
744 | |||
745 | /* All good! */ | ||
443 | return 0; | 746 | return 0; |
444 | } | 747 | } |
445 | 748 | ||
749 | /* Cleaning up is just the same code, backwards. With a little French. */ | ||
446 | static void __exit fini(void) | 750 | static void __exit fini(void) |
447 | { | 751 | { |
448 | lguest_device_remove(); | 752 | lguest_device_remove(); |
449 | free_pagetables(); | 753 | free_pagetables(); |
450 | unmap_switcher(); | 754 | unmap_switcher(); |
755 | |||
756 | /* If we had PGE before we started, turn it back on now. */ | ||
451 | lock_cpu_hotplug(); | 757 | lock_cpu_hotplug(); |
452 | if (cpu_had_pge) { | 758 | if (cpu_had_pge) { |
453 | set_bit(X86_FEATURE_PGE, boot_cpu_data.x86_capability); | 759 | set_bit(X86_FEATURE_PGE, boot_cpu_data.x86_capability); |
760 | /* adjust_pge's argument "1" means set PGE. */ | ||
454 | on_each_cpu(adjust_pge, (void *)1, 0, 1); | 761 | on_each_cpu(adjust_pge, (void *)1, 0, 1); |
455 | } | 762 | } |
456 | unlock_cpu_hotplug(); | 763 | unlock_cpu_hotplug(); |
457 | } | 764 | } |
458 | 765 | ||
766 | /* The Host side of lguest can be a module. This is a nice way for people to | ||
767 | * play with it. */ | ||
459 | module_init(init); | 768 | module_init(init); |
460 | module_exit(fini); | 769 | module_exit(fini); |
461 | MODULE_LICENSE("GPL"); | 770 | MODULE_LICENSE("GPL"); |
diff --git a/drivers/lguest/hypercalls.c b/drivers/lguest/hypercalls.c index ea52ca451f74..7a5299f9679d 100644 --- a/drivers/lguest/hypercalls.c +++ b/drivers/lguest/hypercalls.c | |||
@@ -1,5 +1,10 @@ | |||
1 | /* Actual hypercalls, which allow guests to actually do something. | 1 | /*P:500 Just as userspace programs request kernel operations through a system |
2 | Copyright (C) 2006 Rusty Russell IBM Corporation | 2 | * call, the Guest requests Host operations through a "hypercall". You might |
3 | * notice this nomenclature doesn't really follow any logic, but the name has | ||
4 | * been around for long enough that we're stuck with it. As you'd expect, this | ||
5 | * code is basically a one big switch statement. :*/ | ||
6 | |||
7 | /* Copyright (C) 2006 Rusty Russell IBM Corporation | ||
3 | 8 | ||
4 | This program is free software; you can redistribute it and/or modify | 9 | This program is free software; you can redistribute it and/or modify |
5 | it under the terms of the GNU General Public License as published by | 10 | it under the terms of the GNU General Public License as published by |
@@ -23,37 +28,63 @@ | |||
23 | #include <irq_vectors.h> | 28 | #include <irq_vectors.h> |
24 | #include "lg.h" | 29 | #include "lg.h" |
25 | 30 | ||
31 | /*H:120 This is the core hypercall routine: where the Guest gets what it | ||
32 | * wants. Or gets killed. Or, in the case of LHCALL_CRASH, both. | ||
33 | * | ||
34 | * Remember from the Guest: %eax == which call to make, and the arguments are | ||
35 | * packed into %edx, %ebx and %ecx if needed. */ | ||
26 | static void do_hcall(struct lguest *lg, struct lguest_regs *regs) | 36 | static void do_hcall(struct lguest *lg, struct lguest_regs *regs) |
27 | { | 37 | { |
28 | switch (regs->eax) { | 38 | switch (regs->eax) { |
29 | case LHCALL_FLUSH_ASYNC: | 39 | case LHCALL_FLUSH_ASYNC: |
40 | /* This call does nothing, except by breaking out of the Guest | ||
41 | * it makes us process all the asynchronous hypercalls. */ | ||
30 | break; | 42 | break; |
31 | case LHCALL_LGUEST_INIT: | 43 | case LHCALL_LGUEST_INIT: |
44 | /* You can't get here unless you're already initialized. Don't | ||
45 | * do that. */ | ||
32 | kill_guest(lg, "already have lguest_data"); | 46 | kill_guest(lg, "already have lguest_data"); |
33 | break; | 47 | break; |
34 | case LHCALL_CRASH: { | 48 | case LHCALL_CRASH: { |
49 | /* Crash is such a trivial hypercall that we do it in four | ||
50 | * lines right here. */ | ||
35 | char msg[128]; | 51 | char msg[128]; |
52 | /* If the lgread fails, it will call kill_guest() itself; the | ||
53 | * kill_guest() with the message will be ignored. */ | ||
36 | lgread(lg, msg, regs->edx, sizeof(msg)); | 54 | lgread(lg, msg, regs->edx, sizeof(msg)); |
37 | msg[sizeof(msg)-1] = '\0'; | 55 | msg[sizeof(msg)-1] = '\0'; |
38 | kill_guest(lg, "CRASH: %s", msg); | 56 | kill_guest(lg, "CRASH: %s", msg); |
39 | break; | 57 | break; |
40 | } | 58 | } |
41 | case LHCALL_FLUSH_TLB: | 59 | case LHCALL_FLUSH_TLB: |
60 | /* FLUSH_TLB comes in two flavors, depending on the | ||
61 | * argument: */ | ||
42 | if (regs->edx) | 62 | if (regs->edx) |
43 | guest_pagetable_clear_all(lg); | 63 | guest_pagetable_clear_all(lg); |
44 | else | 64 | else |
45 | guest_pagetable_flush_user(lg); | 65 | guest_pagetable_flush_user(lg); |
46 | break; | 66 | break; |
47 | case LHCALL_GET_WALLCLOCK: { | 67 | case LHCALL_GET_WALLCLOCK: { |
68 | /* The Guest wants to know the real time in seconds since 1970, | ||
69 | * in good Unix tradition. */ | ||
48 | struct timespec ts; | 70 | struct timespec ts; |
49 | ktime_get_real_ts(&ts); | 71 | ktime_get_real_ts(&ts); |
50 | regs->eax = ts.tv_sec; | 72 | regs->eax = ts.tv_sec; |
51 | break; | 73 | break; |
52 | } | 74 | } |
53 | case LHCALL_BIND_DMA: | 75 | case LHCALL_BIND_DMA: |
76 | /* BIND_DMA really wants four arguments, but it's the only call | ||
77 | * which does. So the Guest packs the number of buffers and | ||
78 | * the interrupt number into the final argument, and we decode | ||
79 | * it here. This can legitimately fail, since we currently | ||
80 | * place a limit on the number of DMA pools a Guest can have. | ||
81 | * So we return true or false from this call. */ | ||
54 | regs->eax = bind_dma(lg, regs->edx, regs->ebx, | 82 | regs->eax = bind_dma(lg, regs->edx, regs->ebx, |
55 | regs->ecx >> 8, regs->ecx & 0xFF); | 83 | regs->ecx >> 8, regs->ecx & 0xFF); |
56 | break; | 84 | break; |
85 | |||
86 | /* All these calls simply pass the arguments through to the right | ||
87 | * routines. */ | ||
57 | case LHCALL_SEND_DMA: | 88 | case LHCALL_SEND_DMA: |
58 | send_dma(lg, regs->edx, regs->ebx); | 89 | send_dma(lg, regs->edx, regs->ebx); |
59 | break; | 90 | break; |
@@ -81,10 +112,13 @@ static void do_hcall(struct lguest *lg, struct lguest_regs *regs) | |||
81 | case LHCALL_SET_CLOCKEVENT: | 112 | case LHCALL_SET_CLOCKEVENT: |
82 | guest_set_clockevent(lg, regs->edx); | 113 | guest_set_clockevent(lg, regs->edx); |
83 | break; | 114 | break; |
115 | |||
84 | case LHCALL_TS: | 116 | case LHCALL_TS: |
117 | /* This sets the TS flag, as we saw used in run_guest(). */ | ||
85 | lg->ts = regs->edx; | 118 | lg->ts = regs->edx; |
86 | break; | 119 | break; |
87 | case LHCALL_HALT: | 120 | case LHCALL_HALT: |
121 | /* Similarly, this sets the halted flag for run_guest(). */ | ||
88 | lg->halted = 1; | 122 | lg->halted = 1; |
89 | break; | 123 | break; |
90 | default: | 124 | default: |
@@ -92,25 +126,42 @@ static void do_hcall(struct lguest *lg, struct lguest_regs *regs) | |||
92 | } | 126 | } |
93 | } | 127 | } |
94 | 128 | ||
95 | /* We always do queued calls before actual hypercall. */ | 129 | /* Asynchronous hypercalls are easy: we just look in the array in the Guest's |
130 | * "struct lguest_data" and see if there are any new ones marked "ready". | ||
131 | * | ||
132 | * We are careful to do these in order: obviously we respect the order the | ||
133 | * Guest put them in the ring, but we also promise the Guest that they will | ||
134 | * happen before any normal hypercall (which is why we check this before | ||
135 | * checking for a normal hcall). */ | ||
96 | static void do_async_hcalls(struct lguest *lg) | 136 | static void do_async_hcalls(struct lguest *lg) |
97 | { | 137 | { |
98 | unsigned int i; | 138 | unsigned int i; |
99 | u8 st[LHCALL_RING_SIZE]; | 139 | u8 st[LHCALL_RING_SIZE]; |
100 | 140 | ||
141 | /* For simplicity, we copy the entire call status array in at once. */ | ||
101 | if (copy_from_user(&st, &lg->lguest_data->hcall_status, sizeof(st))) | 142 | if (copy_from_user(&st, &lg->lguest_data->hcall_status, sizeof(st))) |
102 | return; | 143 | return; |
103 | 144 | ||
145 | |||
146 | /* We process "struct lguest_data"s hcalls[] ring once. */ | ||
104 | for (i = 0; i < ARRAY_SIZE(st); i++) { | 147 | for (i = 0; i < ARRAY_SIZE(st); i++) { |
105 | struct lguest_regs regs; | 148 | struct lguest_regs regs; |
149 | /* We remember where we were up to from last time. This makes | ||
150 | * sure that the hypercalls are done in the order the Guest | ||
151 | * places them in the ring. */ | ||
106 | unsigned int n = lg->next_hcall; | 152 | unsigned int n = lg->next_hcall; |
107 | 153 | ||
154 | /* 0xFF means there's no call here (yet). */ | ||
108 | if (st[n] == 0xFF) | 155 | if (st[n] == 0xFF) |
109 | break; | 156 | break; |
110 | 157 | ||
158 | /* OK, we have hypercall. Increment the "next_hcall" cursor, | ||
159 | * and wrap back to 0 if we reach the end. */ | ||
111 | if (++lg->next_hcall == LHCALL_RING_SIZE) | 160 | if (++lg->next_hcall == LHCALL_RING_SIZE) |
112 | lg->next_hcall = 0; | 161 | lg->next_hcall = 0; |
113 | 162 | ||
163 | /* We copy the hypercall arguments into a fake register | ||
164 | * structure. This makes life simple for do_hcall(). */ | ||
114 | if (get_user(regs.eax, &lg->lguest_data->hcalls[n].eax) | 165 | if (get_user(regs.eax, &lg->lguest_data->hcalls[n].eax) |
115 | || get_user(regs.edx, &lg->lguest_data->hcalls[n].edx) | 166 | || get_user(regs.edx, &lg->lguest_data->hcalls[n].edx) |
116 | || get_user(regs.ecx, &lg->lguest_data->hcalls[n].ecx) | 167 | || get_user(regs.ecx, &lg->lguest_data->hcalls[n].ecx) |
@@ -119,74 +170,126 @@ static void do_async_hcalls(struct lguest *lg) | |||
119 | break; | 170 | break; |
120 | } | 171 | } |
121 | 172 | ||
173 | /* Do the hypercall, same as a normal one. */ | ||
122 | do_hcall(lg, ®s); | 174 | do_hcall(lg, ®s); |
175 | |||
176 | /* Mark the hypercall done. */ | ||
123 | if (put_user(0xFF, &lg->lguest_data->hcall_status[n])) { | 177 | if (put_user(0xFF, &lg->lguest_data->hcall_status[n])) { |
124 | kill_guest(lg, "Writing result for async hypercall"); | 178 | kill_guest(lg, "Writing result for async hypercall"); |
125 | break; | 179 | break; |
126 | } | 180 | } |
127 | 181 | ||
182 | /* Stop doing hypercalls if we've just done a DMA to the | ||
183 | * Launcher: it needs to service this first. */ | ||
128 | if (lg->dma_is_pending) | 184 | if (lg->dma_is_pending) |
129 | break; | 185 | break; |
130 | } | 186 | } |
131 | } | 187 | } |
132 | 188 | ||
189 | /* Last of all, we look at what happens first of all. The very first time the | ||
190 | * Guest makes a hypercall, we end up here to set things up: */ | ||
133 | static void initialize(struct lguest *lg) | 191 | static void initialize(struct lguest *lg) |
134 | { | 192 | { |
135 | u32 tsc_speed; | 193 | u32 tsc_speed; |
136 | 194 | ||
195 | /* You can't do anything until you're initialized. The Guest knows the | ||
196 | * rules, so we're unforgiving here. */ | ||
137 | if (lg->regs->eax != LHCALL_LGUEST_INIT) { | 197 | if (lg->regs->eax != LHCALL_LGUEST_INIT) { |
138 | kill_guest(lg, "hypercall %li before LGUEST_INIT", | 198 | kill_guest(lg, "hypercall %li before LGUEST_INIT", |
139 | lg->regs->eax); | 199 | lg->regs->eax); |
140 | return; | 200 | return; |
141 | } | 201 | } |
142 | 202 | ||
143 | /* We only tell the guest to use the TSC if it's reliable. */ | 203 | /* We insist that the Time Stamp Counter exist and doesn't change with |
204 | * cpu frequency. Some devious chip manufacturers decided that TSC | ||
205 | * changes could be handled in software. I decided that time going | ||
206 | * backwards might be good for benchmarks, but it's bad for users. | ||
207 | * | ||
208 | * We also insist that the TSC be stable: the kernel detects unreliable | ||
209 | * TSCs for its own purposes, and we use that here. */ | ||
144 | if (boot_cpu_has(X86_FEATURE_CONSTANT_TSC) && !check_tsc_unstable()) | 210 | if (boot_cpu_has(X86_FEATURE_CONSTANT_TSC) && !check_tsc_unstable()) |
145 | tsc_speed = tsc_khz; | 211 | tsc_speed = tsc_khz; |
146 | else | 212 | else |
147 | tsc_speed = 0; | 213 | tsc_speed = 0; |
148 | 214 | ||
215 | /* The pointer to the Guest's "struct lguest_data" is the only | ||
216 | * argument. */ | ||
149 | lg->lguest_data = (struct lguest_data __user *)lg->regs->edx; | 217 | lg->lguest_data = (struct lguest_data __user *)lg->regs->edx; |
150 | /* We check here so we can simply copy_to_user/from_user */ | 218 | /* If we check the address they gave is OK now, we can simply |
219 | * copy_to_user/from_user from now on rather than using lgread/lgwrite. | ||
220 | * I put this in to show that I'm not immune to writing stupid | ||
221 | * optimizations. */ | ||
151 | if (!lguest_address_ok(lg, lg->regs->edx, sizeof(*lg->lguest_data))) { | 222 | if (!lguest_address_ok(lg, lg->regs->edx, sizeof(*lg->lguest_data))) { |
152 | kill_guest(lg, "bad guest page %p", lg->lguest_data); | 223 | kill_guest(lg, "bad guest page %p", lg->lguest_data); |
153 | return; | 224 | return; |
154 | } | 225 | } |
226 | /* The Guest tells us where we're not to deliver interrupts by putting | ||
227 | * the range of addresses into "struct lguest_data". */ | ||
155 | if (get_user(lg->noirq_start, &lg->lguest_data->noirq_start) | 228 | if (get_user(lg->noirq_start, &lg->lguest_data->noirq_start) |
156 | || get_user(lg->noirq_end, &lg->lguest_data->noirq_end) | 229 | || get_user(lg->noirq_end, &lg->lguest_data->noirq_end) |
157 | /* We reserve the top pgd entry. */ | 230 | /* We tell the Guest that it can't use the top 4MB of virtual |
231 | * addresses used by the Switcher. */ | ||
158 | || put_user(4U*1024*1024, &lg->lguest_data->reserve_mem) | 232 | || put_user(4U*1024*1024, &lg->lguest_data->reserve_mem) |
159 | || put_user(tsc_speed, &lg->lguest_data->tsc_khz) | 233 | || put_user(tsc_speed, &lg->lguest_data->tsc_khz) |
234 | /* We also give the Guest a unique id, as used in lguest_net.c. */ | ||
160 | || put_user(lg->guestid, &lg->lguest_data->guestid)) | 235 | || put_user(lg->guestid, &lg->lguest_data->guestid)) |
161 | kill_guest(lg, "bad guest page %p", lg->lguest_data); | 236 | kill_guest(lg, "bad guest page %p", lg->lguest_data); |
162 | 237 | ||
163 | /* This is the one case where the above accesses might have | 238 | /* This is the one case where the above accesses might have been the |
164 | * been the first write to a Guest page. This may have caused | 239 | * first write to a Guest page. This may have caused a copy-on-write |
165 | * a copy-on-write fault, but the Guest might be referring to | 240 | * fault, but the Guest might be referring to the old (read-only) |
166 | * the old (read-only) page. */ | 241 | * page. */ |
167 | guest_pagetable_clear_all(lg); | 242 | guest_pagetable_clear_all(lg); |
168 | } | 243 | } |
244 | /* Now we've examined the hypercall code; our Guest can make requests. There | ||
245 | * is one other way we can do things for the Guest, as we see in | ||
246 | * emulate_insn(). */ | ||
169 | 247 | ||
170 | /* Even if we go out to userspace and come back, we don't want to do | 248 | /*H:110 Tricky point: we mark the hypercall as "done" once we've done it. |
171 | * the hypercall again. */ | 249 | * Normally we don't need to do this: the Guest will run again and update the |
250 | * trap number before we come back around the run_guest() loop to | ||
251 | * do_hypercalls(). | ||
252 | * | ||
253 | * However, if we are signalled or the Guest sends DMA to the Launcher, that | ||
254 | * loop will exit without running the Guest. When it comes back it would try | ||
255 | * to re-run the hypercall. */ | ||
172 | static void clear_hcall(struct lguest *lg) | 256 | static void clear_hcall(struct lguest *lg) |
173 | { | 257 | { |
174 | lg->regs->trapnum = 255; | 258 | lg->regs->trapnum = 255; |
175 | } | 259 | } |
176 | 260 | ||
261 | /*H:100 | ||
262 | * Hypercalls | ||
263 | * | ||
264 | * Remember from the Guest, hypercalls come in two flavors: normal and | ||
265 | * asynchronous. This file handles both of types. | ||
266 | */ | ||
177 | void do_hypercalls(struct lguest *lg) | 267 | void do_hypercalls(struct lguest *lg) |
178 | { | 268 | { |
269 | /* Not initialized yet? */ | ||
179 | if (unlikely(!lg->lguest_data)) { | 270 | if (unlikely(!lg->lguest_data)) { |
271 | /* Did the Guest make a hypercall? We might have come back for | ||
272 | * some other reason (an interrupt, a different trap). */ | ||
180 | if (lg->regs->trapnum == LGUEST_TRAP_ENTRY) { | 273 | if (lg->regs->trapnum == LGUEST_TRAP_ENTRY) { |
274 | /* Set up the "struct lguest_data" */ | ||
181 | initialize(lg); | 275 | initialize(lg); |
276 | /* The hypercall is done. */ | ||
182 | clear_hcall(lg); | 277 | clear_hcall(lg); |
183 | } | 278 | } |
184 | return; | 279 | return; |
185 | } | 280 | } |
186 | 281 | ||
282 | /* The Guest has initialized. | ||
283 | * | ||
284 | * Look in the hypercall ring for the async hypercalls: */ | ||
187 | do_async_hcalls(lg); | 285 | do_async_hcalls(lg); |
286 | |||
287 | /* If we stopped reading the hypercall ring because the Guest did a | ||
288 | * SEND_DMA to the Launcher, we want to return now. Otherwise if the | ||
289 | * Guest asked us to do a hypercall, we do it. */ | ||
188 | if (!lg->dma_is_pending && lg->regs->trapnum == LGUEST_TRAP_ENTRY) { | 290 | if (!lg->dma_is_pending && lg->regs->trapnum == LGUEST_TRAP_ENTRY) { |
189 | do_hcall(lg, lg->regs); | 291 | do_hcall(lg, lg->regs); |
292 | /* The hypercall is done. */ | ||
190 | clear_hcall(lg); | 293 | clear_hcall(lg); |
191 | } | 294 | } |
192 | } | 295 | } |
diff --git a/drivers/lguest/interrupts_and_traps.c b/drivers/lguest/interrupts_and_traps.c index bee029bb2c7b..bd0091bf79ec 100644 --- a/drivers/lguest/interrupts_and_traps.c +++ b/drivers/lguest/interrupts_and_traps.c | |||
@@ -1,100 +1,160 @@ | |||
1 | /*P:800 Interrupts (traps) are complicated enough to earn their own file. | ||
2 | * There are three classes of interrupts: | ||
3 | * | ||
4 | * 1) Real hardware interrupts which occur while we're running the Guest, | ||
5 | * 2) Interrupts for virtual devices attached to the Guest, and | ||
6 | * 3) Traps and faults from the Guest. | ||
7 | * | ||
8 | * Real hardware interrupts must be delivered to the Host, not the Guest. | ||
9 | * Virtual interrupts must be delivered to the Guest, but we make them look | ||
10 | * just like real hardware would deliver them. Traps from the Guest can be set | ||
11 | * up to go directly back into the Guest, but sometimes the Host wants to see | ||
12 | * them first, so we also have a way of "reflecting" them into the Guest as if | ||
13 | * they had been delivered to it directly. :*/ | ||
1 | #include <linux/uaccess.h> | 14 | #include <linux/uaccess.h> |
2 | #include "lg.h" | 15 | #include "lg.h" |
3 | 16 | ||
17 | /* The address of the interrupt handler is split into two bits: */ | ||
4 | static unsigned long idt_address(u32 lo, u32 hi) | 18 | static unsigned long idt_address(u32 lo, u32 hi) |
5 | { | 19 | { |
6 | return (lo & 0x0000FFFF) | (hi & 0xFFFF0000); | 20 | return (lo & 0x0000FFFF) | (hi & 0xFFFF0000); |
7 | } | 21 | } |
8 | 22 | ||
23 | /* The "type" of the interrupt handler is a 4 bit field: we only support a | ||
24 | * couple of types. */ | ||
9 | static int idt_type(u32 lo, u32 hi) | 25 | static int idt_type(u32 lo, u32 hi) |
10 | { | 26 | { |
11 | return (hi >> 8) & 0xF; | 27 | return (hi >> 8) & 0xF; |
12 | } | 28 | } |
13 | 29 | ||
30 | /* An IDT entry can't be used unless the "present" bit is set. */ | ||
14 | static int idt_present(u32 lo, u32 hi) | 31 | static int idt_present(u32 lo, u32 hi) |
15 | { | 32 | { |
16 | return (hi & 0x8000); | 33 | return (hi & 0x8000); |
17 | } | 34 | } |
18 | 35 | ||
36 | /* We need a helper to "push" a value onto the Guest's stack, since that's a | ||
37 | * big part of what delivering an interrupt does. */ | ||
19 | static void push_guest_stack(struct lguest *lg, unsigned long *gstack, u32 val) | 38 | static void push_guest_stack(struct lguest *lg, unsigned long *gstack, u32 val) |
20 | { | 39 | { |
40 | /* Stack grows upwards: move stack then write value. */ | ||
21 | *gstack -= 4; | 41 | *gstack -= 4; |
22 | lgwrite_u32(lg, *gstack, val); | 42 | lgwrite_u32(lg, *gstack, val); |
23 | } | 43 | } |
24 | 44 | ||
45 | /*H:210 The set_guest_interrupt() routine actually delivers the interrupt or | ||
46 | * trap. The mechanics of delivering traps and interrupts to the Guest are the | ||
47 | * same, except some traps have an "error code" which gets pushed onto the | ||
48 | * stack as well: the caller tells us if this is one. | ||
49 | * | ||
50 | * "lo" and "hi" are the two parts of the Interrupt Descriptor Table for this | ||
51 | * interrupt or trap. It's split into two parts for traditional reasons: gcc | ||
52 | * on i386 used to be frightened by 64 bit numbers. | ||
53 | * | ||
54 | * We set up the stack just like the CPU does for a real interrupt, so it's | ||
55 | * identical for the Guest (and the standard "iret" instruction will undo | ||
56 | * it). */ | ||
25 | static void set_guest_interrupt(struct lguest *lg, u32 lo, u32 hi, int has_err) | 57 | static void set_guest_interrupt(struct lguest *lg, u32 lo, u32 hi, int has_err) |
26 | { | 58 | { |
27 | unsigned long gstack; | 59 | unsigned long gstack; |
28 | u32 eflags, ss, irq_enable; | 60 | u32 eflags, ss, irq_enable; |
29 | 61 | ||
30 | /* If they want a ring change, we use new stack and push old ss/esp */ | 62 | /* There are two cases for interrupts: one where the Guest is already |
63 | * in the kernel, and a more complex one where the Guest is in | ||
64 | * userspace. We check the privilege level to find out. */ | ||
31 | if ((lg->regs->ss&0x3) != GUEST_PL) { | 65 | if ((lg->regs->ss&0x3) != GUEST_PL) { |
66 | /* The Guest told us their kernel stack with the SET_STACK | ||
67 | * hypercall: both the virtual address and the segment */ | ||
32 | gstack = guest_pa(lg, lg->esp1); | 68 | gstack = guest_pa(lg, lg->esp1); |
33 | ss = lg->ss1; | 69 | ss = lg->ss1; |
70 | /* We push the old stack segment and pointer onto the new | ||
71 | * stack: when the Guest does an "iret" back from the interrupt | ||
72 | * handler the CPU will notice they're dropping privilege | ||
73 | * levels and expect these here. */ | ||
34 | push_guest_stack(lg, &gstack, lg->regs->ss); | 74 | push_guest_stack(lg, &gstack, lg->regs->ss); |
35 | push_guest_stack(lg, &gstack, lg->regs->esp); | 75 | push_guest_stack(lg, &gstack, lg->regs->esp); |
36 | } else { | 76 | } else { |
77 | /* We're staying on the same Guest (kernel) stack. */ | ||
37 | gstack = guest_pa(lg, lg->regs->esp); | 78 | gstack = guest_pa(lg, lg->regs->esp); |
38 | ss = lg->regs->ss; | 79 | ss = lg->regs->ss; |
39 | } | 80 | } |
40 | 81 | ||
41 | /* We use IF bit in eflags to indicate whether irqs were enabled | 82 | /* Remember that we never let the Guest actually disable interrupts, so |
42 | (it's always 1, since irqs are enabled when guest is running). */ | 83 | * the "Interrupt Flag" bit is always set. We copy that bit from the |
84 | * Guest's "irq_enabled" field into the eflags word: the Guest copies | ||
85 | * it back in "lguest_iret". */ | ||
43 | eflags = lg->regs->eflags; | 86 | eflags = lg->regs->eflags; |
44 | if (get_user(irq_enable, &lg->lguest_data->irq_enabled) == 0 | 87 | if (get_user(irq_enable, &lg->lguest_data->irq_enabled) == 0 |
45 | && !(irq_enable & X86_EFLAGS_IF)) | 88 | && !(irq_enable & X86_EFLAGS_IF)) |
46 | eflags &= ~X86_EFLAGS_IF; | 89 | eflags &= ~X86_EFLAGS_IF; |
47 | 90 | ||
91 | /* An interrupt is expected to push three things on the stack: the old | ||
92 | * "eflags" word, the old code segment, and the old instruction | ||
93 | * pointer. */ | ||
48 | push_guest_stack(lg, &gstack, eflags); | 94 | push_guest_stack(lg, &gstack, eflags); |
49 | push_guest_stack(lg, &gstack, lg->regs->cs); | 95 | push_guest_stack(lg, &gstack, lg->regs->cs); |
50 | push_guest_stack(lg, &gstack, lg->regs->eip); | 96 | push_guest_stack(lg, &gstack, lg->regs->eip); |
51 | 97 | ||
98 | /* For the six traps which supply an error code, we push that, too. */ | ||
52 | if (has_err) | 99 | if (has_err) |
53 | push_guest_stack(lg, &gstack, lg->regs->errcode); | 100 | push_guest_stack(lg, &gstack, lg->regs->errcode); |
54 | 101 | ||
55 | /* Change the real stack so switcher returns to trap handler */ | 102 | /* Now we've pushed all the old state, we change the stack, the code |
103 | * segment and the address to execute. */ | ||
56 | lg->regs->ss = ss; | 104 | lg->regs->ss = ss; |
57 | lg->regs->esp = gstack + lg->page_offset; | 105 | lg->regs->esp = gstack + lg->page_offset; |
58 | lg->regs->cs = (__KERNEL_CS|GUEST_PL); | 106 | lg->regs->cs = (__KERNEL_CS|GUEST_PL); |
59 | lg->regs->eip = idt_address(lo, hi); | 107 | lg->regs->eip = idt_address(lo, hi); |
60 | 108 | ||
61 | /* Disable interrupts for an interrupt gate. */ | 109 | /* There are two kinds of interrupt handlers: 0xE is an "interrupt |
110 | * gate" which expects interrupts to be disabled on entry. */ | ||
62 | if (idt_type(lo, hi) == 0xE) | 111 | if (idt_type(lo, hi) == 0xE) |
63 | if (put_user(0, &lg->lguest_data->irq_enabled)) | 112 | if (put_user(0, &lg->lguest_data->irq_enabled)) |
64 | kill_guest(lg, "Disabling interrupts"); | 113 | kill_guest(lg, "Disabling interrupts"); |
65 | } | 114 | } |
66 | 115 | ||
116 | /*H:200 | ||
117 | * Virtual Interrupts. | ||
118 | * | ||
119 | * maybe_do_interrupt() gets called before every entry to the Guest, to see if | ||
120 | * we should divert the Guest to running an interrupt handler. */ | ||
67 | void maybe_do_interrupt(struct lguest *lg) | 121 | void maybe_do_interrupt(struct lguest *lg) |
68 | { | 122 | { |
69 | unsigned int irq; | 123 | unsigned int irq; |
70 | DECLARE_BITMAP(blk, LGUEST_IRQS); | 124 | DECLARE_BITMAP(blk, LGUEST_IRQS); |
71 | struct desc_struct *idt; | 125 | struct desc_struct *idt; |
72 | 126 | ||
127 | /* If the Guest hasn't even initialized yet, we can do nothing. */ | ||
73 | if (!lg->lguest_data) | 128 | if (!lg->lguest_data) |
74 | return; | 129 | return; |
75 | 130 | ||
76 | /* Mask out any interrupts they have blocked. */ | 131 | /* Take our "irqs_pending" array and remove any interrupts the Guest |
132 | * wants blocked: the result ends up in "blk". */ | ||
77 | if (copy_from_user(&blk, lg->lguest_data->blocked_interrupts, | 133 | if (copy_from_user(&blk, lg->lguest_data->blocked_interrupts, |
78 | sizeof(blk))) | 134 | sizeof(blk))) |
79 | return; | 135 | return; |
80 | 136 | ||
81 | bitmap_andnot(blk, lg->irqs_pending, blk, LGUEST_IRQS); | 137 | bitmap_andnot(blk, lg->irqs_pending, blk, LGUEST_IRQS); |
82 | 138 | ||
139 | /* Find the first interrupt. */ | ||
83 | irq = find_first_bit(blk, LGUEST_IRQS); | 140 | irq = find_first_bit(blk, LGUEST_IRQS); |
141 | /* None? Nothing to do */ | ||
84 | if (irq >= LGUEST_IRQS) | 142 | if (irq >= LGUEST_IRQS) |
85 | return; | 143 | return; |
86 | 144 | ||
145 | /* They may be in the middle of an iret, where they asked us never to | ||
146 | * deliver interrupts. */ | ||
87 | if (lg->regs->eip >= lg->noirq_start && lg->regs->eip < lg->noirq_end) | 147 | if (lg->regs->eip >= lg->noirq_start && lg->regs->eip < lg->noirq_end) |
88 | return; | 148 | return; |
89 | 149 | ||
90 | /* If they're halted, we re-enable interrupts. */ | 150 | /* If they're halted, interrupts restart them. */ |
91 | if (lg->halted) { | 151 | if (lg->halted) { |
92 | /* Re-enable interrupts. */ | 152 | /* Re-enable interrupts. */ |
93 | if (put_user(X86_EFLAGS_IF, &lg->lguest_data->irq_enabled)) | 153 | if (put_user(X86_EFLAGS_IF, &lg->lguest_data->irq_enabled)) |
94 | kill_guest(lg, "Re-enabling interrupts"); | 154 | kill_guest(lg, "Re-enabling interrupts"); |
95 | lg->halted = 0; | 155 | lg->halted = 0; |
96 | } else { | 156 | } else { |
97 | /* Maybe they have interrupts disabled? */ | 157 | /* Otherwise we check if they have interrupts disabled. */ |
98 | u32 irq_enabled; | 158 | u32 irq_enabled; |
99 | if (get_user(irq_enabled, &lg->lguest_data->irq_enabled)) | 159 | if (get_user(irq_enabled, &lg->lguest_data->irq_enabled)) |
100 | irq_enabled = 0; | 160 | irq_enabled = 0; |
@@ -102,112 +162,211 @@ void maybe_do_interrupt(struct lguest *lg) | |||
102 | return; | 162 | return; |
103 | } | 163 | } |
104 | 164 | ||
165 | /* Look at the IDT entry the Guest gave us for this interrupt. The | ||
166 | * first 32 (FIRST_EXTERNAL_VECTOR) entries are for traps, so we skip | ||
167 | * over them. */ | ||
105 | idt = &lg->idt[FIRST_EXTERNAL_VECTOR+irq]; | 168 | idt = &lg->idt[FIRST_EXTERNAL_VECTOR+irq]; |
169 | /* If they don't have a handler (yet?), we just ignore it */ | ||
106 | if (idt_present(idt->a, idt->b)) { | 170 | if (idt_present(idt->a, idt->b)) { |
171 | /* OK, mark it no longer pending and deliver it. */ | ||
107 | clear_bit(irq, lg->irqs_pending); | 172 | clear_bit(irq, lg->irqs_pending); |
173 | /* set_guest_interrupt() takes the interrupt descriptor and a | ||
174 | * flag to say whether this interrupt pushes an error code onto | ||
175 | * the stack as well: virtual interrupts never do. */ | ||
108 | set_guest_interrupt(lg, idt->a, idt->b, 0); | 176 | set_guest_interrupt(lg, idt->a, idt->b, 0); |
109 | } | 177 | } |
110 | } | 178 | } |
111 | 179 | ||
180 | /*H:220 Now we've got the routines to deliver interrupts, delivering traps | ||
181 | * like page fault is easy. The only trick is that Intel decided that some | ||
182 | * traps should have error codes: */ | ||
112 | static int has_err(unsigned int trap) | 183 | static int has_err(unsigned int trap) |
113 | { | 184 | { |
114 | return (trap == 8 || (trap >= 10 && trap <= 14) || trap == 17); | 185 | return (trap == 8 || (trap >= 10 && trap <= 14) || trap == 17); |
115 | } | 186 | } |
116 | 187 | ||
188 | /* deliver_trap() returns true if it could deliver the trap. */ | ||
117 | int deliver_trap(struct lguest *lg, unsigned int num) | 189 | int deliver_trap(struct lguest *lg, unsigned int num) |
118 | { | 190 | { |
119 | u32 lo = lg->idt[num].a, hi = lg->idt[num].b; | 191 | u32 lo = lg->idt[num].a, hi = lg->idt[num].b; |
120 | 192 | ||
193 | /* Early on the Guest hasn't set the IDT entries (or maybe it put a | ||
194 | * bogus one in): if we fail here, the Guest will be killed. */ | ||
121 | if (!idt_present(lo, hi)) | 195 | if (!idt_present(lo, hi)) |
122 | return 0; | 196 | return 0; |
123 | set_guest_interrupt(lg, lo, hi, has_err(num)); | 197 | set_guest_interrupt(lg, lo, hi, has_err(num)); |
124 | return 1; | 198 | return 1; |
125 | } | 199 | } |
126 | 200 | ||
201 | /*H:250 Here's the hard part: returning to the Host every time a trap happens | ||
202 | * and then calling deliver_trap() and re-entering the Guest is slow. | ||
203 | * Particularly because Guest userspace system calls are traps (trap 128). | ||
204 | * | ||
205 | * So we'd like to set up the IDT to tell the CPU to deliver traps directly | ||
206 | * into the Guest. This is possible, but the complexities cause the size of | ||
207 | * this file to double! However, 150 lines of code is worth writing for taking | ||
208 | * system calls down from 1750ns to 270ns. Plus, if lguest didn't do it, all | ||
209 | * the other hypervisors would tease it. | ||
210 | * | ||
211 | * This routine determines if a trap can be delivered directly. */ | ||
127 | static int direct_trap(const struct lguest *lg, | 212 | static int direct_trap(const struct lguest *lg, |
128 | const struct desc_struct *trap, | 213 | const struct desc_struct *trap, |
129 | unsigned int num) | 214 | unsigned int num) |
130 | { | 215 | { |
131 | /* Hardware interrupts don't go to guest (except syscall). */ | 216 | /* Hardware interrupts don't go to the Guest at all (except system |
217 | * call). */ | ||
132 | if (num >= FIRST_EXTERNAL_VECTOR && num != SYSCALL_VECTOR) | 218 | if (num >= FIRST_EXTERNAL_VECTOR && num != SYSCALL_VECTOR) |
133 | return 0; | 219 | return 0; |
134 | 220 | ||
135 | /* We intercept page fault (demand shadow paging & cr2 saving) | 221 | /* The Host needs to see page faults (for shadow paging and to save the |
136 | protection fault (in/out emulation) and device not | 222 | * fault address), general protection faults (in/out emulation) and |
137 | available (TS handling), and hypercall */ | 223 | * device not available (TS handling), and of course, the hypercall |
224 | * trap. */ | ||
138 | if (num == 14 || num == 13 || num == 7 || num == LGUEST_TRAP_ENTRY) | 225 | if (num == 14 || num == 13 || num == 7 || num == LGUEST_TRAP_ENTRY) |
139 | return 0; | 226 | return 0; |
140 | 227 | ||
141 | /* Interrupt gates (0xE) or not present (0x0) can't go direct. */ | 228 | /* Only trap gates (type 15) can go direct to the Guest. Interrupt |
229 | * gates (type 14) disable interrupts as they are entered, which we | ||
230 | * never let the Guest do. Not present entries (type 0x0) also can't | ||
231 | * go direct, of course 8) */ | ||
142 | return idt_type(trap->a, trap->b) == 0xF; | 232 | return idt_type(trap->a, trap->b) == 0xF; |
143 | } | 233 | } |
144 | 234 | /*:*/ | |
235 | |||
236 | /*M:005 The Guest has the ability to turn its interrupt gates into trap gates, | ||
237 | * if it is careful. The Host will let trap gates can go directly to the | ||
238 | * Guest, but the Guest needs the interrupts atomically disabled for an | ||
239 | * interrupt gate. It can do this by pointing the trap gate at instructions | ||
240 | * within noirq_start and noirq_end, where it can safely disable interrupts. */ | ||
241 | |||
242 | /*M:006 The Guests do not use the sysenter (fast system call) instruction, | ||
243 | * because it's hardcoded to enter privilege level 0 and so can't go direct. | ||
244 | * It's about twice as fast as the older "int 0x80" system call, so it might | ||
245 | * still be worthwhile to handle it in the Switcher and lcall down to the | ||
246 | * Guest. The sysenter semantics are hairy tho: search for that keyword in | ||
247 | * entry.S :*/ | ||
248 | |||
249 | /*H:260 When we make traps go directly into the Guest, we need to make sure | ||
250 | * the kernel stack is valid (ie. mapped in the page tables). Otherwise, the | ||
251 | * CPU trying to deliver the trap will fault while trying to push the interrupt | ||
252 | * words on the stack: this is called a double fault, and it forces us to kill | ||
253 | * the Guest. | ||
254 | * | ||
255 | * Which is deeply unfair, because (literally!) it wasn't the Guests' fault. */ | ||
145 | void pin_stack_pages(struct lguest *lg) | 256 | void pin_stack_pages(struct lguest *lg) |
146 | { | 257 | { |
147 | unsigned int i; | 258 | unsigned int i; |
148 | 259 | ||
260 | /* Depending on the CONFIG_4KSTACKS option, the Guest can have one or | ||
261 | * two pages of stack space. */ | ||
149 | for (i = 0; i < lg->stack_pages; i++) | 262 | for (i = 0; i < lg->stack_pages; i++) |
263 | /* The stack grows *upwards*, hence the subtraction */ | ||
150 | pin_page(lg, lg->esp1 - i * PAGE_SIZE); | 264 | pin_page(lg, lg->esp1 - i * PAGE_SIZE); |
151 | } | 265 | } |
152 | 266 | ||
267 | /* Direct traps also mean that we need to know whenever the Guest wants to use | ||
268 | * a different kernel stack, so we can change the IDT entries to use that | ||
269 | * stack. The IDT entries expect a virtual address, so unlike most addresses | ||
270 | * the Guest gives us, the "esp" (stack pointer) value here is virtual, not | ||
271 | * physical. | ||
272 | * | ||
273 | * In Linux each process has its own kernel stack, so this happens a lot: we | ||
274 | * change stacks on each context switch. */ | ||
153 | void guest_set_stack(struct lguest *lg, u32 seg, u32 esp, unsigned int pages) | 275 | void guest_set_stack(struct lguest *lg, u32 seg, u32 esp, unsigned int pages) |
154 | { | 276 | { |
155 | /* You cannot have a stack segment with priv level 0. */ | 277 | /* You are not allowd have a stack segment with privilege level 0: bad |
278 | * Guest! */ | ||
156 | if ((seg & 0x3) != GUEST_PL) | 279 | if ((seg & 0x3) != GUEST_PL) |
157 | kill_guest(lg, "bad stack segment %i", seg); | 280 | kill_guest(lg, "bad stack segment %i", seg); |
281 | /* We only expect one or two stack pages. */ | ||
158 | if (pages > 2) | 282 | if (pages > 2) |
159 | kill_guest(lg, "bad stack pages %u", pages); | 283 | kill_guest(lg, "bad stack pages %u", pages); |
284 | /* Save where the stack is, and how many pages */ | ||
160 | lg->ss1 = seg; | 285 | lg->ss1 = seg; |
161 | lg->esp1 = esp; | 286 | lg->esp1 = esp; |
162 | lg->stack_pages = pages; | 287 | lg->stack_pages = pages; |
288 | /* Make sure the new stack pages are mapped */ | ||
163 | pin_stack_pages(lg); | 289 | pin_stack_pages(lg); |
164 | } | 290 | } |
165 | 291 | ||
166 | /* Set up trap in IDT. */ | 292 | /* All this reference to mapping stacks leads us neatly into the other complex |
293 | * part of the Host: page table handling. */ | ||
294 | |||
295 | /*H:235 This is the routine which actually checks the Guest's IDT entry and | ||
296 | * transfers it into our entry in "struct lguest": */ | ||
167 | static void set_trap(struct lguest *lg, struct desc_struct *trap, | 297 | static void set_trap(struct lguest *lg, struct desc_struct *trap, |
168 | unsigned int num, u32 lo, u32 hi) | 298 | unsigned int num, u32 lo, u32 hi) |
169 | { | 299 | { |
170 | u8 type = idt_type(lo, hi); | 300 | u8 type = idt_type(lo, hi); |
171 | 301 | ||
302 | /* We zero-out a not-present entry */ | ||
172 | if (!idt_present(lo, hi)) { | 303 | if (!idt_present(lo, hi)) { |
173 | trap->a = trap->b = 0; | 304 | trap->a = trap->b = 0; |
174 | return; | 305 | return; |
175 | } | 306 | } |
176 | 307 | ||
308 | /* We only support interrupt and trap gates. */ | ||
177 | if (type != 0xE && type != 0xF) | 309 | if (type != 0xE && type != 0xF) |
178 | kill_guest(lg, "bad IDT type %i", type); | 310 | kill_guest(lg, "bad IDT type %i", type); |
179 | 311 | ||
312 | /* We only copy the handler address, present bit, privilege level and | ||
313 | * type. The privilege level controls where the trap can be triggered | ||
314 | * manually with an "int" instruction. This is usually GUEST_PL, | ||
315 | * except for system calls which userspace can use. */ | ||
180 | trap->a = ((__KERNEL_CS|GUEST_PL)<<16) | (lo&0x0000FFFF); | 316 | trap->a = ((__KERNEL_CS|GUEST_PL)<<16) | (lo&0x0000FFFF); |
181 | trap->b = (hi&0xFFFFEF00); | 317 | trap->b = (hi&0xFFFFEF00); |
182 | } | 318 | } |
183 | 319 | ||
320 | /*H:230 While we're here, dealing with delivering traps and interrupts to the | ||
321 | * Guest, we might as well complete the picture: how the Guest tells us where | ||
322 | * it wants them to go. This would be simple, except making traps fast | ||
323 | * requires some tricks. | ||
324 | * | ||
325 | * We saw the Guest setting Interrupt Descriptor Table (IDT) entries with the | ||
326 | * LHCALL_LOAD_IDT_ENTRY hypercall before: that comes here. */ | ||
184 | void load_guest_idt_entry(struct lguest *lg, unsigned int num, u32 lo, u32 hi) | 327 | void load_guest_idt_entry(struct lguest *lg, unsigned int num, u32 lo, u32 hi) |
185 | { | 328 | { |
186 | /* Guest never handles: NMI, doublefault, hypercall, spurious irq. */ | 329 | /* Guest never handles: NMI, doublefault, spurious interrupt or |
330 | * hypercall. We ignore when it tries to set them. */ | ||
187 | if (num == 2 || num == 8 || num == 15 || num == LGUEST_TRAP_ENTRY) | 331 | if (num == 2 || num == 8 || num == 15 || num == LGUEST_TRAP_ENTRY) |
188 | return; | 332 | return; |
189 | 333 | ||
334 | /* Mark the IDT as changed: next time the Guest runs we'll know we have | ||
335 | * to copy this again. */ | ||
190 | lg->changed |= CHANGED_IDT; | 336 | lg->changed |= CHANGED_IDT; |
337 | |||
338 | /* The IDT which we keep in "struct lguest" only contains 32 entries | ||
339 | * for the traps and LGUEST_IRQS (32) entries for interrupts. We | ||
340 | * ignore attempts to set handlers for higher interrupt numbers, except | ||
341 | * for the system call "interrupt" at 128: we have a special IDT entry | ||
342 | * for that. */ | ||
191 | if (num < ARRAY_SIZE(lg->idt)) | 343 | if (num < ARRAY_SIZE(lg->idt)) |
192 | set_trap(lg, &lg->idt[num], num, lo, hi); | 344 | set_trap(lg, &lg->idt[num], num, lo, hi); |
193 | else if (num == SYSCALL_VECTOR) | 345 | else if (num == SYSCALL_VECTOR) |
194 | set_trap(lg, &lg->syscall_idt, num, lo, hi); | 346 | set_trap(lg, &lg->syscall_idt, num, lo, hi); |
195 | } | 347 | } |
196 | 348 | ||
349 | /* The default entry for each interrupt points into the Switcher routines which | ||
350 | * simply return to the Host. The run_guest() loop will then call | ||
351 | * deliver_trap() to bounce it back into the Guest. */ | ||
197 | static void default_idt_entry(struct desc_struct *idt, | 352 | static void default_idt_entry(struct desc_struct *idt, |
198 | int trap, | 353 | int trap, |
199 | const unsigned long handler) | 354 | const unsigned long handler) |
200 | { | 355 | { |
356 | /* A present interrupt gate. */ | ||
201 | u32 flags = 0x8e00; | 357 | u32 flags = 0x8e00; |
202 | 358 | ||
203 | /* They can't "int" into any of them except hypercall. */ | 359 | /* Set the privilege level on the entry for the hypercall: this allows |
360 | * the Guest to use the "int" instruction to trigger it. */ | ||
204 | if (trap == LGUEST_TRAP_ENTRY) | 361 | if (trap == LGUEST_TRAP_ENTRY) |
205 | flags |= (GUEST_PL << 13); | 362 | flags |= (GUEST_PL << 13); |
206 | 363 | ||
364 | /* Now pack it into the IDT entry in its weird format. */ | ||
207 | idt->a = (LGUEST_CS<<16) | (handler&0x0000FFFF); | 365 | idt->a = (LGUEST_CS<<16) | (handler&0x0000FFFF); |
208 | idt->b = (handler&0xFFFF0000) | flags; | 366 | idt->b = (handler&0xFFFF0000) | flags; |
209 | } | 367 | } |
210 | 368 | ||
369 | /* When the Guest first starts, we put default entries into the IDT. */ | ||
211 | void setup_default_idt_entries(struct lguest_ro_state *state, | 370 | void setup_default_idt_entries(struct lguest_ro_state *state, |
212 | const unsigned long *def) | 371 | const unsigned long *def) |
213 | { | 372 | { |
@@ -217,19 +376,25 @@ void setup_default_idt_entries(struct lguest_ro_state *state, | |||
217 | default_idt_entry(&state->guest_idt[i], i, def[i]); | 376 | default_idt_entry(&state->guest_idt[i], i, def[i]); |
218 | } | 377 | } |
219 | 378 | ||
379 | /*H:240 We don't use the IDT entries in the "struct lguest" directly, instead | ||
380 | * we copy them into the IDT which we've set up for Guests on this CPU, just | ||
381 | * before we run the Guest. This routine does that copy. */ | ||
220 | void copy_traps(const struct lguest *lg, struct desc_struct *idt, | 382 | void copy_traps(const struct lguest *lg, struct desc_struct *idt, |
221 | const unsigned long *def) | 383 | const unsigned long *def) |
222 | { | 384 | { |
223 | unsigned int i; | 385 | unsigned int i; |
224 | 386 | ||
225 | /* All hardware interrupts are same whatever the guest: only the | 387 | /* We can simply copy the direct traps, otherwise we use the default |
226 | * traps might be different. */ | 388 | * ones in the Switcher: they will return to the Host. */ |
227 | for (i = 0; i < FIRST_EXTERNAL_VECTOR; i++) { | 389 | for (i = 0; i < FIRST_EXTERNAL_VECTOR; i++) { |
228 | if (direct_trap(lg, &lg->idt[i], i)) | 390 | if (direct_trap(lg, &lg->idt[i], i)) |
229 | idt[i] = lg->idt[i]; | 391 | idt[i] = lg->idt[i]; |
230 | else | 392 | else |
231 | default_idt_entry(&idt[i], i, def[i]); | 393 | default_idt_entry(&idt[i], i, def[i]); |
232 | } | 394 | } |
395 | |||
396 | /* Don't forget the system call trap! The IDT entries for other | ||
397 | * interupts never change, so no need to copy them. */ | ||
233 | i = SYSCALL_VECTOR; | 398 | i = SYSCALL_VECTOR; |
234 | if (direct_trap(lg, &lg->syscall_idt, i)) | 399 | if (direct_trap(lg, &lg->syscall_idt, i)) |
235 | idt[i] = lg->syscall_idt; | 400 | idt[i] = lg->syscall_idt; |
diff --git a/drivers/lguest/io.c b/drivers/lguest/io.c index c8eb79266991..ea68613b43f6 100644 --- a/drivers/lguest/io.c +++ b/drivers/lguest/io.c | |||
@@ -1,5 +1,9 @@ | |||
1 | /* Simple I/O model for guests, based on shared memory. | 1 | /*P:300 The I/O mechanism in lguest is simple yet flexible, allowing the Guest |
2 | * Copyright (C) 2006 Rusty Russell IBM Corporation | 2 | * to talk to the Launcher or directly to another Guest. It uses familiar |
3 | * concepts of DMA and interrupts, plus some neat code stolen from | ||
4 | * futexes... :*/ | ||
5 | |||
6 | /* Copyright (C) 2006 Rusty Russell IBM Corporation | ||
3 | * | 7 | * |
4 | * This program is free software; you can redistribute it and/or modify | 8 | * This program is free software; you can redistribute it and/or modify |
5 | * it under the terms of the GNU General Public License as published by | 9 | * it under the terms of the GNU General Public License as published by |
@@ -23,8 +27,36 @@ | |||
23 | #include <linux/uaccess.h> | 27 | #include <linux/uaccess.h> |
24 | #include "lg.h" | 28 | #include "lg.h" |
25 | 29 | ||
30 | /*L:300 | ||
31 | * I/O | ||
32 | * | ||
33 | * Getting data in and out of the Guest is quite an art. There are numerous | ||
34 | * ways to do it, and they all suck differently. We try to keep things fairly | ||
35 | * close to "real" hardware so our Guest's drivers don't look like an alien | ||
36 | * visitation in the middle of the Linux code, and yet make sure that Guests | ||
37 | * can talk directly to other Guests, not just the Launcher. | ||
38 | * | ||
39 | * To do this, the Guest gives us a key when it binds or sends DMA buffers. | ||
40 | * The key corresponds to a "physical" address inside the Guest (ie. a virtual | ||
41 | * address inside the Launcher process). We don't, however, use this key | ||
42 | * directly. | ||
43 | * | ||
44 | * We want Guests which share memory to be able to DMA to each other: two | ||
45 | * Launchers can mmap memory the same file, then the Guests can communicate. | ||
46 | * Fortunately, the futex code provides us with a way to get a "union | ||
47 | * futex_key" corresponding to the memory lying at a virtual address: if the | ||
48 | * two processes share memory, the "union futex_key" for that memory will match | ||
49 | * even if the memory is mapped at different addresses in each. So we always | ||
50 | * convert the keys to "union futex_key"s to compare them. | ||
51 | * | ||
52 | * Before we dive into this though, we need to look at another set of helper | ||
53 | * routines used throughout the Host kernel code to access Guest memory. | ||
54 | :*/ | ||
26 | static struct list_head dma_hash[61]; | 55 | static struct list_head dma_hash[61]; |
27 | 56 | ||
57 | /* An unfortunate side effect of the Linux double-linked list implementation is | ||
58 | * that there's no good way to statically initialize an array of linked | ||
59 | * lists. */ | ||
28 | void lguest_io_init(void) | 60 | void lguest_io_init(void) |
29 | { | 61 | { |
30 | unsigned int i; | 62 | unsigned int i; |
@@ -56,6 +88,19 @@ kill: | |||
56 | return 0; | 88 | return 0; |
57 | } | 89 | } |
58 | 90 | ||
91 | /*L:330 This is our hash function, using the wonderful Jenkins hash. | ||
92 | * | ||
93 | * The futex key is a union with three parts: an unsigned long word, a pointer, | ||
94 | * and an int "offset". We could use jhash_2words() which takes three u32s. | ||
95 | * (Ok, the hash functions are great: the naming sucks though). | ||
96 | * | ||
97 | * It's nice to be portable to 64-bit platforms, so we use the more generic | ||
98 | * jhash2(), which takes an array of u32, the number of u32s, and an initial | ||
99 | * u32 to roll in. This is uglier, but breaks down to almost the same code on | ||
100 | * 32-bit platforms like this one. | ||
101 | * | ||
102 | * We want a position in the array, so we modulo ARRAY_SIZE(dma_hash) (ie. 61). | ||
103 | */ | ||
59 | static unsigned int hash(const union futex_key *key) | 104 | static unsigned int hash(const union futex_key *key) |
60 | { | 105 | { |
61 | return jhash2((u32*)&key->both.word, | 106 | return jhash2((u32*)&key->both.word, |
@@ -64,6 +109,9 @@ static unsigned int hash(const union futex_key *key) | |||
64 | % ARRAY_SIZE(dma_hash); | 109 | % ARRAY_SIZE(dma_hash); |
65 | } | 110 | } |
66 | 111 | ||
112 | /* This is a convenience routine to compare two keys. It's a much bemoaned C | ||
113 | * weakness that it doesn't allow '==' on structures or unions, so we have to | ||
114 | * open-code it like this. */ | ||
67 | static inline int key_eq(const union futex_key *a, const union futex_key *b) | 115 | static inline int key_eq(const union futex_key *a, const union futex_key *b) |
68 | { | 116 | { |
69 | return (a->both.word == b->both.word | 117 | return (a->both.word == b->both.word |
@@ -71,22 +119,36 @@ static inline int key_eq(const union futex_key *a, const union futex_key *b) | |||
71 | && a->both.offset == b->both.offset); | 119 | && a->both.offset == b->both.offset); |
72 | } | 120 | } |
73 | 121 | ||
74 | /* Must hold read lock on dmainfo owner's current->mm->mmap_sem */ | 122 | /*L:360 OK, when we need to actually free up a Guest's DMA array we do several |
123 | * things, so we have a convenient function to do it. | ||
124 | * | ||
125 | * The caller must hold a read lock on dmainfo owner's current->mm->mmap_sem | ||
126 | * for the drop_futex_key_refs(). */ | ||
75 | static void unlink_dma(struct lguest_dma_info *dmainfo) | 127 | static void unlink_dma(struct lguest_dma_info *dmainfo) |
76 | { | 128 | { |
129 | /* You locked this too, right? */ | ||
77 | BUG_ON(!mutex_is_locked(&lguest_lock)); | 130 | BUG_ON(!mutex_is_locked(&lguest_lock)); |
131 | /* This is how we know that the entry is free. */ | ||
78 | dmainfo->interrupt = 0; | 132 | dmainfo->interrupt = 0; |
133 | /* Remove it from the hash table. */ | ||
79 | list_del(&dmainfo->list); | 134 | list_del(&dmainfo->list); |
135 | /* Drop the references we were holding (to the inode or mm). */ | ||
80 | drop_futex_key_refs(&dmainfo->key); | 136 | drop_futex_key_refs(&dmainfo->key); |
81 | } | 137 | } |
82 | 138 | ||
139 | /*L:350 This is the routine which we call when the Guest asks to unregister a | ||
140 | * DMA array attached to a given key. Returns true if the array was found. */ | ||
83 | static int unbind_dma(struct lguest *lg, | 141 | static int unbind_dma(struct lguest *lg, |
84 | const union futex_key *key, | 142 | const union futex_key *key, |
85 | unsigned long dmas) | 143 | unsigned long dmas) |
86 | { | 144 | { |
87 | int i, ret = 0; | 145 | int i, ret = 0; |
88 | 146 | ||
147 | /* We don't bother with the hash table, just look through all this | ||
148 | * Guest's DMA arrays. */ | ||
89 | for (i = 0; i < LGUEST_MAX_DMA; i++) { | 149 | for (i = 0; i < LGUEST_MAX_DMA; i++) { |
150 | /* In theory it could have more than one array on the same key, | ||
151 | * or one array on multiple keys, so we check both */ | ||
90 | if (key_eq(key, &lg->dma[i].key) && dmas == lg->dma[i].dmas) { | 152 | if (key_eq(key, &lg->dma[i].key) && dmas == lg->dma[i].dmas) { |
91 | unlink_dma(&lg->dma[i]); | 153 | unlink_dma(&lg->dma[i]); |
92 | ret = 1; | 154 | ret = 1; |
@@ -96,51 +158,91 @@ static int unbind_dma(struct lguest *lg, | |||
96 | return ret; | 158 | return ret; |
97 | } | 159 | } |
98 | 160 | ||
161 | /*L:340 BIND_DMA: this is the hypercall which sets up an array of "struct | ||
162 | * lguest_dma" for receiving I/O. | ||
163 | * | ||
164 | * The Guest wants to bind an array of "struct lguest_dma"s to a particular key | ||
165 | * to receive input. This only happens when the Guest is setting up a new | ||
166 | * device, so it doesn't have to be very fast. | ||
167 | * | ||
168 | * It returns 1 on a successful registration (it can fail if we hit the limit | ||
169 | * of registrations for this Guest). | ||
170 | */ | ||
99 | int bind_dma(struct lguest *lg, | 171 | int bind_dma(struct lguest *lg, |
100 | unsigned long ukey, unsigned long dmas, u16 numdmas, u8 interrupt) | 172 | unsigned long ukey, unsigned long dmas, u16 numdmas, u8 interrupt) |
101 | { | 173 | { |
102 | unsigned int i; | 174 | unsigned int i; |
103 | int ret = 0; | 175 | int ret = 0; |
104 | union futex_key key; | 176 | union futex_key key; |
177 | /* Futex code needs the mmap_sem. */ | ||
105 | struct rw_semaphore *fshared = ¤t->mm->mmap_sem; | 178 | struct rw_semaphore *fshared = ¤t->mm->mmap_sem; |
106 | 179 | ||
180 | /* Invalid interrupt? (We could kill the guest here). */ | ||
107 | if (interrupt >= LGUEST_IRQS) | 181 | if (interrupt >= LGUEST_IRQS) |
108 | return 0; | 182 | return 0; |
109 | 183 | ||
184 | /* We need to grab the Big Lguest Lock, because other Guests may be | ||
185 | * trying to look through this Guest's DMAs to send something while | ||
186 | * we're doing this. */ | ||
110 | mutex_lock(&lguest_lock); | 187 | mutex_lock(&lguest_lock); |
111 | down_read(fshared); | 188 | down_read(fshared); |
112 | if (get_futex_key((u32 __user *)ukey, fshared, &key) != 0) { | 189 | if (get_futex_key((u32 __user *)ukey, fshared, &key) != 0) { |
113 | kill_guest(lg, "bad dma key %#lx", ukey); | 190 | kill_guest(lg, "bad dma key %#lx", ukey); |
114 | goto unlock; | 191 | goto unlock; |
115 | } | 192 | } |
193 | |||
194 | /* We want to keep this key valid once we drop mmap_sem, so we have to | ||
195 | * hold a reference. */ | ||
116 | get_futex_key_refs(&key); | 196 | get_futex_key_refs(&key); |
117 | 197 | ||
198 | /* If the Guest specified an interrupt of 0, that means they want to | ||
199 | * unregister this array of "struct lguest_dma"s. */ | ||
118 | if (interrupt == 0) | 200 | if (interrupt == 0) |
119 | ret = unbind_dma(lg, &key, dmas); | 201 | ret = unbind_dma(lg, &key, dmas); |
120 | else { | 202 | else { |
203 | /* Look through this Guest's dma array for an unused entry. */ | ||
121 | for (i = 0; i < LGUEST_MAX_DMA; i++) { | 204 | for (i = 0; i < LGUEST_MAX_DMA; i++) { |
205 | /* If the interrupt is non-zero, the entry is already | ||
206 | * used. */ | ||
122 | if (lg->dma[i].interrupt) | 207 | if (lg->dma[i].interrupt) |
123 | continue; | 208 | continue; |
124 | 209 | ||
210 | /* OK, a free one! Fill on our details. */ | ||
125 | lg->dma[i].dmas = dmas; | 211 | lg->dma[i].dmas = dmas; |
126 | lg->dma[i].num_dmas = numdmas; | 212 | lg->dma[i].num_dmas = numdmas; |
127 | lg->dma[i].next_dma = 0; | 213 | lg->dma[i].next_dma = 0; |
128 | lg->dma[i].key = key; | 214 | lg->dma[i].key = key; |
129 | lg->dma[i].guestid = lg->guestid; | 215 | lg->dma[i].guestid = lg->guestid; |
130 | lg->dma[i].interrupt = interrupt; | 216 | lg->dma[i].interrupt = interrupt; |
217 | |||
218 | /* Now we add it to the hash table: the position | ||
219 | * depends on the futex key that we got. */ | ||
131 | list_add(&lg->dma[i].list, &dma_hash[hash(&key)]); | 220 | list_add(&lg->dma[i].list, &dma_hash[hash(&key)]); |
221 | /* Success! */ | ||
132 | ret = 1; | 222 | ret = 1; |
133 | goto unlock; | 223 | goto unlock; |
134 | } | 224 | } |
135 | } | 225 | } |
226 | /* If we didn't find a slot to put the key in, drop the reference | ||
227 | * again. */ | ||
136 | drop_futex_key_refs(&key); | 228 | drop_futex_key_refs(&key); |
137 | unlock: | 229 | unlock: |
230 | /* Unlock and out. */ | ||
138 | up_read(fshared); | 231 | up_read(fshared); |
139 | mutex_unlock(&lguest_lock); | 232 | mutex_unlock(&lguest_lock); |
140 | return ret; | 233 | return ret; |
141 | } | 234 | } |
142 | 235 | ||
143 | /* lgread from another guest */ | 236 | /*L:385 Note that our routines to access a different Guest's memory are called |
237 | * lgread_other() and lgwrite_other(): these names emphasize that they are only | ||
238 | * used when the Guest is *not* the current Guest. | ||
239 | * | ||
240 | * The interface for copying from another process's memory is called | ||
241 | * access_process_vm(), with a final argument of 0 for a read, and 1 for a | ||
242 | * write. | ||
243 | * | ||
244 | * We need lgread_other() to read the destination Guest's "struct lguest_dma" | ||
245 | * array. */ | ||
144 | static int lgread_other(struct lguest *lg, | 246 | static int lgread_other(struct lguest *lg, |
145 | void *buf, u32 addr, unsigned bytes) | 247 | void *buf, u32 addr, unsigned bytes) |
146 | { | 248 | { |
@@ -153,7 +255,8 @@ static int lgread_other(struct lguest *lg, | |||
153 | return 1; | 255 | return 1; |
154 | } | 256 | } |
155 | 257 | ||
156 | /* lgwrite to another guest */ | 258 | /* "lgwrite()" to another Guest: used to update the destination "used_len" once |
259 | * we've transferred data into the buffer. */ | ||
157 | static int lgwrite_other(struct lguest *lg, u32 addr, | 260 | static int lgwrite_other(struct lguest *lg, u32 addr, |
158 | const void *buf, unsigned bytes) | 261 | const void *buf, unsigned bytes) |
159 | { | 262 | { |
@@ -166,6 +269,15 @@ static int lgwrite_other(struct lguest *lg, u32 addr, | |||
166 | return 1; | 269 | return 1; |
167 | } | 270 | } |
168 | 271 | ||
272 | /*L:400 This is the generic engine which copies from a source "struct | ||
273 | * lguest_dma" from this Guest into another Guest's "struct lguest_dma". The | ||
274 | * destination Guest's pages have already been mapped, as contained in the | ||
275 | * pages array. | ||
276 | * | ||
277 | * If you're wondering if there's a nice "copy from one process to another" | ||
278 | * routine, so was I. But Linux isn't really set up to copy between two | ||
279 | * unrelated processes, so we have to write it ourselves. | ||
280 | */ | ||
169 | static u32 copy_data(struct lguest *srclg, | 281 | static u32 copy_data(struct lguest *srclg, |
170 | const struct lguest_dma *src, | 282 | const struct lguest_dma *src, |
171 | const struct lguest_dma *dst, | 283 | const struct lguest_dma *dst, |
@@ -174,33 +286,59 @@ static u32 copy_data(struct lguest *srclg, | |||
174 | unsigned int totlen, si, di, srcoff, dstoff; | 286 | unsigned int totlen, si, di, srcoff, dstoff; |
175 | void *maddr = NULL; | 287 | void *maddr = NULL; |
176 | 288 | ||
289 | /* We return the total length transferred. */ | ||
177 | totlen = 0; | 290 | totlen = 0; |
291 | |||
292 | /* We keep indexes into the source and destination "struct lguest_dma", | ||
293 | * and an offset within each region. */ | ||
178 | si = di = 0; | 294 | si = di = 0; |
179 | srcoff = dstoff = 0; | 295 | srcoff = dstoff = 0; |
296 | |||
297 | /* We loop until the source or destination is exhausted. */ | ||
180 | while (si < LGUEST_MAX_DMA_SECTIONS && src->len[si] | 298 | while (si < LGUEST_MAX_DMA_SECTIONS && src->len[si] |
181 | && di < LGUEST_MAX_DMA_SECTIONS && dst->len[di]) { | 299 | && di < LGUEST_MAX_DMA_SECTIONS && dst->len[di]) { |
300 | /* We can only transfer the rest of the src buffer, or as much | ||
301 | * as will fit into the destination buffer. */ | ||
182 | u32 len = min(src->len[si] - srcoff, dst->len[di] - dstoff); | 302 | u32 len = min(src->len[si] - srcoff, dst->len[di] - dstoff); |
183 | 303 | ||
304 | /* For systems using "highmem" we need to use kmap() to access | ||
305 | * the page we want. We often use the same page over and over, | ||
306 | * so rather than kmap() it on every loop, we set the maddr | ||
307 | * pointer to NULL when we need to move to the next | ||
308 | * destination page. */ | ||
184 | if (!maddr) | 309 | if (!maddr) |
185 | maddr = kmap(pages[di]); | 310 | maddr = kmap(pages[di]); |
186 | 311 | ||
187 | /* FIXME: This is not completely portable, since | 312 | /* Copy directly from (this Guest's) source address to the |
188 | archs do different things for copy_to_user_page. */ | 313 | * destination Guest's kmap()ed buffer. Note that maddr points |
314 | * to the start of the page: we need to add the offset of the | ||
315 | * destination address and offset within the buffer. */ | ||
316 | |||
317 | /* FIXME: This is not completely portable. I looked at | ||
318 | * copy_to_user_page(), and some arch's seem to need special | ||
319 | * flushes. x86 is fine. */ | ||
189 | if (copy_from_user(maddr + (dst->addr[di] + dstoff)%PAGE_SIZE, | 320 | if (copy_from_user(maddr + (dst->addr[di] + dstoff)%PAGE_SIZE, |
190 | (void __user *)src->addr[si], len) != 0) { | 321 | (void __user *)src->addr[si], len) != 0) { |
322 | /* If a copy failed, it's the source's fault. */ | ||
191 | kill_guest(srclg, "bad address in sending DMA"); | 323 | kill_guest(srclg, "bad address in sending DMA"); |
192 | totlen = 0; | 324 | totlen = 0; |
193 | break; | 325 | break; |
194 | } | 326 | } |
195 | 327 | ||
328 | /* Increment the total and src & dst offsets */ | ||
196 | totlen += len; | 329 | totlen += len; |
197 | srcoff += len; | 330 | srcoff += len; |
198 | dstoff += len; | 331 | dstoff += len; |
332 | |||
333 | /* Presumably we reached the end of the src or dest buffers: */ | ||
199 | if (srcoff == src->len[si]) { | 334 | if (srcoff == src->len[si]) { |
335 | /* Move to the next buffer at offset 0 */ | ||
200 | si++; | 336 | si++; |
201 | srcoff = 0; | 337 | srcoff = 0; |
202 | } | 338 | } |
203 | if (dstoff == dst->len[di]) { | 339 | if (dstoff == dst->len[di]) { |
340 | /* We need to unmap that destination page and reset | ||
341 | * maddr ready for the next one. */ | ||
204 | kunmap(pages[di]); | 342 | kunmap(pages[di]); |
205 | maddr = NULL; | 343 | maddr = NULL; |
206 | di++; | 344 | di++; |
@@ -208,13 +346,15 @@ static u32 copy_data(struct lguest *srclg, | |||
208 | } | 346 | } |
209 | } | 347 | } |
210 | 348 | ||
349 | /* If we still had a page mapped at the end, unmap now. */ | ||
211 | if (maddr) | 350 | if (maddr) |
212 | kunmap(pages[di]); | 351 | kunmap(pages[di]); |
213 | 352 | ||
214 | return totlen; | 353 | return totlen; |
215 | } | 354 | } |
216 | 355 | ||
217 | /* Src is us, ie. current. */ | 356 | /*L:390 This is how we transfer a "struct lguest_dma" from the source Guest |
357 | * (the current Guest which called SEND_DMA) to another Guest. */ | ||
218 | static u32 do_dma(struct lguest *srclg, const struct lguest_dma *src, | 358 | static u32 do_dma(struct lguest *srclg, const struct lguest_dma *src, |
219 | struct lguest *dstlg, const struct lguest_dma *dst) | 359 | struct lguest *dstlg, const struct lguest_dma *dst) |
220 | { | 360 | { |
@@ -222,23 +362,31 @@ static u32 do_dma(struct lguest *srclg, const struct lguest_dma *src, | |||
222 | u32 ret; | 362 | u32 ret; |
223 | struct page *pages[LGUEST_MAX_DMA_SECTIONS]; | 363 | struct page *pages[LGUEST_MAX_DMA_SECTIONS]; |
224 | 364 | ||
365 | /* We check that both source and destination "struct lguest_dma"s are | ||
366 | * within the bounds of the source and destination Guests */ | ||
225 | if (!check_dma_list(dstlg, dst) || !check_dma_list(srclg, src)) | 367 | if (!check_dma_list(dstlg, dst) || !check_dma_list(srclg, src)) |
226 | return 0; | 368 | return 0; |
227 | 369 | ||
228 | /* First get the destination pages */ | 370 | /* We need to map the pages which correspond to each parts of |
371 | * destination buffer. */ | ||
229 | for (i = 0; i < LGUEST_MAX_DMA_SECTIONS; i++) { | 372 | for (i = 0; i < LGUEST_MAX_DMA_SECTIONS; i++) { |
230 | if (dst->len[i] == 0) | 373 | if (dst->len[i] == 0) |
231 | break; | 374 | break; |
375 | /* get_user_pages() is a complicated function, especially since | ||
376 | * we only want a single page. But it works, and returns the | ||
377 | * number of pages. Note that we're holding the destination's | ||
378 | * mmap_sem, as get_user_pages() requires. */ | ||
232 | if (get_user_pages(dstlg->tsk, dstlg->mm, | 379 | if (get_user_pages(dstlg->tsk, dstlg->mm, |
233 | dst->addr[i], 1, 1, 1, pages+i, NULL) | 380 | dst->addr[i], 1, 1, 1, pages+i, NULL) |
234 | != 1) { | 381 | != 1) { |
382 | /* This means the destination gave us a bogus buffer */ | ||
235 | kill_guest(dstlg, "Error mapping DMA pages"); | 383 | kill_guest(dstlg, "Error mapping DMA pages"); |
236 | ret = 0; | 384 | ret = 0; |
237 | goto drop_pages; | 385 | goto drop_pages; |
238 | } | 386 | } |
239 | } | 387 | } |
240 | 388 | ||
241 | /* Now copy until we run out of src or dst. */ | 389 | /* Now copy the data until we run out of src or dst. */ |
242 | ret = copy_data(srclg, src, dst, pages); | 390 | ret = copy_data(srclg, src, dst, pages); |
243 | 391 | ||
244 | drop_pages: | 392 | drop_pages: |
@@ -247,6 +395,11 @@ drop_pages: | |||
247 | return ret; | 395 | return ret; |
248 | } | 396 | } |
249 | 397 | ||
398 | /*L:380 Transferring data from one Guest to another is not as simple as I'd | ||
399 | * like. We've found the "struct lguest_dma_info" bound to the same address as | ||
400 | * the send, we need to copy into it. | ||
401 | * | ||
402 | * This function returns true if the destination array was empty. */ | ||
250 | static int dma_transfer(struct lguest *srclg, | 403 | static int dma_transfer(struct lguest *srclg, |
251 | unsigned long udma, | 404 | unsigned long udma, |
252 | struct lguest_dma_info *dst) | 405 | struct lguest_dma_info *dst) |
@@ -255,15 +408,23 @@ static int dma_transfer(struct lguest *srclg, | |||
255 | struct lguest *dstlg; | 408 | struct lguest *dstlg; |
256 | u32 i, dma = 0; | 409 | u32 i, dma = 0; |
257 | 410 | ||
411 | /* From the "struct lguest_dma_info" we found in the hash, grab the | ||
412 | * Guest. */ | ||
258 | dstlg = &lguests[dst->guestid]; | 413 | dstlg = &lguests[dst->guestid]; |
259 | /* Get our dma list. */ | 414 | /* Read in the source "struct lguest_dma" handed to SEND_DMA. */ |
260 | lgread(srclg, &src_dma, udma, sizeof(src_dma)); | 415 | lgread(srclg, &src_dma, udma, sizeof(src_dma)); |
261 | 416 | ||
262 | /* We can't deadlock against them dmaing to us, because this | 417 | /* We need the destination's mmap_sem, and we already hold the source's |
263 | * is all under the lguest_lock. */ | 418 | * mmap_sem for the futex key lookup. Normally this would suggest that |
419 | * we could deadlock if the destination Guest was trying to send to | ||
420 | * this source Guest at the same time, which is another reason that all | ||
421 | * I/O is done under the big lguest_lock. */ | ||
264 | down_read(&dstlg->mm->mmap_sem); | 422 | down_read(&dstlg->mm->mmap_sem); |
265 | 423 | ||
424 | /* Look through the destination DMA array for an available buffer. */ | ||
266 | for (i = 0; i < dst->num_dmas; i++) { | 425 | for (i = 0; i < dst->num_dmas; i++) { |
426 | /* We keep a "next_dma" pointer which often helps us avoid | ||
427 | * looking at lots of previously-filled entries. */ | ||
267 | dma = (dst->next_dma + i) % dst->num_dmas; | 428 | dma = (dst->next_dma + i) % dst->num_dmas; |
268 | if (!lgread_other(dstlg, &dst_dma, | 429 | if (!lgread_other(dstlg, &dst_dma, |
269 | dst->dmas + dma * sizeof(struct lguest_dma), | 430 | dst->dmas + dma * sizeof(struct lguest_dma), |
@@ -273,30 +434,46 @@ static int dma_transfer(struct lguest *srclg, | |||
273 | if (!dst_dma.used_len) | 434 | if (!dst_dma.used_len) |
274 | break; | 435 | break; |
275 | } | 436 | } |
437 | |||
438 | /* If we found a buffer, we do the actual data copy. */ | ||
276 | if (i != dst->num_dmas) { | 439 | if (i != dst->num_dmas) { |
277 | unsigned long used_lenp; | 440 | unsigned long used_lenp; |
278 | unsigned int ret; | 441 | unsigned int ret; |
279 | 442 | ||
280 | ret = do_dma(srclg, &src_dma, dstlg, &dst_dma); | 443 | ret = do_dma(srclg, &src_dma, dstlg, &dst_dma); |
281 | /* Put used length in src. */ | 444 | /* Put used length in the source "struct lguest_dma"'s used_len |
445 | * field. It's a little tricky to figure out where that is, | ||
446 | * though. */ | ||
282 | lgwrite_u32(srclg, | 447 | lgwrite_u32(srclg, |
283 | udma+offsetof(struct lguest_dma, used_len), ret); | 448 | udma+offsetof(struct lguest_dma, used_len), ret); |
449 | /* Tranferring 0 bytes is OK if the source buffer was empty. */ | ||
284 | if (ret == 0 && src_dma.len[0] != 0) | 450 | if (ret == 0 && src_dma.len[0] != 0) |
285 | goto fail; | 451 | goto fail; |
286 | 452 | ||
287 | /* Make sure destination sees contents before length. */ | 453 | /* The destination Guest might be running on a different CPU: |
454 | * we have to make sure that it will see the "used_len" field | ||
455 | * change to non-zero *after* it sees the data we copied into | ||
456 | * the buffer. Hence a write memory barrier. */ | ||
288 | wmb(); | 457 | wmb(); |
458 | /* Figuring out where the destination's used_len field for this | ||
459 | * "struct lguest_dma" in the array is also a little ugly. */ | ||
289 | used_lenp = dst->dmas | 460 | used_lenp = dst->dmas |
290 | + dma * sizeof(struct lguest_dma) | 461 | + dma * sizeof(struct lguest_dma) |
291 | + offsetof(struct lguest_dma, used_len); | 462 | + offsetof(struct lguest_dma, used_len); |
292 | lgwrite_other(dstlg, used_lenp, &ret, sizeof(ret)); | 463 | lgwrite_other(dstlg, used_lenp, &ret, sizeof(ret)); |
464 | /* Move the cursor for next time. */ | ||
293 | dst->next_dma++; | 465 | dst->next_dma++; |
294 | } | 466 | } |
295 | up_read(&dstlg->mm->mmap_sem); | 467 | up_read(&dstlg->mm->mmap_sem); |
296 | 468 | ||
297 | /* Do this last so dst doesn't simply sleep on lock. */ | 469 | /* We trigger the destination interrupt, even if the destination was |
470 | * empty and we didn't transfer anything: this gives them a chance to | ||
471 | * wake up and refill. */ | ||
298 | set_bit(dst->interrupt, dstlg->irqs_pending); | 472 | set_bit(dst->interrupt, dstlg->irqs_pending); |
473 | /* Wake up the destination process. */ | ||
299 | wake_up_process(dstlg->tsk); | 474 | wake_up_process(dstlg->tsk); |
475 | /* If we passed the last "struct lguest_dma", the receive had no | ||
476 | * buffers left. */ | ||
300 | return i == dst->num_dmas; | 477 | return i == dst->num_dmas; |
301 | 478 | ||
302 | fail: | 479 | fail: |
@@ -304,6 +481,8 @@ fail: | |||
304 | return 0; | 481 | return 0; |
305 | } | 482 | } |
306 | 483 | ||
484 | /*L:370 This is the counter-side to the BIND_DMA hypercall; the SEND_DMA | ||
485 | * hypercall. We find out who's listening, and send to them. */ | ||
307 | void send_dma(struct lguest *lg, unsigned long ukey, unsigned long udma) | 486 | void send_dma(struct lguest *lg, unsigned long ukey, unsigned long udma) |
308 | { | 487 | { |
309 | union futex_key key; | 488 | union futex_key key; |
@@ -313,31 +492,43 @@ void send_dma(struct lguest *lg, unsigned long ukey, unsigned long udma) | |||
313 | again: | 492 | again: |
314 | mutex_lock(&lguest_lock); | 493 | mutex_lock(&lguest_lock); |
315 | down_read(fshared); | 494 | down_read(fshared); |
495 | /* Get the futex key for the key the Guest gave us */ | ||
316 | if (get_futex_key((u32 __user *)ukey, fshared, &key) != 0) { | 496 | if (get_futex_key((u32 __user *)ukey, fshared, &key) != 0) { |
317 | kill_guest(lg, "bad sending DMA key"); | 497 | kill_guest(lg, "bad sending DMA key"); |
318 | goto unlock; | 498 | goto unlock; |
319 | } | 499 | } |
320 | /* Shared mapping? Look for other guests... */ | 500 | /* Since the key must be a multiple of 4, the futex key uses the lower |
501 | * bit of the "offset" field (which would always be 0) to indicate a | ||
502 | * mapping which is shared with other processes (ie. Guests). */ | ||
321 | if (key.shared.offset & 1) { | 503 | if (key.shared.offset & 1) { |
322 | struct lguest_dma_info *i; | 504 | struct lguest_dma_info *i; |
505 | /* Look through the hash for other Guests. */ | ||
323 | list_for_each_entry(i, &dma_hash[hash(&key)], list) { | 506 | list_for_each_entry(i, &dma_hash[hash(&key)], list) { |
507 | /* Don't send to ourselves. */ | ||
324 | if (i->guestid == lg->guestid) | 508 | if (i->guestid == lg->guestid) |
325 | continue; | 509 | continue; |
326 | if (!key_eq(&key, &i->key)) | 510 | if (!key_eq(&key, &i->key)) |
327 | continue; | 511 | continue; |
328 | 512 | ||
513 | /* If dma_transfer() tells us the destination has no | ||
514 | * available buffers, we increment "empty". */ | ||
329 | empty += dma_transfer(lg, udma, i); | 515 | empty += dma_transfer(lg, udma, i); |
330 | break; | 516 | break; |
331 | } | 517 | } |
518 | /* If the destination is empty, we release our locks and | ||
519 | * give the destination Guest a brief chance to restock. */ | ||
332 | if (empty == 1) { | 520 | if (empty == 1) { |
333 | /* Give any recipients one chance to restock. */ | 521 | /* Give any recipients one chance to restock. */ |
334 | up_read(¤t->mm->mmap_sem); | 522 | up_read(¤t->mm->mmap_sem); |
335 | mutex_unlock(&lguest_lock); | 523 | mutex_unlock(&lguest_lock); |
524 | /* Next time, we won't try again. */ | ||
336 | empty++; | 525 | empty++; |
337 | goto again; | 526 | goto again; |
338 | } | 527 | } |
339 | } else { | 528 | } else { |
340 | /* Private mapping: tell our userspace. */ | 529 | /* Private mapping: Guest is sending to its Launcher. We set |
530 | * the "dma_is_pending" flag so that the main loop will exit | ||
531 | * and the Launcher's read() from /dev/lguest will return. */ | ||
341 | lg->dma_is_pending = 1; | 532 | lg->dma_is_pending = 1; |
342 | lg->pending_dma = udma; | 533 | lg->pending_dma = udma; |
343 | lg->pending_key = ukey; | 534 | lg->pending_key = ukey; |
@@ -346,6 +537,7 @@ unlock: | |||
346 | up_read(fshared); | 537 | up_read(fshared); |
347 | mutex_unlock(&lguest_lock); | 538 | mutex_unlock(&lguest_lock); |
348 | } | 539 | } |
540 | /*:*/ | ||
349 | 541 | ||
350 | void release_all_dma(struct lguest *lg) | 542 | void release_all_dma(struct lguest *lg) |
351 | { | 543 | { |
@@ -361,7 +553,18 @@ void release_all_dma(struct lguest *lg) | |||
361 | up_read(&lg->mm->mmap_sem); | 553 | up_read(&lg->mm->mmap_sem); |
362 | } | 554 | } |
363 | 555 | ||
364 | /* Userspace wants a dma buffer from this guest. */ | 556 | /*M:007 We only return a single DMA buffer to the Launcher, but it would be |
557 | * more efficient to return a pointer to the entire array of DMA buffers, which | ||
558 | * it can cache and choose one whenever it wants. | ||
559 | * | ||
560 | * Currently the Launcher uses a write to /dev/lguest, and the return value is | ||
561 | * the address of the DMA structure with the interrupt number placed in | ||
562 | * dma->used_len. If we wanted to return the entire array, we need to return | ||
563 | * the address, array size and interrupt number: this seems to require an | ||
564 | * ioctl(). :*/ | ||
565 | |||
566 | /*L:320 This routine looks for a DMA buffer registered by the Guest on the | ||
567 | * given key (using the BIND_DMA hypercall). */ | ||
365 | unsigned long get_dma_buffer(struct lguest *lg, | 568 | unsigned long get_dma_buffer(struct lguest *lg, |
366 | unsigned long ukey, unsigned long *interrupt) | 569 | unsigned long ukey, unsigned long *interrupt) |
367 | { | 570 | { |
@@ -370,15 +573,29 @@ unsigned long get_dma_buffer(struct lguest *lg, | |||
370 | struct lguest_dma_info *i; | 573 | struct lguest_dma_info *i; |
371 | struct rw_semaphore *fshared = ¤t->mm->mmap_sem; | 574 | struct rw_semaphore *fshared = ¤t->mm->mmap_sem; |
372 | 575 | ||
576 | /* Take the Big Lguest Lock to stop other Guests sending this Guest DMA | ||
577 | * at the same time. */ | ||
373 | mutex_lock(&lguest_lock); | 578 | mutex_lock(&lguest_lock); |
579 | /* To match between Guests sharing the same underlying memory we steal | ||
580 | * code from the futex infrastructure. This requires that we hold the | ||
581 | * "mmap_sem" for our process (the Launcher), and pass it to the futex | ||
582 | * code. */ | ||
374 | down_read(fshared); | 583 | down_read(fshared); |
584 | |||
585 | /* This can fail if it's not a valid address, or if the address is not | ||
586 | * divisible by 4 (the futex code needs that, we don't really). */ | ||
375 | if (get_futex_key((u32 __user *)ukey, fshared, &key) != 0) { | 587 | if (get_futex_key((u32 __user *)ukey, fshared, &key) != 0) { |
376 | kill_guest(lg, "bad registered DMA buffer"); | 588 | kill_guest(lg, "bad registered DMA buffer"); |
377 | goto unlock; | 589 | goto unlock; |
378 | } | 590 | } |
591 | /* Search the hash table for matching entries (the Launcher can only | ||
592 | * send to its own Guest for the moment, so the entry must be for this | ||
593 | * Guest) */ | ||
379 | list_for_each_entry(i, &dma_hash[hash(&key)], list) { | 594 | list_for_each_entry(i, &dma_hash[hash(&key)], list) { |
380 | if (key_eq(&key, &i->key) && i->guestid == lg->guestid) { | 595 | if (key_eq(&key, &i->key) && i->guestid == lg->guestid) { |
381 | unsigned int j; | 596 | unsigned int j; |
597 | /* Look through the registered DMA array for an | ||
598 | * available buffer. */ | ||
382 | for (j = 0; j < i->num_dmas; j++) { | 599 | for (j = 0; j < i->num_dmas; j++) { |
383 | struct lguest_dma dma; | 600 | struct lguest_dma dma; |
384 | 601 | ||
@@ -387,6 +604,8 @@ unsigned long get_dma_buffer(struct lguest *lg, | |||
387 | if (dma.used_len == 0) | 604 | if (dma.used_len == 0) |
388 | break; | 605 | break; |
389 | } | 606 | } |
607 | /* Store the interrupt the Guest wants when the buffer | ||
608 | * is used. */ | ||
390 | *interrupt = i->interrupt; | 609 | *interrupt = i->interrupt; |
391 | break; | 610 | break; |
392 | } | 611 | } |
@@ -396,4 +615,12 @@ unlock: | |||
396 | mutex_unlock(&lguest_lock); | 615 | mutex_unlock(&lguest_lock); |
397 | return ret; | 616 | return ret; |
398 | } | 617 | } |
618 | /*:*/ | ||
399 | 619 | ||
620 | /*L:410 This really has completed the Launcher. Not only have we now finished | ||
621 | * the longest chapter in our journey, but this also means we are over halfway | ||
622 | * through! | ||
623 | * | ||
624 | * Enough prevaricating around the bush: it is time for us to dive into the | ||
625 | * core of the Host, in "make Host". | ||
626 | */ | ||
diff --git a/drivers/lguest/lg.h b/drivers/lguest/lg.h index 3e2ddfbc816e..269116eee85f 100644 --- a/drivers/lguest/lg.h +++ b/drivers/lguest/lg.h | |||
@@ -58,9 +58,18 @@ struct lguest_dma_info | |||
58 | u8 interrupt; /* 0 when not registered */ | 58 | u8 interrupt; /* 0 when not registered */ |
59 | }; | 59 | }; |
60 | 60 | ||
61 | /* We have separate types for the guest's ptes & pgds and the shadow ptes & | 61 | /*H:310 The page-table code owes a great debt of gratitude to Andi Kleen. He |
62 | * pgds. Since this host might use three-level pagetables and the guest and | 62 | * reviewed the original code which used "u32" for all page table entries, and |
63 | * shadow pagetables don't, we can't use the normal pte_t/pgd_t. */ | 63 | * insisted that it would be far clearer with explicit typing. I thought it |
64 | * was overkill, but he was right: it is much clearer than it was before. | ||
65 | * | ||
66 | * We have separate types for the Guest's ptes & pgds and the shadow ptes & | ||
67 | * pgds. There's already a Linux type for these (pte_t and pgd_t) but they | ||
68 | * change depending on kernel config options (PAE). */ | ||
69 | |||
70 | /* Each entry is identical: lower 12 bits of flags and upper 20 bits for the | ||
71 | * "page frame number" (0 == first physical page, etc). They are different | ||
72 | * types so the compiler will warn us if we mix them improperly. */ | ||
64 | typedef union { | 73 | typedef union { |
65 | struct { unsigned flags:12, pfn:20; }; | 74 | struct { unsigned flags:12, pfn:20; }; |
66 | struct { unsigned long val; } raw; | 75 | struct { unsigned long val; } raw; |
@@ -77,8 +86,12 @@ typedef union { | |||
77 | struct { unsigned flags:12, pfn:20; }; | 86 | struct { unsigned flags:12, pfn:20; }; |
78 | struct { unsigned long val; } raw; | 87 | struct { unsigned long val; } raw; |
79 | } gpte_t; | 88 | } gpte_t; |
89 | |||
90 | /* We have two convenient macros to convert a "raw" value as handed to us by | ||
91 | * the Guest into the correct Guest PGD or PTE type. */ | ||
80 | #define mkgpte(_val) ((gpte_t){.raw.val = _val}) | 92 | #define mkgpte(_val) ((gpte_t){.raw.val = _val}) |
81 | #define mkgpgd(_val) ((gpgd_t){.raw.val = _val}) | 93 | #define mkgpgd(_val) ((gpgd_t){.raw.val = _val}) |
94 | /*:*/ | ||
82 | 95 | ||
83 | struct pgdir | 96 | struct pgdir |
84 | { | 97 | { |
@@ -244,6 +257,30 @@ unsigned long get_dma_buffer(struct lguest *lg, unsigned long key, | |||
244 | /* hypercalls.c: */ | 257 | /* hypercalls.c: */ |
245 | void do_hypercalls(struct lguest *lg); | 258 | void do_hypercalls(struct lguest *lg); |
246 | 259 | ||
260 | /*L:035 | ||
261 | * Let's step aside for the moment, to study one important routine that's used | ||
262 | * widely in the Host code. | ||
263 | * | ||
264 | * There are many cases where the Guest does something invalid, like pass crap | ||
265 | * to a hypercall. Since only the Guest kernel can make hypercalls, it's quite | ||
266 | * acceptable to simply terminate the Guest and give the Launcher a nicely | ||
267 | * formatted reason. It's also simpler for the Guest itself, which doesn't | ||
268 | * need to check most hypercalls for "success"; if you're still running, it | ||
269 | * succeeded. | ||
270 | * | ||
271 | * Once this is called, the Guest will never run again, so most Host code can | ||
272 | * call this then continue as if nothing had happened. This means many | ||
273 | * functions don't have to explicitly return an error code, which keeps the | ||
274 | * code simple. | ||
275 | * | ||
276 | * It also means that this can be called more than once: only the first one is | ||
277 | * remembered. The only trick is that we still need to kill the Guest even if | ||
278 | * we can't allocate memory to store the reason. Linux has a neat way of | ||
279 | * packing error codes into invalid pointers, so we use that here. | ||
280 | * | ||
281 | * Like any macro which uses an "if", it is safely wrapped in a run-once "do { | ||
282 | * } while(0)". | ||
283 | */ | ||
247 | #define kill_guest(lg, fmt...) \ | 284 | #define kill_guest(lg, fmt...) \ |
248 | do { \ | 285 | do { \ |
249 | if (!(lg)->dead) { \ | 286 | if (!(lg)->dead) { \ |
@@ -252,6 +289,7 @@ do { \ | |||
252 | (lg)->dead = ERR_PTR(-ENOMEM); \ | 289 | (lg)->dead = ERR_PTR(-ENOMEM); \ |
253 | } \ | 290 | } \ |
254 | } while(0) | 291 | } while(0) |
292 | /* (End of aside) :*/ | ||
255 | 293 | ||
256 | static inline unsigned long guest_pa(struct lguest *lg, unsigned long vaddr) | 294 | static inline unsigned long guest_pa(struct lguest *lg, unsigned long vaddr) |
257 | { | 295 | { |
diff --git a/drivers/lguest/lguest.c b/drivers/lguest/lguest.c index 18dade06d4a9..6dfe568523a2 100644 --- a/drivers/lguest/lguest.c +++ b/drivers/lguest/lguest.c | |||
@@ -1,6 +1,32 @@ | |||
1 | /* | 1 | /*P:010 |
2 | * Lguest specific paravirt-ops implementation | 2 | * A hypervisor allows multiple Operating Systems to run on a single machine. |
3 | * To quote David Wheeler: "Any problem in computer science can be solved with | ||
4 | * another layer of indirection." | ||
5 | * | ||
6 | * We keep things simple in two ways. First, we start with a normal Linux | ||
7 | * kernel and insert a module (lg.ko) which allows us to run other Linux | ||
8 | * kernels the same way we'd run processes. We call the first kernel the Host, | ||
9 | * and the others the Guests. The program which sets up and configures Guests | ||
10 | * (such as the example in Documentation/lguest/lguest.c) is called the | ||
11 | * Launcher. | ||
12 | * | ||
13 | * Secondly, we only run specially modified Guests, not normal kernels. When | ||
14 | * you set CONFIG_LGUEST to 'y' or 'm', this automatically sets | ||
15 | * CONFIG_LGUEST_GUEST=y, which compiles this file into the kernel so it knows | ||
16 | * how to be a Guest. This means that you can use the same kernel you boot | ||
17 | * normally (ie. as a Host) as a Guest. | ||
3 | * | 18 | * |
19 | * These Guests know that they cannot do privileged operations, such as disable | ||
20 | * interrupts, and that they have to ask the Host to do such things explicitly. | ||
21 | * This file consists of all the replacements for such low-level native | ||
22 | * hardware operations: these special Guest versions call the Host. | ||
23 | * | ||
24 | * So how does the kernel know it's a Guest? The Guest starts at a special | ||
25 | * entry point marked with a magic string, which sets up a few things then | ||
26 | * calls here. We replace the native functions in "struct paravirt_ops" | ||
27 | * with our Guest versions, then boot like normal. :*/ | ||
28 | |||
29 | /* | ||
4 | * Copyright (C) 2006, Rusty Russell <rusty@rustcorp.com.au> IBM Corporation. | 30 | * Copyright (C) 2006, Rusty Russell <rusty@rustcorp.com.au> IBM Corporation. |
5 | * | 31 | * |
6 | * This program is free software; you can redistribute it and/or modify | 32 | * This program is free software; you can redistribute it and/or modify |
@@ -40,6 +66,12 @@ | |||
40 | #include <asm/mce.h> | 66 | #include <asm/mce.h> |
41 | #include <asm/io.h> | 67 | #include <asm/io.h> |
42 | 68 | ||
69 | /*G:010 Welcome to the Guest! | ||
70 | * | ||
71 | * The Guest in our tale is a simple creature: identical to the Host but | ||
72 | * behaving in simplified but equivalent ways. In particular, the Guest is the | ||
73 | * same kernel as the Host (or at least, built from the same source code). :*/ | ||
74 | |||
43 | /* Declarations for definitions in lguest_guest.S */ | 75 | /* Declarations for definitions in lguest_guest.S */ |
44 | extern char lguest_noirq_start[], lguest_noirq_end[]; | 76 | extern char lguest_noirq_start[], lguest_noirq_end[]; |
45 | extern const char lgstart_cli[], lgend_cli[]; | 77 | extern const char lgstart_cli[], lgend_cli[]; |
@@ -58,7 +90,26 @@ struct lguest_data lguest_data = { | |||
58 | struct lguest_device_desc *lguest_devices; | 90 | struct lguest_device_desc *lguest_devices; |
59 | static cycle_t clock_base; | 91 | static cycle_t clock_base; |
60 | 92 | ||
61 | static enum paravirt_lazy_mode lazy_mode; | 93 | /*G:035 Notice the lazy_hcall() above, rather than hcall(). This is our first |
94 | * real optimization trick! | ||
95 | * | ||
96 | * When lazy_mode is set, it means we're allowed to defer all hypercalls and do | ||
97 | * them as a batch when lazy_mode is eventually turned off. Because hypercalls | ||
98 | * are reasonably expensive, batching them up makes sense. For example, a | ||
99 | * large mmap might update dozens of page table entries: that code calls | ||
100 | * lguest_lazy_mode(PARAVIRT_LAZY_MMU), does the dozen updates, then calls | ||
101 | * lguest_lazy_mode(PARAVIRT_LAZY_NONE). | ||
102 | * | ||
103 | * So, when we're in lazy mode, we call async_hypercall() to store the call for | ||
104 | * future processing. When lazy mode is turned off we issue a hypercall to | ||
105 | * flush the stored calls. | ||
106 | * | ||
107 | * There's also a hack where "mode" is set to "PARAVIRT_LAZY_FLUSH" which | ||
108 | * indicates we're to flush any outstanding calls immediately. This is used | ||
109 | * when an interrupt handler does a kmap_atomic(): the page table changes must | ||
110 | * happen immediately even if we're in the middle of a batch. Usually we're | ||
111 | * not, though, so there's nothing to do. */ | ||
112 | static enum paravirt_lazy_mode lazy_mode; /* Note: not SMP-safe! */ | ||
62 | static void lguest_lazy_mode(enum paravirt_lazy_mode mode) | 113 | static void lguest_lazy_mode(enum paravirt_lazy_mode mode) |
63 | { | 114 | { |
64 | if (mode == PARAVIRT_LAZY_FLUSH) { | 115 | if (mode == PARAVIRT_LAZY_FLUSH) { |
@@ -82,6 +133,16 @@ static void lazy_hcall(unsigned long call, | |||
82 | async_hcall(call, arg1, arg2, arg3); | 133 | async_hcall(call, arg1, arg2, arg3); |
83 | } | 134 | } |
84 | 135 | ||
136 | /* async_hcall() is pretty simple: I'm quite proud of it really. We have a | ||
137 | * ring buffer of stored hypercalls which the Host will run though next time we | ||
138 | * do a normal hypercall. Each entry in the ring has 4 slots for the hypercall | ||
139 | * arguments, and a "hcall_status" word which is 0 if the call is ready to go, | ||
140 | * and 255 once the Host has finished with it. | ||
141 | * | ||
142 | * If we come around to a slot which hasn't been finished, then the table is | ||
143 | * full and we just make the hypercall directly. This has the nice side | ||
144 | * effect of causing the Host to run all the stored calls in the ring buffer | ||
145 | * which empties it for next time! */ | ||
85 | void async_hcall(unsigned long call, | 146 | void async_hcall(unsigned long call, |
86 | unsigned long arg1, unsigned long arg2, unsigned long arg3) | 147 | unsigned long arg1, unsigned long arg2, unsigned long arg3) |
87 | { | 148 | { |
@@ -89,6 +150,9 @@ void async_hcall(unsigned long call, | |||
89 | static unsigned int next_call; | 150 | static unsigned int next_call; |
90 | unsigned long flags; | 151 | unsigned long flags; |
91 | 152 | ||
153 | /* Disable interrupts if not already disabled: we don't want an | ||
154 | * interrupt handler making a hypercall while we're already doing | ||
155 | * one! */ | ||
92 | local_irq_save(flags); | 156 | local_irq_save(flags); |
93 | if (lguest_data.hcall_status[next_call] != 0xFF) { | 157 | if (lguest_data.hcall_status[next_call] != 0xFF) { |
94 | /* Table full, so do normal hcall which will flush table. */ | 158 | /* Table full, so do normal hcall which will flush table. */ |
@@ -98,7 +162,7 @@ void async_hcall(unsigned long call, | |||
98 | lguest_data.hcalls[next_call].edx = arg1; | 162 | lguest_data.hcalls[next_call].edx = arg1; |
99 | lguest_data.hcalls[next_call].ebx = arg2; | 163 | lguest_data.hcalls[next_call].ebx = arg2; |
100 | lguest_data.hcalls[next_call].ecx = arg3; | 164 | lguest_data.hcalls[next_call].ecx = arg3; |
101 | /* Make sure host sees arguments before "valid" flag. */ | 165 | /* Arguments must all be written before we mark it to go */ |
102 | wmb(); | 166 | wmb(); |
103 | lguest_data.hcall_status[next_call] = 0; | 167 | lguest_data.hcall_status[next_call] = 0; |
104 | if (++next_call == LHCALL_RING_SIZE) | 168 | if (++next_call == LHCALL_RING_SIZE) |
@@ -106,9 +170,14 @@ void async_hcall(unsigned long call, | |||
106 | } | 170 | } |
107 | local_irq_restore(flags); | 171 | local_irq_restore(flags); |
108 | } | 172 | } |
173 | /*:*/ | ||
109 | 174 | ||
175 | /* Wrappers for the SEND_DMA and BIND_DMA hypercalls. This is mainly because | ||
176 | * Jeff Garzik complained that __pa() should never appear in drivers, and this | ||
177 | * helps remove most of them. But also, it wraps some ugliness. */ | ||
110 | void lguest_send_dma(unsigned long key, struct lguest_dma *dma) | 178 | void lguest_send_dma(unsigned long key, struct lguest_dma *dma) |
111 | { | 179 | { |
180 | /* The hcall might not write this if something goes wrong */ | ||
112 | dma->used_len = 0; | 181 | dma->used_len = 0; |
113 | hcall(LHCALL_SEND_DMA, key, __pa(dma), 0); | 182 | hcall(LHCALL_SEND_DMA, key, __pa(dma), 0); |
114 | } | 183 | } |
@@ -116,11 +185,16 @@ void lguest_send_dma(unsigned long key, struct lguest_dma *dma) | |||
116 | int lguest_bind_dma(unsigned long key, struct lguest_dma *dmas, | 185 | int lguest_bind_dma(unsigned long key, struct lguest_dma *dmas, |
117 | unsigned int num, u8 irq) | 186 | unsigned int num, u8 irq) |
118 | { | 187 | { |
188 | /* This is the only hypercall which actually wants 5 arguments, and we | ||
189 | * only support 4. Fortunately the interrupt number is always less | ||
190 | * than 256, so we can pack it with the number of dmas in the final | ||
191 | * argument. */ | ||
119 | if (!hcall(LHCALL_BIND_DMA, key, __pa(dmas), (num << 8) | irq)) | 192 | if (!hcall(LHCALL_BIND_DMA, key, __pa(dmas), (num << 8) | irq)) |
120 | return -ENOMEM; | 193 | return -ENOMEM; |
121 | return 0; | 194 | return 0; |
122 | } | 195 | } |
123 | 196 | ||
197 | /* Unbinding is the same hypercall as binding, but with 0 num & irq. */ | ||
124 | void lguest_unbind_dma(unsigned long key, struct lguest_dma *dmas) | 198 | void lguest_unbind_dma(unsigned long key, struct lguest_dma *dmas) |
125 | { | 199 | { |
126 | hcall(LHCALL_BIND_DMA, key, __pa(dmas), 0); | 200 | hcall(LHCALL_BIND_DMA, key, __pa(dmas), 0); |
@@ -138,35 +212,73 @@ void lguest_unmap(void *addr) | |||
138 | iounmap((__force void __iomem *)addr); | 212 | iounmap((__force void __iomem *)addr); |
139 | } | 213 | } |
140 | 214 | ||
215 | /*G:033 | ||
216 | * Here are our first native-instruction replacements: four functions for | ||
217 | * interrupt control. | ||
218 | * | ||
219 | * The simplest way of implementing these would be to have "turn interrupts | ||
220 | * off" and "turn interrupts on" hypercalls. Unfortunately, this is too slow: | ||
221 | * these are by far the most commonly called functions of those we override. | ||
222 | * | ||
223 | * So instead we keep an "irq_enabled" field inside our "struct lguest_data", | ||
224 | * which the Guest can update with a single instruction. The Host knows to | ||
225 | * check there when it wants to deliver an interrupt. | ||
226 | */ | ||
227 | |||
228 | /* save_flags() is expected to return the processor state (ie. "eflags"). The | ||
229 | * eflags word contains all kind of stuff, but in practice Linux only cares | ||
230 | * about the interrupt flag. Our "save_flags()" just returns that. */ | ||
141 | static unsigned long save_fl(void) | 231 | static unsigned long save_fl(void) |
142 | { | 232 | { |
143 | return lguest_data.irq_enabled; | 233 | return lguest_data.irq_enabled; |
144 | } | 234 | } |
145 | 235 | ||
236 | /* "restore_flags" just sets the flags back to the value given. */ | ||
146 | static void restore_fl(unsigned long flags) | 237 | static void restore_fl(unsigned long flags) |
147 | { | 238 | { |
148 | /* FIXME: Check if interrupt pending... */ | ||
149 | lguest_data.irq_enabled = flags; | 239 | lguest_data.irq_enabled = flags; |
150 | } | 240 | } |
151 | 241 | ||
242 | /* Interrupts go off... */ | ||
152 | static void irq_disable(void) | 243 | static void irq_disable(void) |
153 | { | 244 | { |
154 | lguest_data.irq_enabled = 0; | 245 | lguest_data.irq_enabled = 0; |
155 | } | 246 | } |
156 | 247 | ||
248 | /* Interrupts go on... */ | ||
157 | static void irq_enable(void) | 249 | static void irq_enable(void) |
158 | { | 250 | { |
159 | /* FIXME: Check if interrupt pending... */ | ||
160 | lguest_data.irq_enabled = X86_EFLAGS_IF; | 251 | lguest_data.irq_enabled = X86_EFLAGS_IF; |
161 | } | 252 | } |
162 | 253 | /*:*/ | |
254 | /*M:003 Note that we don't check for outstanding interrupts when we re-enable | ||
255 | * them (or when we unmask an interrupt). This seems to work for the moment, | ||
256 | * since interrupts are rare and we'll just get the interrupt on the next timer | ||
257 | * tick, but when we turn on CONFIG_NO_HZ, we should revisit this. One way | ||
258 | * would be to put the "irq_enabled" field in a page by itself, and have the | ||
259 | * Host write-protect it when an interrupt comes in when irqs are disabled. | ||
260 | * There will then be a page fault as soon as interrupts are re-enabled. :*/ | ||
261 | |||
262 | /*G:034 | ||
263 | * The Interrupt Descriptor Table (IDT). | ||
264 | * | ||
265 | * The IDT tells the processor what to do when an interrupt comes in. Each | ||
266 | * entry in the table is a 64-bit descriptor: this holds the privilege level, | ||
267 | * address of the handler, and... well, who cares? The Guest just asks the | ||
268 | * Host to make the change anyway, because the Host controls the real IDT. | ||
269 | */ | ||
163 | static void lguest_write_idt_entry(struct desc_struct *dt, | 270 | static void lguest_write_idt_entry(struct desc_struct *dt, |
164 | int entrynum, u32 low, u32 high) | 271 | int entrynum, u32 low, u32 high) |
165 | { | 272 | { |
273 | /* Keep the local copy up to date. */ | ||
166 | write_dt_entry(dt, entrynum, low, high); | 274 | write_dt_entry(dt, entrynum, low, high); |
275 | /* Tell Host about this new entry. */ | ||
167 | hcall(LHCALL_LOAD_IDT_ENTRY, entrynum, low, high); | 276 | hcall(LHCALL_LOAD_IDT_ENTRY, entrynum, low, high); |
168 | } | 277 | } |
169 | 278 | ||
279 | /* Changing to a different IDT is very rare: we keep the IDT up-to-date every | ||
280 | * time it is written, so we can simply loop through all entries and tell the | ||
281 | * Host about them. */ | ||
170 | static void lguest_load_idt(const struct Xgt_desc_struct *desc) | 282 | static void lguest_load_idt(const struct Xgt_desc_struct *desc) |
171 | { | 283 | { |
172 | unsigned int i; | 284 | unsigned int i; |
@@ -176,12 +288,29 @@ static void lguest_load_idt(const struct Xgt_desc_struct *desc) | |||
176 | hcall(LHCALL_LOAD_IDT_ENTRY, i, idt[i].a, idt[i].b); | 288 | hcall(LHCALL_LOAD_IDT_ENTRY, i, idt[i].a, idt[i].b); |
177 | } | 289 | } |
178 | 290 | ||
291 | /* | ||
292 | * The Global Descriptor Table. | ||
293 | * | ||
294 | * The Intel architecture defines another table, called the Global Descriptor | ||
295 | * Table (GDT). You tell the CPU where it is (and its size) using the "lgdt" | ||
296 | * instruction, and then several other instructions refer to entries in the | ||
297 | * table. There are three entries which the Switcher needs, so the Host simply | ||
298 | * controls the entire thing and the Guest asks it to make changes using the | ||
299 | * LOAD_GDT hypercall. | ||
300 | * | ||
301 | * This is the opposite of the IDT code where we have a LOAD_IDT_ENTRY | ||
302 | * hypercall and use that repeatedly to load a new IDT. I don't think it | ||
303 | * really matters, but wouldn't it be nice if they were the same? | ||
304 | */ | ||
179 | static void lguest_load_gdt(const struct Xgt_desc_struct *desc) | 305 | static void lguest_load_gdt(const struct Xgt_desc_struct *desc) |
180 | { | 306 | { |
181 | BUG_ON((desc->size+1)/8 != GDT_ENTRIES); | 307 | BUG_ON((desc->size+1)/8 != GDT_ENTRIES); |
182 | hcall(LHCALL_LOAD_GDT, __pa(desc->address), GDT_ENTRIES, 0); | 308 | hcall(LHCALL_LOAD_GDT, __pa(desc->address), GDT_ENTRIES, 0); |
183 | } | 309 | } |
184 | 310 | ||
311 | /* For a single GDT entry which changes, we do the lazy thing: alter our GDT, | ||
312 | * then tell the Host to reload the entire thing. This operation is so rare | ||
313 | * that this naive implementation is reasonable. */ | ||
185 | static void lguest_write_gdt_entry(struct desc_struct *dt, | 314 | static void lguest_write_gdt_entry(struct desc_struct *dt, |
186 | int entrynum, u32 low, u32 high) | 315 | int entrynum, u32 low, u32 high) |
187 | { | 316 | { |
@@ -189,19 +318,58 @@ static void lguest_write_gdt_entry(struct desc_struct *dt, | |||
189 | hcall(LHCALL_LOAD_GDT, __pa(dt), GDT_ENTRIES, 0); | 318 | hcall(LHCALL_LOAD_GDT, __pa(dt), GDT_ENTRIES, 0); |
190 | } | 319 | } |
191 | 320 | ||
321 | /* OK, I lied. There are three "thread local storage" GDT entries which change | ||
322 | * on every context switch (these three entries are how glibc implements | ||
323 | * __thread variables). So we have a hypercall specifically for this case. */ | ||
192 | static void lguest_load_tls(struct thread_struct *t, unsigned int cpu) | 324 | static void lguest_load_tls(struct thread_struct *t, unsigned int cpu) |
193 | { | 325 | { |
194 | lazy_hcall(LHCALL_LOAD_TLS, __pa(&t->tls_array), cpu, 0); | 326 | lazy_hcall(LHCALL_LOAD_TLS, __pa(&t->tls_array), cpu, 0); |
195 | } | 327 | } |
328 | /*:*/ | ||
196 | 329 | ||
330 | /*G:038 That's enough excitement for now, back to ploughing through each of | ||
331 | * the paravirt_ops (we're about 1/3 of the way through). | ||
332 | * | ||
333 | * This is the Local Descriptor Table, another weird Intel thingy. Linux only | ||
334 | * uses this for some strange applications like Wine. We don't do anything | ||
335 | * here, so they'll get an informative and friendly Segmentation Fault. */ | ||
197 | static void lguest_set_ldt(const void *addr, unsigned entries) | 336 | static void lguest_set_ldt(const void *addr, unsigned entries) |
198 | { | 337 | { |
199 | } | 338 | } |
200 | 339 | ||
340 | /* This loads a GDT entry into the "Task Register": that entry points to a | ||
341 | * structure called the Task State Segment. Some comments scattered though the | ||
342 | * kernel code indicate that this used for task switching in ages past, along | ||
343 | * with blood sacrifice and astrology. | ||
344 | * | ||
345 | * Now there's nothing interesting in here that we don't get told elsewhere. | ||
346 | * But the native version uses the "ltr" instruction, which makes the Host | ||
347 | * complain to the Guest about a Segmentation Fault and it'll oops. So we | ||
348 | * override the native version with a do-nothing version. */ | ||
201 | static void lguest_load_tr_desc(void) | 349 | static void lguest_load_tr_desc(void) |
202 | { | 350 | { |
203 | } | 351 | } |
204 | 352 | ||
353 | /* The "cpuid" instruction is a way of querying both the CPU identity | ||
354 | * (manufacturer, model, etc) and its features. It was introduced before the | ||
355 | * Pentium in 1993 and keeps getting extended by both Intel and AMD. As you | ||
356 | * might imagine, after a decade and a half this treatment, it is now a giant | ||
357 | * ball of hair. Its entry in the current Intel manual runs to 28 pages. | ||
358 | * | ||
359 | * This instruction even it has its own Wikipedia entry. The Wikipedia entry | ||
360 | * has been translated into 4 languages. I am not making this up! | ||
361 | * | ||
362 | * We could get funky here and identify ourselves as "GenuineLguest", but | ||
363 | * instead we just use the real "cpuid" instruction. Then I pretty much turned | ||
364 | * off feature bits until the Guest booted. (Don't say that: you'll damage | ||
365 | * lguest sales!) Shut up, inner voice! (Hey, just pointing out that this is | ||
366 | * hardly future proof.) Noone's listening! They don't like you anyway, | ||
367 | * parenthetic weirdo! | ||
368 | * | ||
369 | * Replacing the cpuid so we can turn features off is great for the kernel, but | ||
370 | * anyone (including userspace) can just use the raw "cpuid" instruction and | ||
371 | * the Host won't even notice since it isn't privileged. So we try not to get | ||
372 | * too worked up about it. */ | ||
205 | static void lguest_cpuid(unsigned int *eax, unsigned int *ebx, | 373 | static void lguest_cpuid(unsigned int *eax, unsigned int *ebx, |
206 | unsigned int *ecx, unsigned int *edx) | 374 | unsigned int *ecx, unsigned int *edx) |
207 | { | 375 | { |
@@ -214,21 +382,43 @@ static void lguest_cpuid(unsigned int *eax, unsigned int *ebx, | |||
214 | *ecx &= 0x00002201; | 382 | *ecx &= 0x00002201; |
215 | /* SSE, SSE2, FXSR, MMX, CMOV, CMPXCHG8B, FPU. */ | 383 | /* SSE, SSE2, FXSR, MMX, CMOV, CMPXCHG8B, FPU. */ |
216 | *edx &= 0x07808101; | 384 | *edx &= 0x07808101; |
217 | /* Host wants to know when we flush kernel pages: set PGE. */ | 385 | /* The Host can do a nice optimization if it knows that the |
386 | * kernel mappings (addresses above 0xC0000000 or whatever | ||
387 | * PAGE_OFFSET is set to) haven't changed. But Linux calls | ||
388 | * flush_tlb_user() for both user and kernel mappings unless | ||
389 | * the Page Global Enable (PGE) feature bit is set. */ | ||
218 | *edx |= 0x00002000; | 390 | *edx |= 0x00002000; |
219 | break; | 391 | break; |
220 | case 0x80000000: | 392 | case 0x80000000: |
221 | /* Futureproof this a little: if they ask how much extended | 393 | /* Futureproof this a little: if they ask how much extended |
222 | * processor information, limit it to known fields. */ | 394 | * processor information there is, limit it to known fields. */ |
223 | if (*eax > 0x80000008) | 395 | if (*eax > 0x80000008) |
224 | *eax = 0x80000008; | 396 | *eax = 0x80000008; |
225 | break; | 397 | break; |
226 | } | 398 | } |
227 | } | 399 | } |
228 | 400 | ||
401 | /* Intel has four control registers, imaginatively named cr0, cr2, cr3 and cr4. | ||
402 | * I assume there's a cr1, but it hasn't bothered us yet, so we'll not bother | ||
403 | * it. The Host needs to know when the Guest wants to change them, so we have | ||
404 | * a whole series of functions like read_cr0() and write_cr0(). | ||
405 | * | ||
406 | * We start with CR0. CR0 allows you to turn on and off all kinds of basic | ||
407 | * features, but Linux only really cares about one: the horrifically-named Task | ||
408 | * Switched (TS) bit at bit 3 (ie. 8) | ||
409 | * | ||
410 | * What does the TS bit do? Well, it causes the CPU to trap (interrupt 7) if | ||
411 | * the floating point unit is used. Which allows us to restore FPU state | ||
412 | * lazily after a task switch, and Linux uses that gratefully, but wouldn't a | ||
413 | * name like "FPUTRAP bit" be a little less cryptic? | ||
414 | * | ||
415 | * We store cr0 (and cr3) locally, because the Host never changes it. The | ||
416 | * Guest sometimes wants to read it and we'd prefer not to bother the Host | ||
417 | * unnecessarily. */ | ||
229 | static unsigned long current_cr0, current_cr3; | 418 | static unsigned long current_cr0, current_cr3; |
230 | static void lguest_write_cr0(unsigned long val) | 419 | static void lguest_write_cr0(unsigned long val) |
231 | { | 420 | { |
421 | /* 8 == TS bit. */ | ||
232 | lazy_hcall(LHCALL_TS, val & 8, 0, 0); | 422 | lazy_hcall(LHCALL_TS, val & 8, 0, 0); |
233 | current_cr0 = val; | 423 | current_cr0 = val; |
234 | } | 424 | } |
@@ -238,17 +428,25 @@ static unsigned long lguest_read_cr0(void) | |||
238 | return current_cr0; | 428 | return current_cr0; |
239 | } | 429 | } |
240 | 430 | ||
431 | /* Intel provided a special instruction to clear the TS bit for people too cool | ||
432 | * to use write_cr0() to do it. This "clts" instruction is faster, because all | ||
433 | * the vowels have been optimized out. */ | ||
241 | static void lguest_clts(void) | 434 | static void lguest_clts(void) |
242 | { | 435 | { |
243 | lazy_hcall(LHCALL_TS, 0, 0, 0); | 436 | lazy_hcall(LHCALL_TS, 0, 0, 0); |
244 | current_cr0 &= ~8U; | 437 | current_cr0 &= ~8U; |
245 | } | 438 | } |
246 | 439 | ||
440 | /* CR2 is the virtual address of the last page fault, which the Guest only ever | ||
441 | * reads. The Host kindly writes this into our "struct lguest_data", so we | ||
442 | * just read it out of there. */ | ||
247 | static unsigned long lguest_read_cr2(void) | 443 | static unsigned long lguest_read_cr2(void) |
248 | { | 444 | { |
249 | return lguest_data.cr2; | 445 | return lguest_data.cr2; |
250 | } | 446 | } |
251 | 447 | ||
448 | /* CR3 is the current toplevel pagetable page: the principle is the same as | ||
449 | * cr0. Keep a local copy, and tell the Host when it changes. */ | ||
252 | static void lguest_write_cr3(unsigned long cr3) | 450 | static void lguest_write_cr3(unsigned long cr3) |
253 | { | 451 | { |
254 | lazy_hcall(LHCALL_NEW_PGTABLE, cr3, 0, 0); | 452 | lazy_hcall(LHCALL_NEW_PGTABLE, cr3, 0, 0); |
@@ -260,7 +458,7 @@ static unsigned long lguest_read_cr3(void) | |||
260 | return current_cr3; | 458 | return current_cr3; |
261 | } | 459 | } |
262 | 460 | ||
263 | /* Used to enable/disable PGE, but we don't care. */ | 461 | /* CR4 is used to enable and disable PGE, but we don't care. */ |
264 | static unsigned long lguest_read_cr4(void) | 462 | static unsigned long lguest_read_cr4(void) |
265 | { | 463 | { |
266 | return 0; | 464 | return 0; |
@@ -270,6 +468,59 @@ static void lguest_write_cr4(unsigned long val) | |||
270 | { | 468 | { |
271 | } | 469 | } |
272 | 470 | ||
471 | /* | ||
472 | * Page Table Handling. | ||
473 | * | ||
474 | * Now would be a good time to take a rest and grab a coffee or similarly | ||
475 | * relaxing stimulant. The easy parts are behind us, and the trek gradually | ||
476 | * winds uphill from here. | ||
477 | * | ||
478 | * Quick refresher: memory is divided into "pages" of 4096 bytes each. The CPU | ||
479 | * maps virtual addresses to physical addresses using "page tables". We could | ||
480 | * use one huge index of 1 million entries: each address is 4 bytes, so that's | ||
481 | * 1024 pages just to hold the page tables. But since most virtual addresses | ||
482 | * are unused, we use a two level index which saves space. The CR3 register | ||
483 | * contains the physical address of the top level "page directory" page, which | ||
484 | * contains physical addresses of up to 1024 second-level pages. Each of these | ||
485 | * second level pages contains up to 1024 physical addresses of actual pages, | ||
486 | * or Page Table Entries (PTEs). | ||
487 | * | ||
488 | * Here's a diagram, where arrows indicate physical addresses: | ||
489 | * | ||
490 | * CR3 ---> +---------+ | ||
491 | * | --------->+---------+ | ||
492 | * | | | PADDR1 | | ||
493 | * Top-level | | PADDR2 | | ||
494 | * (PMD) page | | | | ||
495 | * | | Lower-level | | ||
496 | * | | (PTE) page | | ||
497 | * | | | | | ||
498 | * .... .... | ||
499 | * | ||
500 | * So to convert a virtual address to a physical address, we look up the top | ||
501 | * level, which points us to the second level, which gives us the physical | ||
502 | * address of that page. If the top level entry was not present, or the second | ||
503 | * level entry was not present, then the virtual address is invalid (we | ||
504 | * say "the page was not mapped"). | ||
505 | * | ||
506 | * Put another way, a 32-bit virtual address is divided up like so: | ||
507 | * | ||
508 | * 1 1 0 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 | ||
509 | * |<---- 10 bits ---->|<---- 10 bits ---->|<------ 12 bits ------>| | ||
510 | * Index into top Index into second Offset within page | ||
511 | * page directory page pagetable page | ||
512 | * | ||
513 | * The kernel spends a lot of time changing both the top-level page directory | ||
514 | * and lower-level pagetable pages. The Guest doesn't know physical addresses, | ||
515 | * so while it maintains these page tables exactly like normal, it also needs | ||
516 | * to keep the Host informed whenever it makes a change: the Host will create | ||
517 | * the real page tables based on the Guests'. | ||
518 | */ | ||
519 | |||
520 | /* The Guest calls this to set a second-level entry (pte), ie. to map a page | ||
521 | * into a process' address space. We set the entry then tell the Host the | ||
522 | * toplevel and address this corresponds to. The Guest uses one pagetable per | ||
523 | * process, so we need to tell the Host which one we're changing (mm->pgd). */ | ||
273 | static void lguest_set_pte_at(struct mm_struct *mm, unsigned long addr, | 524 | static void lguest_set_pte_at(struct mm_struct *mm, unsigned long addr, |
274 | pte_t *ptep, pte_t pteval) | 525 | pte_t *ptep, pte_t pteval) |
275 | { | 526 | { |
@@ -277,7 +528,9 @@ static void lguest_set_pte_at(struct mm_struct *mm, unsigned long addr, | |||
277 | lazy_hcall(LHCALL_SET_PTE, __pa(mm->pgd), addr, pteval.pte_low); | 528 | lazy_hcall(LHCALL_SET_PTE, __pa(mm->pgd), addr, pteval.pte_low); |
278 | } | 529 | } |
279 | 530 | ||
280 | /* We only support two-level pagetables at the moment. */ | 531 | /* The Guest calls this to set a top-level entry. Again, we set the entry then |
532 | * tell the Host which top-level page we changed, and the index of the entry we | ||
533 | * changed. */ | ||
281 | static void lguest_set_pmd(pmd_t *pmdp, pmd_t pmdval) | 534 | static void lguest_set_pmd(pmd_t *pmdp, pmd_t pmdval) |
282 | { | 535 | { |
283 | *pmdp = pmdval; | 536 | *pmdp = pmdval; |
@@ -285,7 +538,15 @@ static void lguest_set_pmd(pmd_t *pmdp, pmd_t pmdval) | |||
285 | (__pa(pmdp)&(PAGE_SIZE-1))/4, 0); | 538 | (__pa(pmdp)&(PAGE_SIZE-1))/4, 0); |
286 | } | 539 | } |
287 | 540 | ||
288 | /* FIXME: Eliminate all callers of this. */ | 541 | /* There are a couple of legacy places where the kernel sets a PTE, but we |
542 | * don't know the top level any more. This is useless for us, since we don't | ||
543 | * know which pagetable is changing or what address, so we just tell the Host | ||
544 | * to forget all of them. Fortunately, this is very rare. | ||
545 | * | ||
546 | * ... except in early boot when the kernel sets up the initial pagetables, | ||
547 | * which makes booting astonishingly slow. So we don't even tell the Host | ||
548 | * anything changed until we've done the first page table switch. | ||
549 | */ | ||
289 | static void lguest_set_pte(pte_t *ptep, pte_t pteval) | 550 | static void lguest_set_pte(pte_t *ptep, pte_t pteval) |
290 | { | 551 | { |
291 | *ptep = pteval; | 552 | *ptep = pteval; |
@@ -294,22 +555,51 @@ static void lguest_set_pte(pte_t *ptep, pte_t pteval) | |||
294 | lazy_hcall(LHCALL_FLUSH_TLB, 1, 0, 0); | 555 | lazy_hcall(LHCALL_FLUSH_TLB, 1, 0, 0); |
295 | } | 556 | } |
296 | 557 | ||
558 | /* Unfortunately for Lguest, the paravirt_ops for page tables were based on | ||
559 | * native page table operations. On native hardware you can set a new page | ||
560 | * table entry whenever you want, but if you want to remove one you have to do | ||
561 | * a TLB flush (a TLB is a little cache of page table entries kept by the CPU). | ||
562 | * | ||
563 | * So the lguest_set_pte_at() and lguest_set_pmd() functions above are only | ||
564 | * called when a valid entry is written, not when it's removed (ie. marked not | ||
565 | * present). Instead, this is where we come when the Guest wants to remove a | ||
566 | * page table entry: we tell the Host to set that entry to 0 (ie. the present | ||
567 | * bit is zero). */ | ||
297 | static void lguest_flush_tlb_single(unsigned long addr) | 568 | static void lguest_flush_tlb_single(unsigned long addr) |
298 | { | 569 | { |
299 | /* Simply set it to zero, and it will fault back in. */ | 570 | /* Simply set it to zero: if it was not, it will fault back in. */ |
300 | lazy_hcall(LHCALL_SET_PTE, current_cr3, addr, 0); | 571 | lazy_hcall(LHCALL_SET_PTE, current_cr3, addr, 0); |
301 | } | 572 | } |
302 | 573 | ||
574 | /* This is what happens after the Guest has removed a large number of entries. | ||
575 | * This tells the Host that any of the page table entries for userspace might | ||
576 | * have changed, ie. virtual addresses below PAGE_OFFSET. */ | ||
303 | static void lguest_flush_tlb_user(void) | 577 | static void lguest_flush_tlb_user(void) |
304 | { | 578 | { |
305 | lazy_hcall(LHCALL_FLUSH_TLB, 0, 0, 0); | 579 | lazy_hcall(LHCALL_FLUSH_TLB, 0, 0, 0); |
306 | } | 580 | } |
307 | 581 | ||
582 | /* This is called when the kernel page tables have changed. That's not very | ||
583 | * common (unless the Guest is using highmem, which makes the Guest extremely | ||
584 | * slow), so it's worth separating this from the user flushing above. */ | ||
308 | static void lguest_flush_tlb_kernel(void) | 585 | static void lguest_flush_tlb_kernel(void) |
309 | { | 586 | { |
310 | lazy_hcall(LHCALL_FLUSH_TLB, 1, 0, 0); | 587 | lazy_hcall(LHCALL_FLUSH_TLB, 1, 0, 0); |
311 | } | 588 | } |
312 | 589 | ||
590 | /* | ||
591 | * The Unadvanced Programmable Interrupt Controller. | ||
592 | * | ||
593 | * This is an attempt to implement the simplest possible interrupt controller. | ||
594 | * I spent some time looking though routines like set_irq_chip_and_handler, | ||
595 | * set_irq_chip_and_handler_name, set_irq_chip_data and set_phasers_to_stun and | ||
596 | * I *think* this is as simple as it gets. | ||
597 | * | ||
598 | * We can tell the Host what interrupts we want blocked ready for using the | ||
599 | * lguest_data.interrupts bitmap, so disabling (aka "masking") them is as | ||
600 | * simple as setting a bit. We don't actually "ack" interrupts as such, we | ||
601 | * just mask and unmask them. I wonder if we should be cleverer? | ||
602 | */ | ||
313 | static void disable_lguest_irq(unsigned int irq) | 603 | static void disable_lguest_irq(unsigned int irq) |
314 | { | 604 | { |
315 | set_bit(irq, lguest_data.blocked_interrupts); | 605 | set_bit(irq, lguest_data.blocked_interrupts); |
@@ -318,9 +608,9 @@ static void disable_lguest_irq(unsigned int irq) | |||
318 | static void enable_lguest_irq(unsigned int irq) | 608 | static void enable_lguest_irq(unsigned int irq) |
319 | { | 609 | { |
320 | clear_bit(irq, lguest_data.blocked_interrupts); | 610 | clear_bit(irq, lguest_data.blocked_interrupts); |
321 | /* FIXME: If it's pending? */ | ||
322 | } | 611 | } |
323 | 612 | ||
613 | /* This structure describes the lguest IRQ controller. */ | ||
324 | static struct irq_chip lguest_irq_controller = { | 614 | static struct irq_chip lguest_irq_controller = { |
325 | .name = "lguest", | 615 | .name = "lguest", |
326 | .mask = disable_lguest_irq, | 616 | .mask = disable_lguest_irq, |
@@ -328,6 +618,10 @@ static struct irq_chip lguest_irq_controller = { | |||
328 | .unmask = enable_lguest_irq, | 618 | .unmask = enable_lguest_irq, |
329 | }; | 619 | }; |
330 | 620 | ||
621 | /* This sets up the Interrupt Descriptor Table (IDT) entry for each hardware | ||
622 | * interrupt (except 128, which is used for system calls), and then tells the | ||
623 | * Linux infrastructure that each interrupt is controlled by our level-based | ||
624 | * lguest interrupt controller. */ | ||
331 | static void __init lguest_init_IRQ(void) | 625 | static void __init lguest_init_IRQ(void) |
332 | { | 626 | { |
333 | unsigned int i; | 627 | unsigned int i; |
@@ -340,14 +634,24 @@ static void __init lguest_init_IRQ(void) | |||
340 | handle_level_irq); | 634 | handle_level_irq); |
341 | } | 635 | } |
342 | } | 636 | } |
637 | /* This call is required to set up for 4k stacks, where we have | ||
638 | * separate stacks for hard and soft interrupts. */ | ||
343 | irq_ctx_init(smp_processor_id()); | 639 | irq_ctx_init(smp_processor_id()); |
344 | } | 640 | } |
345 | 641 | ||
642 | /* | ||
643 | * Time. | ||
644 | * | ||
645 | * It would be far better for everyone if the Guest had its own clock, but | ||
646 | * until then it must ask the Host for the time. | ||
647 | */ | ||
346 | static unsigned long lguest_get_wallclock(void) | 648 | static unsigned long lguest_get_wallclock(void) |
347 | { | 649 | { |
348 | return hcall(LHCALL_GET_WALLCLOCK, 0, 0, 0); | 650 | return hcall(LHCALL_GET_WALLCLOCK, 0, 0, 0); |
349 | } | 651 | } |
350 | 652 | ||
653 | /* If the Host tells us we can trust the TSC, we use that, otherwise we simply | ||
654 | * use the imprecise but reliable "jiffies" counter. */ | ||
351 | static cycle_t lguest_clock_read(void) | 655 | static cycle_t lguest_clock_read(void) |
352 | { | 656 | { |
353 | if (lguest_data.tsc_khz) | 657 | if (lguest_data.tsc_khz) |
@@ -428,12 +732,19 @@ static void lguest_time_irq(unsigned int irq, struct irq_desc *desc) | |||
428 | local_irq_restore(flags); | 732 | local_irq_restore(flags); |
429 | } | 733 | } |
430 | 734 | ||
735 | /* At some point in the boot process, we get asked to set up our timing | ||
736 | * infrastructure. The kernel doesn't expect timer interrupts before this, but | ||
737 | * we cleverly initialized the "blocked_interrupts" field of "struct | ||
738 | * lguest_data" so that timer interrupts were blocked until now. */ | ||
431 | static void lguest_time_init(void) | 739 | static void lguest_time_init(void) |
432 | { | 740 | { |
741 | /* Set up the timer interrupt (0) to go to our simple timer routine */ | ||
433 | set_irq_handler(0, lguest_time_irq); | 742 | set_irq_handler(0, lguest_time_irq); |
434 | 743 | ||
435 | /* We use the TSC if the Host tells us we can, otherwise a dumb | 744 | /* Our clock structure look like arch/i386/kernel/tsc.c if we can use |
436 | * jiffies-based clock. */ | 745 | * the TSC, otherwise it looks like kernel/time/jiffies.c. Either way, |
746 | * the "rating" is initialized so high that it's always chosen over any | ||
747 | * other clocksource. */ | ||
437 | if (lguest_data.tsc_khz) { | 748 | if (lguest_data.tsc_khz) { |
438 | lguest_clock.shift = 22; | 749 | lguest_clock.shift = 22; |
439 | lguest_clock.mult = clocksource_khz2mult(lguest_data.tsc_khz, | 750 | lguest_clock.mult = clocksource_khz2mult(lguest_data.tsc_khz, |
@@ -449,13 +760,30 @@ static void lguest_time_init(void) | |||
449 | clock_base = lguest_clock_read(); | 760 | clock_base = lguest_clock_read(); |
450 | clocksource_register(&lguest_clock); | 761 | clocksource_register(&lguest_clock); |
451 | 762 | ||
452 | /* We can't set cpumask in the initializer: damn C limitations! */ | 763 | /* We can't set cpumask in the initializer: damn C limitations! Set it |
764 | * here and register our timer device. */ | ||
453 | lguest_clockevent.cpumask = cpumask_of_cpu(0); | 765 | lguest_clockevent.cpumask = cpumask_of_cpu(0); |
454 | clockevents_register_device(&lguest_clockevent); | 766 | clockevents_register_device(&lguest_clockevent); |
455 | 767 | ||
768 | /* Finally, we unblock the timer interrupt. */ | ||
456 | enable_lguest_irq(0); | 769 | enable_lguest_irq(0); |
457 | } | 770 | } |
458 | 771 | ||
772 | /* | ||
773 | * Miscellaneous bits and pieces. | ||
774 | * | ||
775 | * Here is an oddball collection of functions which the Guest needs for things | ||
776 | * to work. They're pretty simple. | ||
777 | */ | ||
778 | |||
779 | /* The Guest needs to tell the host what stack it expects traps to use. For | ||
780 | * native hardware, this is part of the Task State Segment mentioned above in | ||
781 | * lguest_load_tr_desc(), but to help hypervisors there's this special call. | ||
782 | * | ||
783 | * We tell the Host the segment we want to use (__KERNEL_DS is the kernel data | ||
784 | * segment), the privilege level (we're privilege level 1, the Host is 0 and | ||
785 | * will not tolerate us trying to use that), the stack pointer, and the number | ||
786 | * of pages in the stack. */ | ||
459 | static void lguest_load_esp0(struct tss_struct *tss, | 787 | static void lguest_load_esp0(struct tss_struct *tss, |
460 | struct thread_struct *thread) | 788 | struct thread_struct *thread) |
461 | { | 789 | { |
@@ -463,15 +791,31 @@ static void lguest_load_esp0(struct tss_struct *tss, | |||
463 | THREAD_SIZE/PAGE_SIZE); | 791 | THREAD_SIZE/PAGE_SIZE); |
464 | } | 792 | } |
465 | 793 | ||
794 | /* Let's just say, I wouldn't do debugging under a Guest. */ | ||
466 | static void lguest_set_debugreg(int regno, unsigned long value) | 795 | static void lguest_set_debugreg(int regno, unsigned long value) |
467 | { | 796 | { |
468 | /* FIXME: Implement */ | 797 | /* FIXME: Implement */ |
469 | } | 798 | } |
470 | 799 | ||
800 | /* There are times when the kernel wants to make sure that no memory writes are | ||
801 | * caught in the cache (that they've all reached real hardware devices). This | ||
802 | * doesn't matter for the Guest which has virtual hardware. | ||
803 | * | ||
804 | * On the Pentium 4 and above, cpuid() indicates that the Cache Line Flush | ||
805 | * (clflush) instruction is available and the kernel uses that. Otherwise, it | ||
806 | * uses the older "Write Back and Invalidate Cache" (wbinvd) instruction. | ||
807 | * Unlike clflush, wbinvd can only be run at privilege level 0. So we can | ||
808 | * ignore clflush, but replace wbinvd. | ||
809 | */ | ||
471 | static void lguest_wbinvd(void) | 810 | static void lguest_wbinvd(void) |
472 | { | 811 | { |
473 | } | 812 | } |
474 | 813 | ||
814 | /* If the Guest expects to have an Advanced Programmable Interrupt Controller, | ||
815 | * we play dumb by ignoring writes and returning 0 for reads. So it's no | ||
816 | * longer Programmable nor Controlling anything, and I don't think 8 lines of | ||
817 | * code qualifies for Advanced. It will also never interrupt anything. It | ||
818 | * does, however, allow us to get through the Linux boot code. */ | ||
475 | #ifdef CONFIG_X86_LOCAL_APIC | 819 | #ifdef CONFIG_X86_LOCAL_APIC |
476 | static void lguest_apic_write(unsigned long reg, unsigned long v) | 820 | static void lguest_apic_write(unsigned long reg, unsigned long v) |
477 | { | 821 | { |
@@ -483,19 +827,32 @@ static unsigned long lguest_apic_read(unsigned long reg) | |||
483 | } | 827 | } |
484 | #endif | 828 | #endif |
485 | 829 | ||
830 | /* STOP! Until an interrupt comes in. */ | ||
486 | static void lguest_safe_halt(void) | 831 | static void lguest_safe_halt(void) |
487 | { | 832 | { |
488 | hcall(LHCALL_HALT, 0, 0, 0); | 833 | hcall(LHCALL_HALT, 0, 0, 0); |
489 | } | 834 | } |
490 | 835 | ||
836 | /* Perhaps CRASH isn't the best name for this hypercall, but we use it to get a | ||
837 | * message out when we're crashing as well as elegant termination like powering | ||
838 | * off. | ||
839 | * | ||
840 | * Note that the Host always prefers that the Guest speak in physical addresses | ||
841 | * rather than virtual addresses, so we use __pa() here. */ | ||
491 | static void lguest_power_off(void) | 842 | static void lguest_power_off(void) |
492 | { | 843 | { |
493 | hcall(LHCALL_CRASH, __pa("Power down"), 0, 0); | 844 | hcall(LHCALL_CRASH, __pa("Power down"), 0, 0); |
494 | } | 845 | } |
495 | 846 | ||
847 | /* | ||
848 | * Panicing. | ||
849 | * | ||
850 | * Don't. But if you did, this is what happens. | ||
851 | */ | ||
496 | static int lguest_panic(struct notifier_block *nb, unsigned long l, void *p) | 852 | static int lguest_panic(struct notifier_block *nb, unsigned long l, void *p) |
497 | { | 853 | { |
498 | hcall(LHCALL_CRASH, __pa(p), 0, 0); | 854 | hcall(LHCALL_CRASH, __pa(p), 0, 0); |
855 | /* The hcall won't return, but to keep gcc happy, we're "done". */ | ||
499 | return NOTIFY_DONE; | 856 | return NOTIFY_DONE; |
500 | } | 857 | } |
501 | 858 | ||
@@ -503,15 +860,45 @@ static struct notifier_block paniced = { | |||
503 | .notifier_call = lguest_panic | 860 | .notifier_call = lguest_panic |
504 | }; | 861 | }; |
505 | 862 | ||
863 | /* Setting up memory is fairly easy. */ | ||
506 | static __init char *lguest_memory_setup(void) | 864 | static __init char *lguest_memory_setup(void) |
507 | { | 865 | { |
508 | /* We do this here because lockcheck barfs if before start_kernel */ | 866 | /* We do this here and not earlier because lockcheck barfs if we do it |
867 | * before start_kernel() */ | ||
509 | atomic_notifier_chain_register(&panic_notifier_list, &paniced); | 868 | atomic_notifier_chain_register(&panic_notifier_list, &paniced); |
510 | 869 | ||
870 | /* The Linux bootloader header contains an "e820" memory map: the | ||
871 | * Launcher populated the first entry with our memory limit. */ | ||
511 | add_memory_region(E820_MAP->addr, E820_MAP->size, E820_MAP->type); | 872 | add_memory_region(E820_MAP->addr, E820_MAP->size, E820_MAP->type); |
873 | |||
874 | /* This string is for the boot messages. */ | ||
512 | return "LGUEST"; | 875 | return "LGUEST"; |
513 | } | 876 | } |
514 | 877 | ||
878 | /*G:050 | ||
879 | * Patching (Powerfully Placating Performance Pedants) | ||
880 | * | ||
881 | * We have already seen that "struct paravirt_ops" lets us replace simple | ||
882 | * native instructions with calls to the appropriate back end all throughout | ||
883 | * the kernel. This allows the same kernel to run as a Guest and as a native | ||
884 | * kernel, but it's slow because of all the indirect branches. | ||
885 | * | ||
886 | * Remember that David Wheeler quote about "Any problem in computer science can | ||
887 | * be solved with another layer of indirection"? The rest of that quote is | ||
888 | * "... But that usually will create another problem." This is the first of | ||
889 | * those problems. | ||
890 | * | ||
891 | * Our current solution is to allow the paravirt back end to optionally patch | ||
892 | * over the indirect calls to replace them with something more efficient. We | ||
893 | * patch the four most commonly called functions: disable interrupts, enable | ||
894 | * interrupts, restore interrupts and save interrupts. We usually have 10 | ||
895 | * bytes to patch into: the Guest versions of these operations are small enough | ||
896 | * that we can fit comfortably. | ||
897 | * | ||
898 | * First we need assembly templates of each of the patchable Guest operations, | ||
899 | * and these are in lguest_asm.S. */ | ||
900 | |||
901 | /*G:060 We construct a table from the assembler templates: */ | ||
515 | static const struct lguest_insns | 902 | static const struct lguest_insns |
516 | { | 903 | { |
517 | const char *start, *end; | 904 | const char *start, *end; |
@@ -521,35 +908,52 @@ static const struct lguest_insns | |||
521 | [PARAVIRT_PATCH(restore_fl)] = { lgstart_popf, lgend_popf }, | 908 | [PARAVIRT_PATCH(restore_fl)] = { lgstart_popf, lgend_popf }, |
522 | [PARAVIRT_PATCH(save_fl)] = { lgstart_pushf, lgend_pushf }, | 909 | [PARAVIRT_PATCH(save_fl)] = { lgstart_pushf, lgend_pushf }, |
523 | }; | 910 | }; |
911 | |||
912 | /* Now our patch routine is fairly simple (based on the native one in | ||
913 | * paravirt.c). If we have a replacement, we copy it in and return how much of | ||
914 | * the available space we used. */ | ||
524 | static unsigned lguest_patch(u8 type, u16 clobber, void *insns, unsigned len) | 915 | static unsigned lguest_patch(u8 type, u16 clobber, void *insns, unsigned len) |
525 | { | 916 | { |
526 | unsigned int insn_len; | 917 | unsigned int insn_len; |
527 | 918 | ||
528 | /* Don't touch it if we don't have a replacement */ | 919 | /* Don't do anything special if we don't have a replacement */ |
529 | if (type >= ARRAY_SIZE(lguest_insns) || !lguest_insns[type].start) | 920 | if (type >= ARRAY_SIZE(lguest_insns) || !lguest_insns[type].start) |
530 | return paravirt_patch_default(type, clobber, insns, len); | 921 | return paravirt_patch_default(type, clobber, insns, len); |
531 | 922 | ||
532 | insn_len = lguest_insns[type].end - lguest_insns[type].start; | 923 | insn_len = lguest_insns[type].end - lguest_insns[type].start; |
533 | 924 | ||
534 | /* Similarly if we can't fit replacement. */ | 925 | /* Similarly if we can't fit replacement (shouldn't happen, but let's |
926 | * be thorough). */ | ||
535 | if (len < insn_len) | 927 | if (len < insn_len) |
536 | return paravirt_patch_default(type, clobber, insns, len); | 928 | return paravirt_patch_default(type, clobber, insns, len); |
537 | 929 | ||
930 | /* Copy in our instructions. */ | ||
538 | memcpy(insns, lguest_insns[type].start, insn_len); | 931 | memcpy(insns, lguest_insns[type].start, insn_len); |
539 | return insn_len; | 932 | return insn_len; |
540 | } | 933 | } |
541 | 934 | ||
935 | /*G:030 Once we get to lguest_init(), we know we're a Guest. The paravirt_ops | ||
936 | * structure in the kernel provides a single point for (almost) every routine | ||
937 | * we have to override to avoid privileged instructions. */ | ||
542 | __init void lguest_init(void *boot) | 938 | __init void lguest_init(void *boot) |
543 | { | 939 | { |
544 | /* Copy boot parameters first. */ | 940 | /* Copy boot parameters first: the Launcher put the physical location |
941 | * in %esi, and head.S converted that to a virtual address and handed | ||
942 | * it to us. */ | ||
545 | memcpy(&boot_params, boot, PARAM_SIZE); | 943 | memcpy(&boot_params, boot, PARAM_SIZE); |
944 | /* The boot parameters also tell us where the command-line is: save | ||
945 | * that, too. */ | ||
546 | memcpy(boot_command_line, __va(boot_params.hdr.cmd_line_ptr), | 946 | memcpy(boot_command_line, __va(boot_params.hdr.cmd_line_ptr), |
547 | COMMAND_LINE_SIZE); | 947 | COMMAND_LINE_SIZE); |
548 | 948 | ||
949 | /* We're under lguest, paravirt is enabled, and we're running at | ||
950 | * privilege level 1, not 0 as normal. */ | ||
549 | paravirt_ops.name = "lguest"; | 951 | paravirt_ops.name = "lguest"; |
550 | paravirt_ops.paravirt_enabled = 1; | 952 | paravirt_ops.paravirt_enabled = 1; |
551 | paravirt_ops.kernel_rpl = 1; | 953 | paravirt_ops.kernel_rpl = 1; |
552 | 954 | ||
955 | /* We set up all the lguest overrides for sensitive operations. These | ||
956 | * are detailed with the operations themselves. */ | ||
553 | paravirt_ops.save_fl = save_fl; | 957 | paravirt_ops.save_fl = save_fl; |
554 | paravirt_ops.restore_fl = restore_fl; | 958 | paravirt_ops.restore_fl = restore_fl; |
555 | paravirt_ops.irq_disable = irq_disable; | 959 | paravirt_ops.irq_disable = irq_disable; |
@@ -593,20 +997,45 @@ __init void lguest_init(void *boot) | |||
593 | paravirt_ops.set_lazy_mode = lguest_lazy_mode; | 997 | paravirt_ops.set_lazy_mode = lguest_lazy_mode; |
594 | paravirt_ops.wbinvd = lguest_wbinvd; | 998 | paravirt_ops.wbinvd = lguest_wbinvd; |
595 | paravirt_ops.sched_clock = lguest_sched_clock; | 999 | paravirt_ops.sched_clock = lguest_sched_clock; |
596 | 1000 | /* Now is a good time to look at the implementations of these functions | |
1001 | * before returning to the rest of lguest_init(). */ | ||
1002 | |||
1003 | /*G:070 Now we've seen all the paravirt_ops, we return to | ||
1004 | * lguest_init() where the rest of the fairly chaotic boot setup | ||
1005 | * occurs. | ||
1006 | * | ||
1007 | * The Host expects our first hypercall to tell it where our "struct | ||
1008 | * lguest_data" is, so we do that first. */ | ||
597 | hcall(LHCALL_LGUEST_INIT, __pa(&lguest_data), 0, 0); | 1009 | hcall(LHCALL_LGUEST_INIT, __pa(&lguest_data), 0, 0); |
598 | 1010 | ||
599 | /* We use top of mem for initial pagetables. */ | 1011 | /* The native boot code sets up initial page tables immediately after |
1012 | * the kernel itself, and sets init_pg_tables_end so they're not | ||
1013 | * clobbered. The Launcher places our initial pagetables somewhere at | ||
1014 | * the top of our physical memory, so we don't need extra space: set | ||
1015 | * init_pg_tables_end to the end of the kernel. */ | ||
600 | init_pg_tables_end = __pa(pg0); | 1016 | init_pg_tables_end = __pa(pg0); |
601 | 1017 | ||
1018 | /* Load the %fs segment register (the per-cpu segment register) with | ||
1019 | * the normal data segment to get through booting. */ | ||
602 | asm volatile ("mov %0, %%fs" : : "r" (__KERNEL_DS) : "memory"); | 1020 | asm volatile ("mov %0, %%fs" : : "r" (__KERNEL_DS) : "memory"); |
603 | 1021 | ||
1022 | /* The Host uses the top of the Guest's virtual address space for the | ||
1023 | * Host<->Guest Switcher, and it tells us how much it needs in | ||
1024 | * lguest_data.reserve_mem, set up on the LGUEST_INIT hypercall. */ | ||
604 | reserve_top_address(lguest_data.reserve_mem); | 1025 | reserve_top_address(lguest_data.reserve_mem); |
605 | 1026 | ||
1027 | /* If we don't initialize the lock dependency checker now, it crashes | ||
1028 | * paravirt_disable_iospace. */ | ||
606 | lockdep_init(); | 1029 | lockdep_init(); |
607 | 1030 | ||
1031 | /* The IDE code spends about 3 seconds probing for disks: if we reserve | ||
1032 | * all the I/O ports up front it can't get them and so doesn't probe. | ||
1033 | * Other device drivers are similar (but less severe). This cuts the | ||
1034 | * kernel boot time on my machine from 4.1 seconds to 0.45 seconds. */ | ||
608 | paravirt_disable_iospace(); | 1035 | paravirt_disable_iospace(); |
609 | 1036 | ||
1037 | /* This is messy CPU setup stuff which the native boot code does before | ||
1038 | * start_kernel, so we have to do, too: */ | ||
610 | cpu_detect(&new_cpu_data); | 1039 | cpu_detect(&new_cpu_data); |
611 | /* head.S usually sets up the first capability word, so do it here. */ | 1040 | /* head.S usually sets up the first capability word, so do it here. */ |
612 | new_cpu_data.x86_capability[0] = cpuid_edx(1); | 1041 | new_cpu_data.x86_capability[0] = cpuid_edx(1); |
@@ -617,14 +1046,27 @@ __init void lguest_init(void *boot) | |||
617 | #ifdef CONFIG_X86_MCE | 1046 | #ifdef CONFIG_X86_MCE |
618 | mce_disabled = 1; | 1047 | mce_disabled = 1; |
619 | #endif | 1048 | #endif |
620 | |||
621 | #ifdef CONFIG_ACPI | 1049 | #ifdef CONFIG_ACPI |
622 | acpi_disabled = 1; | 1050 | acpi_disabled = 1; |
623 | acpi_ht = 0; | 1051 | acpi_ht = 0; |
624 | #endif | 1052 | #endif |
625 | 1053 | ||
1054 | /* We set the perferred console to "hvc". This is the "hypervisor | ||
1055 | * virtual console" driver written by the PowerPC people, which we also | ||
1056 | * adapted for lguest's use. */ | ||
626 | add_preferred_console("hvc", 0, NULL); | 1057 | add_preferred_console("hvc", 0, NULL); |
627 | 1058 | ||
1059 | /* Last of all, we set the power management poweroff hook to point to | ||
1060 | * the Guest routine to power off. */ | ||
628 | pm_power_off = lguest_power_off; | 1061 | pm_power_off = lguest_power_off; |
1062 | |||
1063 | /* Now we're set up, call start_kernel() in init/main.c and we proceed | ||
1064 | * to boot as normal. It never returns. */ | ||
629 | start_kernel(); | 1065 | start_kernel(); |
630 | } | 1066 | } |
1067 | /* | ||
1068 | * This marks the end of stage II of our journey, The Guest. | ||
1069 | * | ||
1070 | * It is now time for us to explore the nooks and crannies of the three Guest | ||
1071 | * devices and complete our understanding of the Guest in "make Drivers". | ||
1072 | */ | ||
diff --git a/drivers/lguest/lguest_asm.S b/drivers/lguest/lguest_asm.S index a3dbf22ee365..f182c6a36209 100644 --- a/drivers/lguest/lguest_asm.S +++ b/drivers/lguest/lguest_asm.S | |||
@@ -4,15 +4,15 @@ | |||
4 | #include <asm/thread_info.h> | 4 | #include <asm/thread_info.h> |
5 | #include <asm/processor-flags.h> | 5 | #include <asm/processor-flags.h> |
6 | 6 | ||
7 | /* | 7 | /*G:020 This is where we begin: we have a magic signature which the launcher |
8 | * This is where we begin: we have a magic signature which the launcher looks | 8 | * looks for. The plan is that the Linux boot protocol will be extended with a |
9 | * for. The plan is that the Linux boot protocol will be extended with a | ||
10 | * "platform type" field which will guide us here from the normal entry point, | 9 | * "platform type" field which will guide us here from the normal entry point, |
11 | * but for the moment this suffices. We pass the virtual address of the boot | 10 | * but for the moment this suffices. The normal boot code uses %esi for the |
12 | * info to lguest_init(). | 11 | * boot header, so we do too. We convert it to a virtual address by adding |
12 | * PAGE_OFFSET, and hand it to lguest_init() as its argument (ie. %eax). | ||
13 | * | 13 | * |
14 | * We put it in .init.text will be discarded after boot. | 14 | * The .section line puts this code in .init.text so it will be discarded after |
15 | */ | 15 | * boot. */ |
16 | .section .init.text, "ax", @progbits | 16 | .section .init.text, "ax", @progbits |
17 | .ascii "GenuineLguest" | 17 | .ascii "GenuineLguest" |
18 | /* Set up initial stack. */ | 18 | /* Set up initial stack. */ |
@@ -21,7 +21,9 @@ | |||
21 | addl $__PAGE_OFFSET, %eax | 21 | addl $__PAGE_OFFSET, %eax |
22 | jmp lguest_init | 22 | jmp lguest_init |
23 | 23 | ||
24 | /* The templates for inline patching. */ | 24 | /*G:055 We create a macro which puts the assembler code between lgstart_ and |
25 | * lgend_ markers. These templates end up in the .init.text section, so they | ||
26 | * are discarded after boot. */ | ||
25 | #define LGUEST_PATCH(name, insns...) \ | 27 | #define LGUEST_PATCH(name, insns...) \ |
26 | lgstart_##name: insns; lgend_##name:; \ | 28 | lgstart_##name: insns; lgend_##name:; \ |
27 | .globl lgstart_##name; .globl lgend_##name | 29 | .globl lgstart_##name; .globl lgend_##name |
@@ -30,24 +32,61 @@ LGUEST_PATCH(cli, movl $0, lguest_data+LGUEST_DATA_irq_enabled) | |||
30 | LGUEST_PATCH(sti, movl $X86_EFLAGS_IF, lguest_data+LGUEST_DATA_irq_enabled) | 32 | LGUEST_PATCH(sti, movl $X86_EFLAGS_IF, lguest_data+LGUEST_DATA_irq_enabled) |
31 | LGUEST_PATCH(popf, movl %eax, lguest_data+LGUEST_DATA_irq_enabled) | 33 | LGUEST_PATCH(popf, movl %eax, lguest_data+LGUEST_DATA_irq_enabled) |
32 | LGUEST_PATCH(pushf, movl lguest_data+LGUEST_DATA_irq_enabled, %eax) | 34 | LGUEST_PATCH(pushf, movl lguest_data+LGUEST_DATA_irq_enabled, %eax) |
35 | /*:*/ | ||
33 | 36 | ||
34 | .text | 37 | .text |
35 | /* These demark the EIP range where host should never deliver interrupts. */ | 38 | /* These demark the EIP range where host should never deliver interrupts. */ |
36 | .global lguest_noirq_start | 39 | .global lguest_noirq_start |
37 | .global lguest_noirq_end | 40 | .global lguest_noirq_end |
38 | 41 | ||
39 | /* | 42 | /*M:004 When the Host reflects a trap or injects an interrupt into the Guest, |
40 | * We move eflags word to lguest_data.irq_enabled to restore interrupt state. | 43 | * it sets the eflags interrupt bit on the stack based on |
41 | * For page faults, gpfs and virtual interrupts, the hypervisor has saved | 44 | * lguest_data.irq_enabled, so the Guest iret logic does the right thing when |
42 | * eflags manually, otherwise it was delivered directly and so eflags reflects | 45 | * restoring it. However, when the Host sets the Guest up for direct traps, |
43 | * the real machine IF state, ie. interrupts on. Since the kernel always dies | 46 | * such as system calls, the processor is the one to push eflags onto the |
44 | * if it takes such a trap with interrupts disabled anyway, turning interrupts | 47 | * stack, and the interrupt bit will be 1 (in reality, interrupts are always |
45 | * back on unconditionally here is OK. | 48 | * enabled in the Guest). |
46 | */ | 49 | * |
50 | * This turns out to be harmless: the only trap which should happen under Linux | ||
51 | * with interrupts disabled is Page Fault (due to our lazy mapping of vmalloc | ||
52 | * regions), which has to be reflected through the Host anyway. If another | ||
53 | * trap *does* go off when interrupts are disabled, the Guest will panic, and | ||
54 | * we'll never get to this iret! :*/ | ||
55 | |||
56 | /*G:045 There is one final paravirt_op that the Guest implements, and glancing | ||
57 | * at it you can see why I left it to last. It's *cool*! It's in *assembler*! | ||
58 | * | ||
59 | * The "iret" instruction is used to return from an interrupt or trap. The | ||
60 | * stack looks like this: | ||
61 | * old address | ||
62 | * old code segment & privilege level | ||
63 | * old processor flags ("eflags") | ||
64 | * | ||
65 | * The "iret" instruction pops those values off the stack and restores them all | ||
66 | * at once. The only problem is that eflags includes the Interrupt Flag which | ||
67 | * the Guest can't change: the CPU will simply ignore it when we do an "iret". | ||
68 | * So we have to copy eflags from the stack to lguest_data.irq_enabled before | ||
69 | * we do the "iret". | ||
70 | * | ||
71 | * There are two problems with this: firstly, we need to use a register to do | ||
72 | * the copy and secondly, the whole thing needs to be atomic. The first | ||
73 | * problem is easy to solve: push %eax on the stack so we can use it, and then | ||
74 | * restore it at the end just before the real "iret". | ||
75 | * | ||
76 | * The second is harder: copying eflags to lguest_data.irq_enabled will turn | ||
77 | * interrupts on before we're finished, so we could be interrupted before we | ||
78 | * return to userspace or wherever. Our solution to this is to surround the | ||
79 | * code with lguest_noirq_start: and lguest_noirq_end: labels. We tell the | ||
80 | * Host that it is *never* to interrupt us there, even if interrupts seem to be | ||
81 | * enabled. */ | ||
47 | ENTRY(lguest_iret) | 82 | ENTRY(lguest_iret) |
48 | pushl %eax | 83 | pushl %eax |
49 | movl 12(%esp), %eax | 84 | movl 12(%esp), %eax |
50 | lguest_noirq_start: | 85 | lguest_noirq_start: |
86 | /* Note the %ss: segment prefix here. Normal data accesses use the | ||
87 | * "ds" segment, but that will have already been restored for whatever | ||
88 | * we're returning to (such as userspace): we can't trust it. The %ss: | ||
89 | * prefix makes sure we use the stack segment, which is still valid. */ | ||
51 | movl %eax,%ss:lguest_data+LGUEST_DATA_irq_enabled | 90 | movl %eax,%ss:lguest_data+LGUEST_DATA_irq_enabled |
52 | popl %eax | 91 | popl %eax |
53 | iret | 92 | iret |
diff --git a/drivers/lguest/lguest_bus.c b/drivers/lguest/lguest_bus.c index 18d6ab21a43b..55a7940ca732 100644 --- a/drivers/lguest/lguest_bus.c +++ b/drivers/lguest/lguest_bus.c | |||
@@ -1,3 +1,6 @@ | |||
1 | /*P:050 Lguest guests use a very simple bus for devices. It's a simple array | ||
2 | * of device descriptors contained just above the top of normal memory. The | ||
3 | * lguest bus is 80% tedious boilerplate code. :*/ | ||
1 | #include <linux/init.h> | 4 | #include <linux/init.h> |
2 | #include <linux/bootmem.h> | 5 | #include <linux/bootmem.h> |
3 | #include <linux/lguest_bus.h> | 6 | #include <linux/lguest_bus.h> |
@@ -43,6 +46,10 @@ static struct device_attribute lguest_dev_attrs[] = { | |||
43 | __ATTR_NULL | 46 | __ATTR_NULL |
44 | }; | 47 | }; |
45 | 48 | ||
49 | /*D:130 The generic bus infrastructure requires a function which says whether a | ||
50 | * device matches a driver. For us, it is simple: "struct lguest_driver" | ||
51 | * contains a "device_type" field which indicates what type of device it can | ||
52 | * handle, so we just cast the args and compare: */ | ||
46 | static int lguest_dev_match(struct device *_dev, struct device_driver *_drv) | 53 | static int lguest_dev_match(struct device *_dev, struct device_driver *_drv) |
47 | { | 54 | { |
48 | struct lguest_device *dev = container_of(_dev,struct lguest_device,dev); | 55 | struct lguest_device *dev = container_of(_dev,struct lguest_device,dev); |
@@ -50,6 +57,7 @@ static int lguest_dev_match(struct device *_dev, struct device_driver *_drv) | |||
50 | 57 | ||
51 | return (drv->device_type == lguest_devices[dev->index].type); | 58 | return (drv->device_type == lguest_devices[dev->index].type); |
52 | } | 59 | } |
60 | /*:*/ | ||
53 | 61 | ||
54 | struct lguest_bus { | 62 | struct lguest_bus { |
55 | struct bus_type bus; | 63 | struct bus_type bus; |
@@ -68,11 +76,24 @@ static struct lguest_bus lguest_bus = { | |||
68 | } | 76 | } |
69 | }; | 77 | }; |
70 | 78 | ||
79 | /*D:140 This is the callback which occurs once the bus infrastructure matches | ||
80 | * up a device and driver, ie. in response to add_lguest_device() calling | ||
81 | * device_register(), or register_lguest_driver() calling driver_register(). | ||
82 | * | ||
83 | * At the moment it's always the latter: the devices are added first, since | ||
84 | * scan_devices() is called from a "core_initcall", and the drivers themselves | ||
85 | * called later as a normal "initcall". But it would work the other way too. | ||
86 | * | ||
87 | * So now we have the happy couple, we add the status bit to indicate that we | ||
88 | * found a driver. If the driver truly loves the device, it will return | ||
89 | * happiness from its probe function (ok, perhaps this wasn't my greatest | ||
90 | * analogy), and we set the final "driver ok" bit so the Host sees it's all | ||
91 | * green. */ | ||
71 | static int lguest_dev_probe(struct device *_dev) | 92 | static int lguest_dev_probe(struct device *_dev) |
72 | { | 93 | { |
73 | int ret; | 94 | int ret; |
74 | struct lguest_device *dev = container_of(_dev,struct lguest_device,dev); | 95 | struct lguest_device*dev = container_of(_dev,struct lguest_device,dev); |
75 | struct lguest_driver *drv = container_of(dev->dev.driver, | 96 | struct lguest_driver*drv = container_of(dev->dev.driver, |
76 | struct lguest_driver, drv); | 97 | struct lguest_driver, drv); |
77 | 98 | ||
78 | lguest_devices[dev->index].status |= LGUEST_DEVICE_S_DRIVER; | 99 | lguest_devices[dev->index].status |= LGUEST_DEVICE_S_DRIVER; |
@@ -82,6 +103,10 @@ static int lguest_dev_probe(struct device *_dev) | |||
82 | return ret; | 103 | return ret; |
83 | } | 104 | } |
84 | 105 | ||
106 | /* The last part of the bus infrastructure is the function lguest drivers use | ||
107 | * to register themselves. Firstly, we do nothing if there's no lguest bus | ||
108 | * (ie. this is not a Guest), otherwise we fill in the embedded generic "struct | ||
109 | * driver" fields and call the generic driver_register(). */ | ||
85 | int register_lguest_driver(struct lguest_driver *drv) | 110 | int register_lguest_driver(struct lguest_driver *drv) |
86 | { | 111 | { |
87 | if (!lguest_devices) | 112 | if (!lguest_devices) |
@@ -94,12 +119,36 @@ int register_lguest_driver(struct lguest_driver *drv) | |||
94 | 119 | ||
95 | return driver_register(&drv->drv); | 120 | return driver_register(&drv->drv); |
96 | } | 121 | } |
122 | |||
123 | /* At the moment we build all the drivers into the kernel because they're so | ||
124 | * simple: 8144 bytes for all three of them as I type this. And as the console | ||
125 | * really needs to be built in, it's actually only 3527 bytes for the network | ||
126 | * and block drivers. | ||
127 | * | ||
128 | * If they get complex it will make sense for them to be modularized, so we | ||
129 | * need to explicitly export the symbol. | ||
130 | * | ||
131 | * I don't think non-GPL modules make sense, so it's a GPL-only export. | ||
132 | */ | ||
97 | EXPORT_SYMBOL_GPL(register_lguest_driver); | 133 | EXPORT_SYMBOL_GPL(register_lguest_driver); |
98 | 134 | ||
135 | /*D:120 This is the core of the lguest bus: actually adding a new device. | ||
136 | * It's a separate function because it's neater that way, and because an | ||
137 | * earlier version of the code supported hotplug and unplug. They were removed | ||
138 | * early on because they were never used. | ||
139 | * | ||
140 | * As Andrew Tridgell says, "Untested code is buggy code". | ||
141 | * | ||
142 | * It's worth reading this carefully: we start with an index into the array of | ||
143 | * "struct lguest_device_desc"s indicating the device which is new: */ | ||
99 | static void add_lguest_device(unsigned int index) | 144 | static void add_lguest_device(unsigned int index) |
100 | { | 145 | { |
101 | struct lguest_device *new; | 146 | struct lguest_device *new; |
102 | 147 | ||
148 | /* Each "struct lguest_device_desc" has a "status" field, which the | ||
149 | * Guest updates as the device is probed. In the worst case, the Host | ||
150 | * can look at these bits to tell what part of device setup failed, | ||
151 | * even if the console isn't available. */ | ||
103 | lguest_devices[index].status |= LGUEST_DEVICE_S_ACKNOWLEDGE; | 152 | lguest_devices[index].status |= LGUEST_DEVICE_S_ACKNOWLEDGE; |
104 | new = kmalloc(sizeof(struct lguest_device), GFP_KERNEL); | 153 | new = kmalloc(sizeof(struct lguest_device), GFP_KERNEL); |
105 | if (!new) { | 154 | if (!new) { |
@@ -108,12 +157,17 @@ static void add_lguest_device(unsigned int index) | |||
108 | return; | 157 | return; |
109 | } | 158 | } |
110 | 159 | ||
160 | /* The "struct lguest_device" setup is pretty straight-forward example | ||
161 | * code. */ | ||
111 | new->index = index; | 162 | new->index = index; |
112 | new->private = NULL; | 163 | new->private = NULL; |
113 | memset(&new->dev, 0, sizeof(new->dev)); | 164 | memset(&new->dev, 0, sizeof(new->dev)); |
114 | new->dev.parent = &lguest_bus.dev; | 165 | new->dev.parent = &lguest_bus.dev; |
115 | new->dev.bus = &lguest_bus.bus; | 166 | new->dev.bus = &lguest_bus.bus; |
116 | sprintf(new->dev.bus_id, "%u", index); | 167 | sprintf(new->dev.bus_id, "%u", index); |
168 | |||
169 | /* device_register() causes the bus infrastructure to look for a | ||
170 | * matching driver. */ | ||
117 | if (device_register(&new->dev) != 0) { | 171 | if (device_register(&new->dev) != 0) { |
118 | printk(KERN_EMERG "Cannot register lguest device %u\n", index); | 172 | printk(KERN_EMERG "Cannot register lguest device %u\n", index); |
119 | lguest_devices[index].status |= LGUEST_DEVICE_S_FAILED; | 173 | lguest_devices[index].status |= LGUEST_DEVICE_S_FAILED; |
@@ -121,6 +175,9 @@ static void add_lguest_device(unsigned int index) | |||
121 | } | 175 | } |
122 | } | 176 | } |
123 | 177 | ||
178 | /*D:110 scan_devices() simply iterates through the device array. The type 0 | ||
179 | * is reserved to mean "no device", and anything else means we have found a | ||
180 | * device: add it. */ | ||
124 | static void scan_devices(void) | 181 | static void scan_devices(void) |
125 | { | 182 | { |
126 | unsigned int i; | 183 | unsigned int i; |
@@ -130,12 +187,23 @@ static void scan_devices(void) | |||
130 | add_lguest_device(i); | 187 | add_lguest_device(i); |
131 | } | 188 | } |
132 | 189 | ||
190 | /*D:100 Fairly early in boot, lguest_bus_init() is called to set up the lguest | ||
191 | * bus. We check that we are a Guest by checking paravirt_ops.name: there are | ||
192 | * other ways of checking, but this seems most obvious to me. | ||
193 | * | ||
194 | * So we can access the array of "struct lguest_device_desc"s easily, we map | ||
195 | * that memory and store the pointer in the global "lguest_devices". Then we | ||
196 | * register the bus with the core. Doing two registrations seems clunky to me, | ||
197 | * but it seems to be the correct sysfs incantation. | ||
198 | * | ||
199 | * Finally we call scan_devices() which adds all the devices found in the | ||
200 | * "struct lguest_device_desc" array. */ | ||
133 | static int __init lguest_bus_init(void) | 201 | static int __init lguest_bus_init(void) |
134 | { | 202 | { |
135 | if (strcmp(paravirt_ops.name, "lguest") != 0) | 203 | if (strcmp(paravirt_ops.name, "lguest") != 0) |
136 | return 0; | 204 | return 0; |
137 | 205 | ||
138 | /* Devices are in page above top of "normal" mem. */ | 206 | /* Devices are in a single page above top of "normal" mem */ |
139 | lguest_devices = lguest_map(max_pfn<<PAGE_SHIFT, 1); | 207 | lguest_devices = lguest_map(max_pfn<<PAGE_SHIFT, 1); |
140 | 208 | ||
141 | if (bus_register(&lguest_bus.bus) != 0 | 209 | if (bus_register(&lguest_bus.bus) != 0 |
@@ -145,4 +213,5 @@ static int __init lguest_bus_init(void) | |||
145 | scan_devices(); | 213 | scan_devices(); |
146 | return 0; | 214 | return 0; |
147 | } | 215 | } |
216 | /* Do this after core stuff, before devices. */ | ||
148 | postcore_initcall(lguest_bus_init); | 217 | postcore_initcall(lguest_bus_init); |
diff --git a/drivers/lguest/lguest_user.c b/drivers/lguest/lguest_user.c index e90d7a783daf..80d1b58c7698 100644 --- a/drivers/lguest/lguest_user.c +++ b/drivers/lguest/lguest_user.c | |||
@@ -1,36 +1,70 @@ | |||
1 | /* Userspace control of the guest, via /dev/lguest. */ | 1 | /*P:200 This contains all the /dev/lguest code, whereby the userspace launcher |
2 | * controls and communicates with the Guest. For example, the first write will | ||
3 | * tell us the memory size, pagetable, entry point and kernel address offset. | ||
4 | * A read will run the Guest until a signal is pending (-EINTR), or the Guest | ||
5 | * does a DMA out to the Launcher. Writes are also used to get a DMA buffer | ||
6 | * registered by the Guest and to send the Guest an interrupt. :*/ | ||
2 | #include <linux/uaccess.h> | 7 | #include <linux/uaccess.h> |
3 | #include <linux/miscdevice.h> | 8 | #include <linux/miscdevice.h> |
4 | #include <linux/fs.h> | 9 | #include <linux/fs.h> |
5 | #include "lg.h" | 10 | #include "lg.h" |
6 | 11 | ||
12 | /*L:030 setup_regs() doesn't really belong in this file, but it gives us an | ||
13 | * early glimpse deeper into the Host so it's worth having here. | ||
14 | * | ||
15 | * Most of the Guest's registers are left alone: we used get_zeroed_page() to | ||
16 | * allocate the structure, so they will be 0. */ | ||
7 | static void setup_regs(struct lguest_regs *regs, unsigned long start) | 17 | static void setup_regs(struct lguest_regs *regs, unsigned long start) |
8 | { | 18 | { |
9 | /* Write out stack in format lguest expects, so we can switch to it. */ | 19 | /* There are four "segment" registers which the Guest needs to boot: |
20 | * The "code segment" register (cs) refers to the kernel code segment | ||
21 | * __KERNEL_CS, and the "data", "extra" and "stack" segment registers | ||
22 | * refer to the kernel data segment __KERNEL_DS. | ||
23 | * | ||
24 | * The privilege level is packed into the lower bits. The Guest runs | ||
25 | * at privilege level 1 (GUEST_PL).*/ | ||
10 | regs->ds = regs->es = regs->ss = __KERNEL_DS|GUEST_PL; | 26 | regs->ds = regs->es = regs->ss = __KERNEL_DS|GUEST_PL; |
11 | regs->cs = __KERNEL_CS|GUEST_PL; | 27 | regs->cs = __KERNEL_CS|GUEST_PL; |
12 | regs->eflags = 0x202; /* Interrupts enabled. */ | 28 | |
29 | /* The "eflags" register contains miscellaneous flags. Bit 1 (0x002) | ||
30 | * is supposed to always be "1". Bit 9 (0x200) controls whether | ||
31 | * interrupts are enabled. We always leave interrupts enabled while | ||
32 | * running the Guest. */ | ||
33 | regs->eflags = 0x202; | ||
34 | |||
35 | /* The "Extended Instruction Pointer" register says where the Guest is | ||
36 | * running. */ | ||
13 | regs->eip = start; | 37 | regs->eip = start; |
14 | /* esi points to our boot information (physical address 0) */ | 38 | |
39 | /* %esi points to our boot information, at physical address 0, so don't | ||
40 | * touch it. */ | ||
15 | } | 41 | } |
16 | 42 | ||
17 | /* + addr */ | 43 | /*L:310 To send DMA into the Guest, the Launcher needs to be able to ask for a |
44 | * DMA buffer. This is done by writing LHREQ_GETDMA and the key to | ||
45 | * /dev/lguest. */ | ||
18 | static long user_get_dma(struct lguest *lg, const u32 __user *input) | 46 | static long user_get_dma(struct lguest *lg, const u32 __user *input) |
19 | { | 47 | { |
20 | unsigned long key, udma, irq; | 48 | unsigned long key, udma, irq; |
21 | 49 | ||
50 | /* Fetch the key they wrote to us. */ | ||
22 | if (get_user(key, input) != 0) | 51 | if (get_user(key, input) != 0) |
23 | return -EFAULT; | 52 | return -EFAULT; |
53 | /* Look for a free Guest DMA buffer bound to that key. */ | ||
24 | udma = get_dma_buffer(lg, key, &irq); | 54 | udma = get_dma_buffer(lg, key, &irq); |
25 | if (!udma) | 55 | if (!udma) |
26 | return -ENOENT; | 56 | return -ENOENT; |
27 | 57 | ||
28 | /* We put irq number in udma->used_len. */ | 58 | /* We need to tell the Launcher what interrupt the Guest expects after |
59 | * the buffer is filled. We stash it in udma->used_len. */ | ||
29 | lgwrite_u32(lg, udma + offsetof(struct lguest_dma, used_len), irq); | 60 | lgwrite_u32(lg, udma + offsetof(struct lguest_dma, used_len), irq); |
61 | |||
62 | /* The (guest-physical) address of the DMA buffer is returned from | ||
63 | * the write(). */ | ||
30 | return udma; | 64 | return udma; |
31 | } | 65 | } |
32 | 66 | ||
33 | /* To force the Guest to stop running and return to the Launcher, the | 67 | /*L:315 To force the Guest to stop running and return to the Launcher, the |
34 | * Waker sets writes LHREQ_BREAK and the value "1" to /dev/lguest. The | 68 | * Waker sets writes LHREQ_BREAK and the value "1" to /dev/lguest. The |
35 | * Launcher then writes LHREQ_BREAK and "0" to release the Waker. */ | 69 | * Launcher then writes LHREQ_BREAK and "0" to release the Waker. */ |
36 | static int break_guest_out(struct lguest *lg, const u32 __user *input) | 70 | static int break_guest_out(struct lguest *lg, const u32 __user *input) |
@@ -54,7 +88,8 @@ static int break_guest_out(struct lguest *lg, const u32 __user *input) | |||
54 | } | 88 | } |
55 | } | 89 | } |
56 | 90 | ||
57 | /* + irq */ | 91 | /*L:050 Sending an interrupt is done by writing LHREQ_IRQ and an interrupt |
92 | * number to /dev/lguest. */ | ||
58 | static int user_send_irq(struct lguest *lg, const u32 __user *input) | 93 | static int user_send_irq(struct lguest *lg, const u32 __user *input) |
59 | { | 94 | { |
60 | u32 irq; | 95 | u32 irq; |
@@ -63,14 +98,19 @@ static int user_send_irq(struct lguest *lg, const u32 __user *input) | |||
63 | return -EFAULT; | 98 | return -EFAULT; |
64 | if (irq >= LGUEST_IRQS) | 99 | if (irq >= LGUEST_IRQS) |
65 | return -EINVAL; | 100 | return -EINVAL; |
101 | /* Next time the Guest runs, the core code will see if it can deliver | ||
102 | * this interrupt. */ | ||
66 | set_bit(irq, lg->irqs_pending); | 103 | set_bit(irq, lg->irqs_pending); |
67 | return 0; | 104 | return 0; |
68 | } | 105 | } |
69 | 106 | ||
107 | /*L:040 Once our Guest is initialized, the Launcher makes it run by reading | ||
108 | * from /dev/lguest. */ | ||
70 | static ssize_t read(struct file *file, char __user *user, size_t size,loff_t*o) | 109 | static ssize_t read(struct file *file, char __user *user, size_t size,loff_t*o) |
71 | { | 110 | { |
72 | struct lguest *lg = file->private_data; | 111 | struct lguest *lg = file->private_data; |
73 | 112 | ||
113 | /* You must write LHREQ_INITIALIZE first! */ | ||
74 | if (!lg) | 114 | if (!lg) |
75 | return -EINVAL; | 115 | return -EINVAL; |
76 | 116 | ||
@@ -78,27 +118,52 @@ static ssize_t read(struct file *file, char __user *user, size_t size,loff_t*o) | |||
78 | if (current != lg->tsk) | 118 | if (current != lg->tsk) |
79 | return -EPERM; | 119 | return -EPERM; |
80 | 120 | ||
121 | /* If the guest is already dead, we indicate why */ | ||
81 | if (lg->dead) { | 122 | if (lg->dead) { |
82 | size_t len; | 123 | size_t len; |
83 | 124 | ||
125 | /* lg->dead either contains an error code, or a string. */ | ||
84 | if (IS_ERR(lg->dead)) | 126 | if (IS_ERR(lg->dead)) |
85 | return PTR_ERR(lg->dead); | 127 | return PTR_ERR(lg->dead); |
86 | 128 | ||
129 | /* We can only return as much as the buffer they read with. */ | ||
87 | len = min(size, strlen(lg->dead)+1); | 130 | len = min(size, strlen(lg->dead)+1); |
88 | if (copy_to_user(user, lg->dead, len) != 0) | 131 | if (copy_to_user(user, lg->dead, len) != 0) |
89 | return -EFAULT; | 132 | return -EFAULT; |
90 | return len; | 133 | return len; |
91 | } | 134 | } |
92 | 135 | ||
136 | /* If we returned from read() last time because the Guest sent DMA, | ||
137 | * clear the flag. */ | ||
93 | if (lg->dma_is_pending) | 138 | if (lg->dma_is_pending) |
94 | lg->dma_is_pending = 0; | 139 | lg->dma_is_pending = 0; |
95 | 140 | ||
141 | /* Run the Guest until something interesting happens. */ | ||
96 | return run_guest(lg, (unsigned long __user *)user); | 142 | return run_guest(lg, (unsigned long __user *)user); |
97 | } | 143 | } |
98 | 144 | ||
99 | /* Take: pfnlimit, pgdir, start, pageoffset. */ | 145 | /*L:020 The initialization write supplies 4 32-bit values (in addition to the |
146 | * 32-bit LHREQ_INITIALIZE value). These are: | ||
147 | * | ||
148 | * pfnlimit: The highest (Guest-physical) page number the Guest should be | ||
149 | * allowed to access. The Launcher has to live in Guest memory, so it sets | ||
150 | * this to ensure the Guest can't reach it. | ||
151 | * | ||
152 | * pgdir: The (Guest-physical) address of the top of the initial Guest | ||
153 | * pagetables (which are set up by the Launcher). | ||
154 | * | ||
155 | * start: The first instruction to execute ("eip" in x86-speak). | ||
156 | * | ||
157 | * page_offset: The PAGE_OFFSET constant in the Guest kernel. We should | ||
158 | * probably wean the code off this, but it's a very useful constant! Any | ||
159 | * address above this is within the Guest kernel, and any kernel address can | ||
160 | * quickly converted from physical to virtual by adding PAGE_OFFSET. It's | ||
161 | * 0xC0000000 (3G) by default, but it's configurable at kernel build time. | ||
162 | */ | ||
100 | static int initialize(struct file *file, const u32 __user *input) | 163 | static int initialize(struct file *file, const u32 __user *input) |
101 | { | 164 | { |
165 | /* "struct lguest" contains everything we (the Host) know about a | ||
166 | * Guest. */ | ||
102 | struct lguest *lg; | 167 | struct lguest *lg; |
103 | int err, i; | 168 | int err, i; |
104 | u32 args[4]; | 169 | u32 args[4]; |
@@ -106,7 +171,7 @@ static int initialize(struct file *file, const u32 __user *input) | |||
106 | /* We grab the Big Lguest lock, which protects the global array | 171 | /* We grab the Big Lguest lock, which protects the global array |
107 | * "lguests" and multiple simultaneous initializations. */ | 172 | * "lguests" and multiple simultaneous initializations. */ |
108 | mutex_lock(&lguest_lock); | 173 | mutex_lock(&lguest_lock); |
109 | 174 | /* You can't initialize twice! Close the device and start again... */ | |
110 | if (file->private_data) { | 175 | if (file->private_data) { |
111 | err = -EBUSY; | 176 | err = -EBUSY; |
112 | goto unlock; | 177 | goto unlock; |
@@ -117,37 +182,70 @@ static int initialize(struct file *file, const u32 __user *input) | |||
117 | goto unlock; | 182 | goto unlock; |
118 | } | 183 | } |
119 | 184 | ||
185 | /* Find an unused guest. */ | ||
120 | i = find_free_guest(); | 186 | i = find_free_guest(); |
121 | if (i < 0) { | 187 | if (i < 0) { |
122 | err = -ENOSPC; | 188 | err = -ENOSPC; |
123 | goto unlock; | 189 | goto unlock; |
124 | } | 190 | } |
191 | /* OK, we have an index into the "lguest" array: "lg" is a convenient | ||
192 | * pointer. */ | ||
125 | lg = &lguests[i]; | 193 | lg = &lguests[i]; |
194 | |||
195 | /* Populate the easy fields of our "struct lguest" */ | ||
126 | lg->guestid = i; | 196 | lg->guestid = i; |
127 | lg->pfn_limit = args[0]; | 197 | lg->pfn_limit = args[0]; |
128 | lg->page_offset = args[3]; | 198 | lg->page_offset = args[3]; |
199 | |||
200 | /* We need a complete page for the Guest registers: they are accessible | ||
201 | * to the Guest and we can only grant it access to whole pages. */ | ||
129 | lg->regs_page = get_zeroed_page(GFP_KERNEL); | 202 | lg->regs_page = get_zeroed_page(GFP_KERNEL); |
130 | if (!lg->regs_page) { | 203 | if (!lg->regs_page) { |
131 | err = -ENOMEM; | 204 | err = -ENOMEM; |
132 | goto release_guest; | 205 | goto release_guest; |
133 | } | 206 | } |
207 | /* We actually put the registers at the bottom of the page. */ | ||
134 | lg->regs = (void *)lg->regs_page + PAGE_SIZE - sizeof(*lg->regs); | 208 | lg->regs = (void *)lg->regs_page + PAGE_SIZE - sizeof(*lg->regs); |
135 | 209 | ||
210 | /* Initialize the Guest's shadow page tables, using the toplevel | ||
211 | * address the Launcher gave us. This allocates memory, so can | ||
212 | * fail. */ | ||
136 | err = init_guest_pagetable(lg, args[1]); | 213 | err = init_guest_pagetable(lg, args[1]); |
137 | if (err) | 214 | if (err) |
138 | goto free_regs; | 215 | goto free_regs; |
139 | 216 | ||
217 | /* Now we initialize the Guest's registers, handing it the start | ||
218 | * address. */ | ||
140 | setup_regs(lg->regs, args[2]); | 219 | setup_regs(lg->regs, args[2]); |
220 | |||
221 | /* There are a couple of GDT entries the Guest expects when first | ||
222 | * booting. */ | ||
141 | setup_guest_gdt(lg); | 223 | setup_guest_gdt(lg); |
224 | |||
225 | /* The timer for lguest's clock needs initialization. */ | ||
142 | init_clockdev(lg); | 226 | init_clockdev(lg); |
227 | |||
228 | /* We keep a pointer to the Launcher task (ie. current task) for when | ||
229 | * other Guests want to wake this one (inter-Guest I/O). */ | ||
143 | lg->tsk = current; | 230 | lg->tsk = current; |
231 | /* We need to keep a pointer to the Launcher's memory map, because if | ||
232 | * the Launcher dies we need to clean it up. If we don't keep a | ||
233 | * reference, it is destroyed before close() is called. */ | ||
144 | lg->mm = get_task_mm(lg->tsk); | 234 | lg->mm = get_task_mm(lg->tsk); |
235 | |||
236 | /* Initialize the queue for the waker to wait on */ | ||
145 | init_waitqueue_head(&lg->break_wq); | 237 | init_waitqueue_head(&lg->break_wq); |
238 | |||
239 | /* We remember which CPU's pages this Guest used last, for optimization | ||
240 | * when the same Guest runs on the same CPU twice. */ | ||
146 | lg->last_pages = NULL; | 241 | lg->last_pages = NULL; |
242 | |||
243 | /* We keep our "struct lguest" in the file's private_data. */ | ||
147 | file->private_data = lg; | 244 | file->private_data = lg; |
148 | 245 | ||
149 | mutex_unlock(&lguest_lock); | 246 | mutex_unlock(&lguest_lock); |
150 | 247 | ||
248 | /* And because this is a write() call, we return the length used. */ | ||
151 | return sizeof(args); | 249 | return sizeof(args); |
152 | 250 | ||
153 | free_regs: | 251 | free_regs: |
@@ -159,9 +257,15 @@ unlock: | |||
159 | return err; | 257 | return err; |
160 | } | 258 | } |
161 | 259 | ||
260 | /*L:010 The first operation the Launcher does must be a write. All writes | ||
261 | * start with a 32 bit number: for the first write this must be | ||
262 | * LHREQ_INITIALIZE to set up the Guest. After that the Launcher can use | ||
263 | * writes of other values to get DMA buffers and send interrupts. */ | ||
162 | static ssize_t write(struct file *file, const char __user *input, | 264 | static ssize_t write(struct file *file, const char __user *input, |
163 | size_t size, loff_t *off) | 265 | size_t size, loff_t *off) |
164 | { | 266 | { |
267 | /* Once the guest is initialized, we hold the "struct lguest" in the | ||
268 | * file private data. */ | ||
165 | struct lguest *lg = file->private_data; | 269 | struct lguest *lg = file->private_data; |
166 | u32 req; | 270 | u32 req; |
167 | 271 | ||
@@ -169,8 +273,11 @@ static ssize_t write(struct file *file, const char __user *input, | |||
169 | return -EFAULT; | 273 | return -EFAULT; |
170 | input += sizeof(req); | 274 | input += sizeof(req); |
171 | 275 | ||
276 | /* If you haven't initialized, you must do that first. */ | ||
172 | if (req != LHREQ_INITIALIZE && !lg) | 277 | if (req != LHREQ_INITIALIZE && !lg) |
173 | return -EINVAL; | 278 | return -EINVAL; |
279 | |||
280 | /* Once the Guest is dead, all you can do is read() why it died. */ | ||
174 | if (lg && lg->dead) | 281 | if (lg && lg->dead) |
175 | return -ENOENT; | 282 | return -ENOENT; |
176 | 283 | ||
@@ -192,33 +299,72 @@ static ssize_t write(struct file *file, const char __user *input, | |||
192 | } | 299 | } |
193 | } | 300 | } |
194 | 301 | ||
302 | /*L:060 The final piece of interface code is the close() routine. It reverses | ||
303 | * everything done in initialize(). This is usually called because the | ||
304 | * Launcher exited. | ||
305 | * | ||
306 | * Note that the close routine returns 0 or a negative error number: it can't | ||
307 | * really fail, but it can whine. I blame Sun for this wart, and K&R C for | ||
308 | * letting them do it. :*/ | ||
195 | static int close(struct inode *inode, struct file *file) | 309 | static int close(struct inode *inode, struct file *file) |
196 | { | 310 | { |
197 | struct lguest *lg = file->private_data; | 311 | struct lguest *lg = file->private_data; |
198 | 312 | ||
313 | /* If we never successfully initialized, there's nothing to clean up */ | ||
199 | if (!lg) | 314 | if (!lg) |
200 | return 0; | 315 | return 0; |
201 | 316 | ||
317 | /* We need the big lock, to protect from inter-guest I/O and other | ||
318 | * Launchers initializing guests. */ | ||
202 | mutex_lock(&lguest_lock); | 319 | mutex_lock(&lguest_lock); |
203 | /* Cancels the hrtimer set via LHCALL_SET_CLOCKEVENT. */ | 320 | /* Cancels the hrtimer set via LHCALL_SET_CLOCKEVENT. */ |
204 | hrtimer_cancel(&lg->hrt); | 321 | hrtimer_cancel(&lg->hrt); |
322 | /* Free any DMA buffers the Guest had bound. */ | ||
205 | release_all_dma(lg); | 323 | release_all_dma(lg); |
324 | /* Free up the shadow page tables for the Guest. */ | ||
206 | free_guest_pagetable(lg); | 325 | free_guest_pagetable(lg); |
326 | /* Now all the memory cleanups are done, it's safe to release the | ||
327 | * Launcher's memory management structure. */ | ||
207 | mmput(lg->mm); | 328 | mmput(lg->mm); |
329 | /* If lg->dead doesn't contain an error code it will be NULL or a | ||
330 | * kmalloc()ed string, either of which is ok to hand to kfree(). */ | ||
208 | if (!IS_ERR(lg->dead)) | 331 | if (!IS_ERR(lg->dead)) |
209 | kfree(lg->dead); | 332 | kfree(lg->dead); |
333 | /* We can free up the register page we allocated. */ | ||
210 | free_page(lg->regs_page); | 334 | free_page(lg->regs_page); |
335 | /* We clear the entire structure, which also marks it as free for the | ||
336 | * next user. */ | ||
211 | memset(lg, 0, sizeof(*lg)); | 337 | memset(lg, 0, sizeof(*lg)); |
338 | /* Release lock and exit. */ | ||
212 | mutex_unlock(&lguest_lock); | 339 | mutex_unlock(&lguest_lock); |
340 | |||
213 | return 0; | 341 | return 0; |
214 | } | 342 | } |
215 | 343 | ||
344 | /*L:000 | ||
345 | * Welcome to our journey through the Launcher! | ||
346 | * | ||
347 | * The Launcher is the Host userspace program which sets up, runs and services | ||
348 | * the Guest. In fact, many comments in the Drivers which refer to "the Host" | ||
349 | * doing things are inaccurate: the Launcher does all the device handling for | ||
350 | * the Guest. The Guest can't tell what's done by the the Launcher and what by | ||
351 | * the Host. | ||
352 | * | ||
353 | * Just to confuse you: to the Host kernel, the Launcher *is* the Guest and we | ||
354 | * shall see more of that later. | ||
355 | * | ||
356 | * We begin our understanding with the Host kernel interface which the Launcher | ||
357 | * uses: reading and writing a character device called /dev/lguest. All the | ||
358 | * work happens in the read(), write() and close() routines: */ | ||
216 | static struct file_operations lguest_fops = { | 359 | static struct file_operations lguest_fops = { |
217 | .owner = THIS_MODULE, | 360 | .owner = THIS_MODULE, |
218 | .release = close, | 361 | .release = close, |
219 | .write = write, | 362 | .write = write, |
220 | .read = read, | 363 | .read = read, |
221 | }; | 364 | }; |
365 | |||
366 | /* This is a textbook example of a "misc" character device. Populate a "struct | ||
367 | * miscdevice" and register it with misc_register(). */ | ||
222 | static struct miscdevice lguest_dev = { | 368 | static struct miscdevice lguest_dev = { |
223 | .minor = MISC_DYNAMIC_MINOR, | 369 | .minor = MISC_DYNAMIC_MINOR, |
224 | .name = "lguest", | 370 | .name = "lguest", |
diff --git a/drivers/lguest/page_tables.c b/drivers/lguest/page_tables.c index 1b0ba09b1269..b7a924ace684 100644 --- a/drivers/lguest/page_tables.c +++ b/drivers/lguest/page_tables.c | |||
@@ -1,5 +1,11 @@ | |||
1 | /* Shadow page table operations. | 1 | /*P:700 The pagetable code, on the other hand, still shows the scars of |
2 | * Copyright (C) Rusty Russell IBM Corporation 2006. | 2 | * previous encounters. It's functional, and as neat as it can be in the |
3 | * circumstances, but be wary, for these things are subtle and break easily. | ||
4 | * The Guest provides a virtual to physical mapping, but we can neither trust | ||
5 | * it nor use it: we verify and convert it here to point the hardware to the | ||
6 | * actual Guest pages when running the Guest. :*/ | ||
7 | |||
8 | /* Copyright (C) Rusty Russell IBM Corporation 2006. | ||
3 | * GPL v2 and any later version */ | 9 | * GPL v2 and any later version */ |
4 | #include <linux/mm.h> | 10 | #include <linux/mm.h> |
5 | #include <linux/types.h> | 11 | #include <linux/types.h> |
@@ -9,38 +15,96 @@ | |||
9 | #include <asm/tlbflush.h> | 15 | #include <asm/tlbflush.h> |
10 | #include "lg.h" | 16 | #include "lg.h" |
11 | 17 | ||
18 | /*M:008 We hold reference to pages, which prevents them from being swapped. | ||
19 | * It'd be nice to have a callback in the "struct mm_struct" when Linux wants | ||
20 | * to swap out. If we had this, and a shrinker callback to trim PTE pages, we | ||
21 | * could probably consider launching Guests as non-root. :*/ | ||
22 | |||
23 | /*H:300 | ||
24 | * The Page Table Code | ||
25 | * | ||
26 | * We use two-level page tables for the Guest. If you're not entirely | ||
27 | * comfortable with virtual addresses, physical addresses and page tables then | ||
28 | * I recommend you review lguest.c's "Page Table Handling" (with diagrams!). | ||
29 | * | ||
30 | * The Guest keeps page tables, but we maintain the actual ones here: these are | ||
31 | * called "shadow" page tables. Which is a very Guest-centric name: these are | ||
32 | * the real page tables the CPU uses, although we keep them up to date to | ||
33 | * reflect the Guest's. (See what I mean about weird naming? Since when do | ||
34 | * shadows reflect anything?) | ||
35 | * | ||
36 | * Anyway, this is the most complicated part of the Host code. There are seven | ||
37 | * parts to this: | ||
38 | * (i) Setting up a page table entry for the Guest when it faults, | ||
39 | * (ii) Setting up the page table entry for the Guest stack, | ||
40 | * (iii) Setting up a page table entry when the Guest tells us it has changed, | ||
41 | * (iv) Switching page tables, | ||
42 | * (v) Flushing (thowing away) page tables, | ||
43 | * (vi) Mapping the Switcher when the Guest is about to run, | ||
44 | * (vii) Setting up the page tables initially. | ||
45 | :*/ | ||
46 | |||
47 | /* Pages a 4k long, and each page table entry is 4 bytes long, giving us 1024 | ||
48 | * (or 2^10) entries per page. */ | ||
12 | #define PTES_PER_PAGE_SHIFT 10 | 49 | #define PTES_PER_PAGE_SHIFT 10 |
13 | #define PTES_PER_PAGE (1 << PTES_PER_PAGE_SHIFT) | 50 | #define PTES_PER_PAGE (1 << PTES_PER_PAGE_SHIFT) |
51 | |||
52 | /* 1024 entries in a page table page maps 1024 pages: 4MB. The Switcher is | ||
53 | * conveniently placed at the top 4MB, so it uses a separate, complete PTE | ||
54 | * page. */ | ||
14 | #define SWITCHER_PGD_INDEX (PTES_PER_PAGE - 1) | 55 | #define SWITCHER_PGD_INDEX (PTES_PER_PAGE - 1) |
15 | 56 | ||
57 | /* We actually need a separate PTE page for each CPU. Remember that after the | ||
58 | * Switcher code itself comes two pages for each CPU, and we don't want this | ||
59 | * CPU's guest to see the pages of any other CPU. */ | ||
16 | static DEFINE_PER_CPU(spte_t *, switcher_pte_pages); | 60 | static DEFINE_PER_CPU(spte_t *, switcher_pte_pages); |
17 | #define switcher_pte_page(cpu) per_cpu(switcher_pte_pages, cpu) | 61 | #define switcher_pte_page(cpu) per_cpu(switcher_pte_pages, cpu) |
18 | 62 | ||
63 | /*H:320 With our shadow and Guest types established, we need to deal with | ||
64 | * them: the page table code is curly enough to need helper functions to keep | ||
65 | * it clear and clean. | ||
66 | * | ||
67 | * The first helper takes a virtual address, and says which entry in the top | ||
68 | * level page table deals with that address. Since each top level entry deals | ||
69 | * with 4M, this effectively divides by 4M. */ | ||
19 | static unsigned vaddr_to_pgd_index(unsigned long vaddr) | 70 | static unsigned vaddr_to_pgd_index(unsigned long vaddr) |
20 | { | 71 | { |
21 | return vaddr >> (PAGE_SHIFT + PTES_PER_PAGE_SHIFT); | 72 | return vaddr >> (PAGE_SHIFT + PTES_PER_PAGE_SHIFT); |
22 | } | 73 | } |
23 | 74 | ||
24 | /* These access the shadow versions (ie. the ones used by the CPU). */ | 75 | /* There are two functions which return pointers to the shadow (aka "real") |
76 | * page tables. | ||
77 | * | ||
78 | * spgd_addr() takes the virtual address and returns a pointer to the top-level | ||
79 | * page directory entry for that address. Since we keep track of several page | ||
80 | * tables, the "i" argument tells us which one we're interested in (it's | ||
81 | * usually the current one). */ | ||
25 | static spgd_t *spgd_addr(struct lguest *lg, u32 i, unsigned long vaddr) | 82 | static spgd_t *spgd_addr(struct lguest *lg, u32 i, unsigned long vaddr) |
26 | { | 83 | { |
27 | unsigned int index = vaddr_to_pgd_index(vaddr); | 84 | unsigned int index = vaddr_to_pgd_index(vaddr); |
28 | 85 | ||
86 | /* We kill any Guest trying to touch the Switcher addresses. */ | ||
29 | if (index >= SWITCHER_PGD_INDEX) { | 87 | if (index >= SWITCHER_PGD_INDEX) { |
30 | kill_guest(lg, "attempt to access switcher pages"); | 88 | kill_guest(lg, "attempt to access switcher pages"); |
31 | index = 0; | 89 | index = 0; |
32 | } | 90 | } |
91 | /* Return a pointer index'th pgd entry for the i'th page table. */ | ||
33 | return &lg->pgdirs[i].pgdir[index]; | 92 | return &lg->pgdirs[i].pgdir[index]; |
34 | } | 93 | } |
35 | 94 | ||
95 | /* This routine then takes the PGD entry given above, which contains the | ||
96 | * address of the PTE page. It then returns a pointer to the PTE entry for the | ||
97 | * given address. */ | ||
36 | static spte_t *spte_addr(struct lguest *lg, spgd_t spgd, unsigned long vaddr) | 98 | static spte_t *spte_addr(struct lguest *lg, spgd_t spgd, unsigned long vaddr) |
37 | { | 99 | { |
38 | spte_t *page = __va(spgd.pfn << PAGE_SHIFT); | 100 | spte_t *page = __va(spgd.pfn << PAGE_SHIFT); |
101 | /* You should never call this if the PGD entry wasn't valid */ | ||
39 | BUG_ON(!(spgd.flags & _PAGE_PRESENT)); | 102 | BUG_ON(!(spgd.flags & _PAGE_PRESENT)); |
40 | return &page[(vaddr >> PAGE_SHIFT) % PTES_PER_PAGE]; | 103 | return &page[(vaddr >> PAGE_SHIFT) % PTES_PER_PAGE]; |
41 | } | 104 | } |
42 | 105 | ||
43 | /* These access the guest versions. */ | 106 | /* These two functions just like the above two, except they access the Guest |
107 | * page tables. Hence they return a Guest address. */ | ||
44 | static unsigned long gpgd_addr(struct lguest *lg, unsigned long vaddr) | 108 | static unsigned long gpgd_addr(struct lguest *lg, unsigned long vaddr) |
45 | { | 109 | { |
46 | unsigned int index = vaddr >> (PAGE_SHIFT + PTES_PER_PAGE_SHIFT); | 110 | unsigned int index = vaddr >> (PAGE_SHIFT + PTES_PER_PAGE_SHIFT); |
@@ -55,12 +119,24 @@ static unsigned long gpte_addr(struct lguest *lg, | |||
55 | return gpage + ((vaddr>>PAGE_SHIFT) % PTES_PER_PAGE) * sizeof(gpte_t); | 119 | return gpage + ((vaddr>>PAGE_SHIFT) % PTES_PER_PAGE) * sizeof(gpte_t); |
56 | } | 120 | } |
57 | 121 | ||
58 | /* Do a virtual -> physical mapping on a user page. */ | 122 | /*H:350 This routine takes a page number given by the Guest and converts it to |
123 | * an actual, physical page number. It can fail for several reasons: the | ||
124 | * virtual address might not be mapped by the Launcher, the write flag is set | ||
125 | * and the page is read-only, or the write flag was set and the page was | ||
126 | * shared so had to be copied, but we ran out of memory. | ||
127 | * | ||
128 | * This holds a reference to the page, so release_pte() is careful to | ||
129 | * put that back. */ | ||
59 | static unsigned long get_pfn(unsigned long virtpfn, int write) | 130 | static unsigned long get_pfn(unsigned long virtpfn, int write) |
60 | { | 131 | { |
61 | struct page *page; | 132 | struct page *page; |
133 | /* This value indicates failure. */ | ||
62 | unsigned long ret = -1UL; | 134 | unsigned long ret = -1UL; |
63 | 135 | ||
136 | /* get_user_pages() is a complex interface: it gets the "struct | ||
137 | * vm_area_struct" and "struct page" assocated with a range of pages. | ||
138 | * It also needs the task's mmap_sem held, and is not very quick. | ||
139 | * It returns the number of pages it got. */ | ||
64 | down_read(¤t->mm->mmap_sem); | 140 | down_read(¤t->mm->mmap_sem); |
65 | if (get_user_pages(current, current->mm, virtpfn << PAGE_SHIFT, | 141 | if (get_user_pages(current, current->mm, virtpfn << PAGE_SHIFT, |
66 | 1, write, 1, &page, NULL) == 1) | 142 | 1, write, 1, &page, NULL) == 1) |
@@ -69,28 +145,47 @@ static unsigned long get_pfn(unsigned long virtpfn, int write) | |||
69 | return ret; | 145 | return ret; |
70 | } | 146 | } |
71 | 147 | ||
148 | /*H:340 Converting a Guest page table entry to a shadow (ie. real) page table | ||
149 | * entry can be a little tricky. The flags are (almost) the same, but the | ||
150 | * Guest PTE contains a virtual page number: the CPU needs the real page | ||
151 | * number. */ | ||
72 | static spte_t gpte_to_spte(struct lguest *lg, gpte_t gpte, int write) | 152 | static spte_t gpte_to_spte(struct lguest *lg, gpte_t gpte, int write) |
73 | { | 153 | { |
74 | spte_t spte; | 154 | spte_t spte; |
75 | unsigned long pfn; | 155 | unsigned long pfn; |
76 | 156 | ||
77 | /* We ignore the global flag. */ | 157 | /* The Guest sets the global flag, because it thinks that it is using |
158 | * PGE. We only told it to use PGE so it would tell us whether it was | ||
159 | * flushing a kernel mapping or a userspace mapping. We don't actually | ||
160 | * use the global bit, so throw it away. */ | ||
78 | spte.flags = (gpte.flags & ~_PAGE_GLOBAL); | 161 | spte.flags = (gpte.flags & ~_PAGE_GLOBAL); |
162 | |||
163 | /* We need a temporary "unsigned long" variable to hold the answer from | ||
164 | * get_pfn(), because it returns 0xFFFFFFFF on failure, which wouldn't | ||
165 | * fit in spte.pfn. get_pfn() finds the real physical number of the | ||
166 | * page, given the virtual number. */ | ||
79 | pfn = get_pfn(gpte.pfn, write); | 167 | pfn = get_pfn(gpte.pfn, write); |
80 | if (pfn == -1UL) { | 168 | if (pfn == -1UL) { |
81 | kill_guest(lg, "failed to get page %u", gpte.pfn); | 169 | kill_guest(lg, "failed to get page %u", gpte.pfn); |
82 | /* Must not put_page() bogus page on cleanup. */ | 170 | /* When we destroy the Guest, we'll go through the shadow page |
171 | * tables and release_pte() them. Make sure we don't think | ||
172 | * this one is valid! */ | ||
83 | spte.flags = 0; | 173 | spte.flags = 0; |
84 | } | 174 | } |
175 | /* Now we assign the page number, and our shadow PTE is complete. */ | ||
85 | spte.pfn = pfn; | 176 | spte.pfn = pfn; |
86 | return spte; | 177 | return spte; |
87 | } | 178 | } |
88 | 179 | ||
180 | /*H:460 And to complete the chain, release_pte() looks like this: */ | ||
89 | static void release_pte(spte_t pte) | 181 | static void release_pte(spte_t pte) |
90 | { | 182 | { |
183 | /* Remember that get_user_pages() took a reference to the page, in | ||
184 | * get_pfn()? We have to put it back now. */ | ||
91 | if (pte.flags & _PAGE_PRESENT) | 185 | if (pte.flags & _PAGE_PRESENT) |
92 | put_page(pfn_to_page(pte.pfn)); | 186 | put_page(pfn_to_page(pte.pfn)); |
93 | } | 187 | } |
188 | /*:*/ | ||
94 | 189 | ||
95 | static void check_gpte(struct lguest *lg, gpte_t gpte) | 190 | static void check_gpte(struct lguest *lg, gpte_t gpte) |
96 | { | 191 | { |
@@ -104,11 +199,16 @@ static void check_gpgd(struct lguest *lg, gpgd_t gpgd) | |||
104 | kill_guest(lg, "bad page directory entry"); | 199 | kill_guest(lg, "bad page directory entry"); |
105 | } | 200 | } |
106 | 201 | ||
107 | /* FIXME: We hold reference to pages, which prevents them from being | 202 | /*H:330 |
108 | swapped. It'd be nice to have a callback when Linux wants to swap out. */ | 203 | * (i) Setting up a page table entry for the Guest when it faults |
109 | 204 | * | |
110 | /* We fault pages in, which allows us to update accessed/dirty bits. | 205 | * We saw this call in run_guest(): when we see a page fault in the Guest, we |
111 | * Return true if we got page. */ | 206 | * come here. That's because we only set up the shadow page tables lazily as |
207 | * they're needed, so we get page faults all the time and quietly fix them up | ||
208 | * and return to the Guest without it knowing. | ||
209 | * | ||
210 | * If we fixed up the fault (ie. we mapped the address), this routine returns | ||
211 | * true. */ | ||
112 | int demand_page(struct lguest *lg, unsigned long vaddr, int errcode) | 212 | int demand_page(struct lguest *lg, unsigned long vaddr, int errcode) |
113 | { | 213 | { |
114 | gpgd_t gpgd; | 214 | gpgd_t gpgd; |
@@ -117,106 +217,161 @@ int demand_page(struct lguest *lg, unsigned long vaddr, int errcode) | |||
117 | gpte_t gpte; | 217 | gpte_t gpte; |
118 | spte_t *spte; | 218 | spte_t *spte; |
119 | 219 | ||
220 | /* First step: get the top-level Guest page table entry. */ | ||
120 | gpgd = mkgpgd(lgread_u32(lg, gpgd_addr(lg, vaddr))); | 221 | gpgd = mkgpgd(lgread_u32(lg, gpgd_addr(lg, vaddr))); |
222 | /* Toplevel not present? We can't map it in. */ | ||
121 | if (!(gpgd.flags & _PAGE_PRESENT)) | 223 | if (!(gpgd.flags & _PAGE_PRESENT)) |
122 | return 0; | 224 | return 0; |
123 | 225 | ||
226 | /* Now look at the matching shadow entry. */ | ||
124 | spgd = spgd_addr(lg, lg->pgdidx, vaddr); | 227 | spgd = spgd_addr(lg, lg->pgdidx, vaddr); |
125 | if (!(spgd->flags & _PAGE_PRESENT)) { | 228 | if (!(spgd->flags & _PAGE_PRESENT)) { |
126 | /* Get a page of PTEs for them. */ | 229 | /* No shadow entry: allocate a new shadow PTE page. */ |
127 | unsigned long ptepage = get_zeroed_page(GFP_KERNEL); | 230 | unsigned long ptepage = get_zeroed_page(GFP_KERNEL); |
128 | /* FIXME: Steal from self in this case? */ | 231 | /* This is not really the Guest's fault, but killing it is |
232 | * simple for this corner case. */ | ||
129 | if (!ptepage) { | 233 | if (!ptepage) { |
130 | kill_guest(lg, "out of memory allocating pte page"); | 234 | kill_guest(lg, "out of memory allocating pte page"); |
131 | return 0; | 235 | return 0; |
132 | } | 236 | } |
237 | /* We check that the Guest pgd is OK. */ | ||
133 | check_gpgd(lg, gpgd); | 238 | check_gpgd(lg, gpgd); |
239 | /* And we copy the flags to the shadow PGD entry. The page | ||
240 | * number in the shadow PGD is the page we just allocated. */ | ||
134 | spgd->raw.val = (__pa(ptepage) | gpgd.flags); | 241 | spgd->raw.val = (__pa(ptepage) | gpgd.flags); |
135 | } | 242 | } |
136 | 243 | ||
244 | /* OK, now we look at the lower level in the Guest page table: keep its | ||
245 | * address, because we might update it later. */ | ||
137 | gpte_ptr = gpte_addr(lg, gpgd, vaddr); | 246 | gpte_ptr = gpte_addr(lg, gpgd, vaddr); |
138 | gpte = mkgpte(lgread_u32(lg, gpte_ptr)); | 247 | gpte = mkgpte(lgread_u32(lg, gpte_ptr)); |
139 | 248 | ||
140 | /* No page? */ | 249 | /* If this page isn't in the Guest page tables, we can't page it in. */ |
141 | if (!(gpte.flags & _PAGE_PRESENT)) | 250 | if (!(gpte.flags & _PAGE_PRESENT)) |
142 | return 0; | 251 | return 0; |
143 | 252 | ||
144 | /* Write to read-only page? */ | 253 | /* Check they're not trying to write to a page the Guest wants |
254 | * read-only (bit 2 of errcode == write). */ | ||
145 | if ((errcode & 2) && !(gpte.flags & _PAGE_RW)) | 255 | if ((errcode & 2) && !(gpte.flags & _PAGE_RW)) |
146 | return 0; | 256 | return 0; |
147 | 257 | ||
148 | /* User access to a non-user page? */ | 258 | /* User access to a kernel page? (bit 3 == user access) */ |
149 | if ((errcode & 4) && !(gpte.flags & _PAGE_USER)) | 259 | if ((errcode & 4) && !(gpte.flags & _PAGE_USER)) |
150 | return 0; | 260 | return 0; |
151 | 261 | ||
262 | /* Check that the Guest PTE flags are OK, and the page number is below | ||
263 | * the pfn_limit (ie. not mapping the Launcher binary). */ | ||
152 | check_gpte(lg, gpte); | 264 | check_gpte(lg, gpte); |
265 | /* Add the _PAGE_ACCESSED and (for a write) _PAGE_DIRTY flag */ | ||
153 | gpte.flags |= _PAGE_ACCESSED; | 266 | gpte.flags |= _PAGE_ACCESSED; |
154 | if (errcode & 2) | 267 | if (errcode & 2) |
155 | gpte.flags |= _PAGE_DIRTY; | 268 | gpte.flags |= _PAGE_DIRTY; |
156 | 269 | ||
157 | /* We're done with the old pte. */ | 270 | /* Get the pointer to the shadow PTE entry we're going to set. */ |
158 | spte = spte_addr(lg, *spgd, vaddr); | 271 | spte = spte_addr(lg, *spgd, vaddr); |
272 | /* If there was a valid shadow PTE entry here before, we release it. | ||
273 | * This can happen with a write to a previously read-only entry. */ | ||
159 | release_pte(*spte); | 274 | release_pte(*spte); |
160 | 275 | ||
161 | /* We don't make it writable if this isn't a write: later | 276 | /* If this is a write, we insist that the Guest page is writable (the |
162 | * write will fault so we can set dirty bit in guest. */ | 277 | * final arg to gpte_to_spte()). */ |
163 | if (gpte.flags & _PAGE_DIRTY) | 278 | if (gpte.flags & _PAGE_DIRTY) |
164 | *spte = gpte_to_spte(lg, gpte, 1); | 279 | *spte = gpte_to_spte(lg, gpte, 1); |
165 | else { | 280 | else { |
281 | /* If this is a read, don't set the "writable" bit in the page | ||
282 | * table entry, even if the Guest says it's writable. That way | ||
283 | * we come back here when a write does actually ocur, so we can | ||
284 | * update the Guest's _PAGE_DIRTY flag. */ | ||
166 | gpte_t ro_gpte = gpte; | 285 | gpte_t ro_gpte = gpte; |
167 | ro_gpte.flags &= ~_PAGE_RW; | 286 | ro_gpte.flags &= ~_PAGE_RW; |
168 | *spte = gpte_to_spte(lg, ro_gpte, 0); | 287 | *spte = gpte_to_spte(lg, ro_gpte, 0); |
169 | } | 288 | } |
170 | 289 | ||
171 | /* Now we update dirty/accessed on guest. */ | 290 | /* Finally, we write the Guest PTE entry back: we've set the |
291 | * _PAGE_ACCESSED and maybe the _PAGE_DIRTY flags. */ | ||
172 | lgwrite_u32(lg, gpte_ptr, gpte.raw.val); | 292 | lgwrite_u32(lg, gpte_ptr, gpte.raw.val); |
293 | |||
294 | /* We succeeded in mapping the page! */ | ||
173 | return 1; | 295 | return 1; |
174 | } | 296 | } |
175 | 297 | ||
176 | /* This is much faster than the full demand_page logic. */ | 298 | /*H:360 (ii) Setting up the page table entry for the Guest stack. |
299 | * | ||
300 | * Remember pin_stack_pages() which makes sure the stack is mapped? It could | ||
301 | * simply call demand_page(), but as we've seen that logic is quite long, and | ||
302 | * usually the stack pages are already mapped anyway, so it's not required. | ||
303 | * | ||
304 | * This is a quick version which answers the question: is this virtual address | ||
305 | * mapped by the shadow page tables, and is it writable? */ | ||
177 | static int page_writable(struct lguest *lg, unsigned long vaddr) | 306 | static int page_writable(struct lguest *lg, unsigned long vaddr) |
178 | { | 307 | { |
179 | spgd_t *spgd; | 308 | spgd_t *spgd; |
180 | unsigned long flags; | 309 | unsigned long flags; |
181 | 310 | ||
311 | /* Look at the top level entry: is it present? */ | ||
182 | spgd = spgd_addr(lg, lg->pgdidx, vaddr); | 312 | spgd = spgd_addr(lg, lg->pgdidx, vaddr); |
183 | if (!(spgd->flags & _PAGE_PRESENT)) | 313 | if (!(spgd->flags & _PAGE_PRESENT)) |
184 | return 0; | 314 | return 0; |
185 | 315 | ||
316 | /* Check the flags on the pte entry itself: it must be present and | ||
317 | * writable. */ | ||
186 | flags = spte_addr(lg, *spgd, vaddr)->flags; | 318 | flags = spte_addr(lg, *spgd, vaddr)->flags; |
187 | return (flags & (_PAGE_PRESENT|_PAGE_RW)) == (_PAGE_PRESENT|_PAGE_RW); | 319 | return (flags & (_PAGE_PRESENT|_PAGE_RW)) == (_PAGE_PRESENT|_PAGE_RW); |
188 | } | 320 | } |
189 | 321 | ||
322 | /* So, when pin_stack_pages() asks us to pin a page, we check if it's already | ||
323 | * in the page tables, and if not, we call demand_page() with error code 2 | ||
324 | * (meaning "write"). */ | ||
190 | void pin_page(struct lguest *lg, unsigned long vaddr) | 325 | void pin_page(struct lguest *lg, unsigned long vaddr) |
191 | { | 326 | { |
192 | if (!page_writable(lg, vaddr) && !demand_page(lg, vaddr, 2)) | 327 | if (!page_writable(lg, vaddr) && !demand_page(lg, vaddr, 2)) |
193 | kill_guest(lg, "bad stack page %#lx", vaddr); | 328 | kill_guest(lg, "bad stack page %#lx", vaddr); |
194 | } | 329 | } |
195 | 330 | ||
331 | /*H:450 If we chase down the release_pgd() code, it looks like this: */ | ||
196 | static void release_pgd(struct lguest *lg, spgd_t *spgd) | 332 | static void release_pgd(struct lguest *lg, spgd_t *spgd) |
197 | { | 333 | { |
334 | /* If the entry's not present, there's nothing to release. */ | ||
198 | if (spgd->flags & _PAGE_PRESENT) { | 335 | if (spgd->flags & _PAGE_PRESENT) { |
199 | unsigned int i; | 336 | unsigned int i; |
337 | /* Converting the pfn to find the actual PTE page is easy: turn | ||
338 | * the page number into a physical address, then convert to a | ||
339 | * virtual address (easy for kernel pages like this one). */ | ||
200 | spte_t *ptepage = __va(spgd->pfn << PAGE_SHIFT); | 340 | spte_t *ptepage = __va(spgd->pfn << PAGE_SHIFT); |
341 | /* For each entry in the page, we might need to release it. */ | ||
201 | for (i = 0; i < PTES_PER_PAGE; i++) | 342 | for (i = 0; i < PTES_PER_PAGE; i++) |
202 | release_pte(ptepage[i]); | 343 | release_pte(ptepage[i]); |
344 | /* Now we can free the page of PTEs */ | ||
203 | free_page((long)ptepage); | 345 | free_page((long)ptepage); |
346 | /* And zero out the PGD entry we we never release it twice. */ | ||
204 | spgd->raw.val = 0; | 347 | spgd->raw.val = 0; |
205 | } | 348 | } |
206 | } | 349 | } |
207 | 350 | ||
351 | /*H:440 (v) Flushing (thowing away) page tables, | ||
352 | * | ||
353 | * We saw flush_user_mappings() called when we re-used a top-level pgdir page. | ||
354 | * It simply releases every PTE page from 0 up to the kernel address. */ | ||
208 | static void flush_user_mappings(struct lguest *lg, int idx) | 355 | static void flush_user_mappings(struct lguest *lg, int idx) |
209 | { | 356 | { |
210 | unsigned int i; | 357 | unsigned int i; |
358 | /* Release every pgd entry up to the kernel's address. */ | ||
211 | for (i = 0; i < vaddr_to_pgd_index(lg->page_offset); i++) | 359 | for (i = 0; i < vaddr_to_pgd_index(lg->page_offset); i++) |
212 | release_pgd(lg, lg->pgdirs[idx].pgdir + i); | 360 | release_pgd(lg, lg->pgdirs[idx].pgdir + i); |
213 | } | 361 | } |
214 | 362 | ||
363 | /* The Guest also has a hypercall to do this manually: it's used when a large | ||
364 | * number of mappings have been changed. */ | ||
215 | void guest_pagetable_flush_user(struct lguest *lg) | 365 | void guest_pagetable_flush_user(struct lguest *lg) |
216 | { | 366 | { |
367 | /* Drop the userspace part of the current page table. */ | ||
217 | flush_user_mappings(lg, lg->pgdidx); | 368 | flush_user_mappings(lg, lg->pgdidx); |
218 | } | 369 | } |
370 | /*:*/ | ||
219 | 371 | ||
372 | /* We keep several page tables. This is a simple routine to find the page | ||
373 | * table (if any) corresponding to this top-level address the Guest has given | ||
374 | * us. */ | ||
220 | static unsigned int find_pgdir(struct lguest *lg, unsigned long pgtable) | 375 | static unsigned int find_pgdir(struct lguest *lg, unsigned long pgtable) |
221 | { | 376 | { |
222 | unsigned int i; | 377 | unsigned int i; |
@@ -226,21 +381,30 @@ static unsigned int find_pgdir(struct lguest *lg, unsigned long pgtable) | |||
226 | return i; | 381 | return i; |
227 | } | 382 | } |
228 | 383 | ||
384 | /*H:435 And this is us, creating the new page directory. If we really do | ||
385 | * allocate a new one (and so the kernel parts are not there), we set | ||
386 | * blank_pgdir. */ | ||
229 | static unsigned int new_pgdir(struct lguest *lg, | 387 | static unsigned int new_pgdir(struct lguest *lg, |
230 | unsigned long cr3, | 388 | unsigned long cr3, |
231 | int *blank_pgdir) | 389 | int *blank_pgdir) |
232 | { | 390 | { |
233 | unsigned int next; | 391 | unsigned int next; |
234 | 392 | ||
393 | /* We pick one entry at random to throw out. Choosing the Least | ||
394 | * Recently Used might be better, but this is easy. */ | ||
235 | next = random32() % ARRAY_SIZE(lg->pgdirs); | 395 | next = random32() % ARRAY_SIZE(lg->pgdirs); |
396 | /* If it's never been allocated at all before, try now. */ | ||
236 | if (!lg->pgdirs[next].pgdir) { | 397 | if (!lg->pgdirs[next].pgdir) { |
237 | lg->pgdirs[next].pgdir = (spgd_t *)get_zeroed_page(GFP_KERNEL); | 398 | lg->pgdirs[next].pgdir = (spgd_t *)get_zeroed_page(GFP_KERNEL); |
399 | /* If the allocation fails, just keep using the one we have */ | ||
238 | if (!lg->pgdirs[next].pgdir) | 400 | if (!lg->pgdirs[next].pgdir) |
239 | next = lg->pgdidx; | 401 | next = lg->pgdidx; |
240 | else | 402 | else |
241 | /* There are no mappings: you'll need to re-pin */ | 403 | /* This is a blank page, so there are no kernel |
404 | * mappings: caller must map the stack! */ | ||
242 | *blank_pgdir = 1; | 405 | *blank_pgdir = 1; |
243 | } | 406 | } |
407 | /* Record which Guest toplevel this shadows. */ | ||
244 | lg->pgdirs[next].cr3 = cr3; | 408 | lg->pgdirs[next].cr3 = cr3; |
245 | /* Release all the non-kernel mappings. */ | 409 | /* Release all the non-kernel mappings. */ |
246 | flush_user_mappings(lg, next); | 410 | flush_user_mappings(lg, next); |
@@ -248,82 +412,161 @@ static unsigned int new_pgdir(struct lguest *lg, | |||
248 | return next; | 412 | return next; |
249 | } | 413 | } |
250 | 414 | ||
415 | /*H:430 (iv) Switching page tables | ||
416 | * | ||
417 | * This is what happens when the Guest changes page tables (ie. changes the | ||
418 | * top-level pgdir). This happens on almost every context switch. */ | ||
251 | void guest_new_pagetable(struct lguest *lg, unsigned long pgtable) | 419 | void guest_new_pagetable(struct lguest *lg, unsigned long pgtable) |
252 | { | 420 | { |
253 | int newpgdir, repin = 0; | 421 | int newpgdir, repin = 0; |
254 | 422 | ||
423 | /* Look to see if we have this one already. */ | ||
255 | newpgdir = find_pgdir(lg, pgtable); | 424 | newpgdir = find_pgdir(lg, pgtable); |
425 | /* If not, we allocate or mug an existing one: if it's a fresh one, | ||
426 | * repin gets set to 1. */ | ||
256 | if (newpgdir == ARRAY_SIZE(lg->pgdirs)) | 427 | if (newpgdir == ARRAY_SIZE(lg->pgdirs)) |
257 | newpgdir = new_pgdir(lg, pgtable, &repin); | 428 | newpgdir = new_pgdir(lg, pgtable, &repin); |
429 | /* Change the current pgd index to the new one. */ | ||
258 | lg->pgdidx = newpgdir; | 430 | lg->pgdidx = newpgdir; |
431 | /* If it was completely blank, we map in the Guest kernel stack */ | ||
259 | if (repin) | 432 | if (repin) |
260 | pin_stack_pages(lg); | 433 | pin_stack_pages(lg); |
261 | } | 434 | } |
262 | 435 | ||
436 | /*H:470 Finally, a routine which throws away everything: all PGD entries in all | ||
437 | * the shadow page tables. This is used when we destroy the Guest. */ | ||
263 | static void release_all_pagetables(struct lguest *lg) | 438 | static void release_all_pagetables(struct lguest *lg) |
264 | { | 439 | { |
265 | unsigned int i, j; | 440 | unsigned int i, j; |
266 | 441 | ||
442 | /* Every shadow pagetable this Guest has */ | ||
267 | for (i = 0; i < ARRAY_SIZE(lg->pgdirs); i++) | 443 | for (i = 0; i < ARRAY_SIZE(lg->pgdirs); i++) |
268 | if (lg->pgdirs[i].pgdir) | 444 | if (lg->pgdirs[i].pgdir) |
445 | /* Every PGD entry except the Switcher at the top */ | ||
269 | for (j = 0; j < SWITCHER_PGD_INDEX; j++) | 446 | for (j = 0; j < SWITCHER_PGD_INDEX; j++) |
270 | release_pgd(lg, lg->pgdirs[i].pgdir + j); | 447 | release_pgd(lg, lg->pgdirs[i].pgdir + j); |
271 | } | 448 | } |
272 | 449 | ||
450 | /* We also throw away everything when a Guest tells us it's changed a kernel | ||
451 | * mapping. Since kernel mappings are in every page table, it's easiest to | ||
452 | * throw them all away. This is amazingly slow, but thankfully rare. */ | ||
273 | void guest_pagetable_clear_all(struct lguest *lg) | 453 | void guest_pagetable_clear_all(struct lguest *lg) |
274 | { | 454 | { |
275 | release_all_pagetables(lg); | 455 | release_all_pagetables(lg); |
456 | /* We need the Guest kernel stack mapped again. */ | ||
276 | pin_stack_pages(lg); | 457 | pin_stack_pages(lg); |
277 | } | 458 | } |
278 | 459 | ||
460 | /*H:420 This is the routine which actually sets the page table entry for then | ||
461 | * "idx"'th shadow page table. | ||
462 | * | ||
463 | * Normally, we can just throw out the old entry and replace it with 0: if they | ||
464 | * use it demand_page() will put the new entry in. We need to do this anyway: | ||
465 | * The Guest expects _PAGE_ACCESSED to be set on its PTE the first time a page | ||
466 | * is read from, and _PAGE_DIRTY when it's written to. | ||
467 | * | ||
468 | * But Avi Kivity pointed out that most Operating Systems (Linux included) set | ||
469 | * these bits on PTEs immediately anyway. This is done to save the CPU from | ||
470 | * having to update them, but it helps us the same way: if they set | ||
471 | * _PAGE_ACCESSED then we can put a read-only PTE entry in immediately, and if | ||
472 | * they set _PAGE_DIRTY then we can put a writable PTE entry in immediately. | ||
473 | */ | ||
279 | static void do_set_pte(struct lguest *lg, int idx, | 474 | static void do_set_pte(struct lguest *lg, int idx, |
280 | unsigned long vaddr, gpte_t gpte) | 475 | unsigned long vaddr, gpte_t gpte) |
281 | { | 476 | { |
477 | /* Look up the matching shadow page directot entry. */ | ||
282 | spgd_t *spgd = spgd_addr(lg, idx, vaddr); | 478 | spgd_t *spgd = spgd_addr(lg, idx, vaddr); |
479 | |||
480 | /* If the top level isn't present, there's no entry to update. */ | ||
283 | if (spgd->flags & _PAGE_PRESENT) { | 481 | if (spgd->flags & _PAGE_PRESENT) { |
482 | /* Otherwise, we start by releasing the existing entry. */ | ||
284 | spte_t *spte = spte_addr(lg, *spgd, vaddr); | 483 | spte_t *spte = spte_addr(lg, *spgd, vaddr); |
285 | release_pte(*spte); | 484 | release_pte(*spte); |
485 | |||
486 | /* If they're setting this entry as dirty or accessed, we might | ||
487 | * as well put that entry they've given us in now. This shaves | ||
488 | * 10% off a copy-on-write micro-benchmark. */ | ||
286 | if (gpte.flags & (_PAGE_DIRTY | _PAGE_ACCESSED)) { | 489 | if (gpte.flags & (_PAGE_DIRTY | _PAGE_ACCESSED)) { |
287 | check_gpte(lg, gpte); | 490 | check_gpte(lg, gpte); |
288 | *spte = gpte_to_spte(lg, gpte, gpte.flags&_PAGE_DIRTY); | 491 | *spte = gpte_to_spte(lg, gpte, gpte.flags&_PAGE_DIRTY); |
289 | } else | 492 | } else |
493 | /* Otherwise we can demand_page() it in later. */ | ||
290 | spte->raw.val = 0; | 494 | spte->raw.val = 0; |
291 | } | 495 | } |
292 | } | 496 | } |
293 | 497 | ||
498 | /*H:410 Updating a PTE entry is a little trickier. | ||
499 | * | ||
500 | * We keep track of several different page tables (the Guest uses one for each | ||
501 | * process, so it makes sense to cache at least a few). Each of these have | ||
502 | * identical kernel parts: ie. every mapping above PAGE_OFFSET is the same for | ||
503 | * all processes. So when the page table above that address changes, we update | ||
504 | * all the page tables, not just the current one. This is rare. | ||
505 | * | ||
506 | * The benefit is that when we have to track a new page table, we can copy keep | ||
507 | * all the kernel mappings. This speeds up context switch immensely. */ | ||
294 | void guest_set_pte(struct lguest *lg, | 508 | void guest_set_pte(struct lguest *lg, |
295 | unsigned long cr3, unsigned long vaddr, gpte_t gpte) | 509 | unsigned long cr3, unsigned long vaddr, gpte_t gpte) |
296 | { | 510 | { |
297 | /* Kernel mappings must be changed on all top levels. */ | 511 | /* Kernel mappings must be changed on all top levels. Slow, but |
512 | * doesn't happen often. */ | ||
298 | if (vaddr >= lg->page_offset) { | 513 | if (vaddr >= lg->page_offset) { |
299 | unsigned int i; | 514 | unsigned int i; |
300 | for (i = 0; i < ARRAY_SIZE(lg->pgdirs); i++) | 515 | for (i = 0; i < ARRAY_SIZE(lg->pgdirs); i++) |
301 | if (lg->pgdirs[i].pgdir) | 516 | if (lg->pgdirs[i].pgdir) |
302 | do_set_pte(lg, i, vaddr, gpte); | 517 | do_set_pte(lg, i, vaddr, gpte); |
303 | } else { | 518 | } else { |
519 | /* Is this page table one we have a shadow for? */ | ||
304 | int pgdir = find_pgdir(lg, cr3); | 520 | int pgdir = find_pgdir(lg, cr3); |
305 | if (pgdir != ARRAY_SIZE(lg->pgdirs)) | 521 | if (pgdir != ARRAY_SIZE(lg->pgdirs)) |
522 | /* If so, do the update. */ | ||
306 | do_set_pte(lg, pgdir, vaddr, gpte); | 523 | do_set_pte(lg, pgdir, vaddr, gpte); |
307 | } | 524 | } |
308 | } | 525 | } |
309 | 526 | ||
527 | /*H:400 | ||
528 | * (iii) Setting up a page table entry when the Guest tells us it has changed. | ||
529 | * | ||
530 | * Just like we did in interrupts_and_traps.c, it makes sense for us to deal | ||
531 | * with the other side of page tables while we're here: what happens when the | ||
532 | * Guest asks for a page table to be updated? | ||
533 | * | ||
534 | * We already saw that demand_page() will fill in the shadow page tables when | ||
535 | * needed, so we can simply remove shadow page table entries whenever the Guest | ||
536 | * tells us they've changed. When the Guest tries to use the new entry it will | ||
537 | * fault and demand_page() will fix it up. | ||
538 | * | ||
539 | * So with that in mind here's our code to to update a (top-level) PGD entry: | ||
540 | */ | ||
310 | void guest_set_pmd(struct lguest *lg, unsigned long cr3, u32 idx) | 541 | void guest_set_pmd(struct lguest *lg, unsigned long cr3, u32 idx) |
311 | { | 542 | { |
312 | int pgdir; | 543 | int pgdir; |
313 | 544 | ||
545 | /* The kernel seems to try to initialize this early on: we ignore its | ||
546 | * attempts to map over the Switcher. */ | ||
314 | if (idx >= SWITCHER_PGD_INDEX) | 547 | if (idx >= SWITCHER_PGD_INDEX) |
315 | return; | 548 | return; |
316 | 549 | ||
550 | /* If they're talking about a page table we have a shadow for... */ | ||
317 | pgdir = find_pgdir(lg, cr3); | 551 | pgdir = find_pgdir(lg, cr3); |
318 | if (pgdir < ARRAY_SIZE(lg->pgdirs)) | 552 | if (pgdir < ARRAY_SIZE(lg->pgdirs)) |
553 | /* ... throw it away. */ | ||
319 | release_pgd(lg, lg->pgdirs[pgdir].pgdir + idx); | 554 | release_pgd(lg, lg->pgdirs[pgdir].pgdir + idx); |
320 | } | 555 | } |
321 | 556 | ||
557 | /*H:500 (vii) Setting up the page tables initially. | ||
558 | * | ||
559 | * When a Guest is first created, the Launcher tells us where the toplevel of | ||
560 | * its first page table is. We set some things up here: */ | ||
322 | int init_guest_pagetable(struct lguest *lg, unsigned long pgtable) | 561 | int init_guest_pagetable(struct lguest *lg, unsigned long pgtable) |
323 | { | 562 | { |
324 | /* We assume this in flush_user_mappings, so check now */ | 563 | /* In flush_user_mappings() we loop from 0 to |
564 | * "vaddr_to_pgd_index(lg->page_offset)". This assumes it won't hit | ||
565 | * the Switcher mappings, so check that now. */ | ||
325 | if (vaddr_to_pgd_index(lg->page_offset) >= SWITCHER_PGD_INDEX) | 566 | if (vaddr_to_pgd_index(lg->page_offset) >= SWITCHER_PGD_INDEX) |
326 | return -EINVAL; | 567 | return -EINVAL; |
568 | /* We start on the first shadow page table, and give it a blank PGD | ||
569 | * page. */ | ||
327 | lg->pgdidx = 0; | 570 | lg->pgdidx = 0; |
328 | lg->pgdirs[lg->pgdidx].cr3 = pgtable; | 571 | lg->pgdirs[lg->pgdidx].cr3 = pgtable; |
329 | lg->pgdirs[lg->pgdidx].pgdir = (spgd_t*)get_zeroed_page(GFP_KERNEL); | 572 | lg->pgdirs[lg->pgdidx].pgdir = (spgd_t*)get_zeroed_page(GFP_KERNEL); |
@@ -332,33 +575,48 @@ int init_guest_pagetable(struct lguest *lg, unsigned long pgtable) | |||
332 | return 0; | 575 | return 0; |
333 | } | 576 | } |
334 | 577 | ||
578 | /* When a Guest dies, our cleanup is fairly simple. */ | ||
335 | void free_guest_pagetable(struct lguest *lg) | 579 | void free_guest_pagetable(struct lguest *lg) |
336 | { | 580 | { |
337 | unsigned int i; | 581 | unsigned int i; |
338 | 582 | ||
583 | /* Throw away all page table pages. */ | ||
339 | release_all_pagetables(lg); | 584 | release_all_pagetables(lg); |
585 | /* Now free the top levels: free_page() can handle 0 just fine. */ | ||
340 | for (i = 0; i < ARRAY_SIZE(lg->pgdirs); i++) | 586 | for (i = 0; i < ARRAY_SIZE(lg->pgdirs); i++) |
341 | free_page((long)lg->pgdirs[i].pgdir); | 587 | free_page((long)lg->pgdirs[i].pgdir); |
342 | } | 588 | } |
343 | 589 | ||
344 | /* Caller must be preempt-safe */ | 590 | /*H:480 (vi) Mapping the Switcher when the Guest is about to run. |
591 | * | ||
592 | * The Switcher and the two pages for this CPU need to be available to the | ||
593 | * Guest (and not the pages for other CPUs). We have the appropriate PTE pages | ||
594 | * for each CPU already set up, we just need to hook them in. */ | ||
345 | void map_switcher_in_guest(struct lguest *lg, struct lguest_pages *pages) | 595 | void map_switcher_in_guest(struct lguest *lg, struct lguest_pages *pages) |
346 | { | 596 | { |
347 | spte_t *switcher_pte_page = __get_cpu_var(switcher_pte_pages); | 597 | spte_t *switcher_pte_page = __get_cpu_var(switcher_pte_pages); |
348 | spgd_t switcher_pgd; | 598 | spgd_t switcher_pgd; |
349 | spte_t regs_pte; | 599 | spte_t regs_pte; |
350 | 600 | ||
351 | /* Since switcher less that 4MB, we simply mug top pte page. */ | 601 | /* Make the last PGD entry for this Guest point to the Switcher's PTE |
602 | * page for this CPU (with appropriate flags). */ | ||
352 | switcher_pgd.pfn = __pa(switcher_pte_page) >> PAGE_SHIFT; | 603 | switcher_pgd.pfn = __pa(switcher_pte_page) >> PAGE_SHIFT; |
353 | switcher_pgd.flags = _PAGE_KERNEL; | 604 | switcher_pgd.flags = _PAGE_KERNEL; |
354 | lg->pgdirs[lg->pgdidx].pgdir[SWITCHER_PGD_INDEX] = switcher_pgd; | 605 | lg->pgdirs[lg->pgdidx].pgdir[SWITCHER_PGD_INDEX] = switcher_pgd; |
355 | 606 | ||
356 | /* Map our regs page over stack page. */ | 607 | /* We also change the Switcher PTE page. When we're running the Guest, |
608 | * we want the Guest's "regs" page to appear where the first Switcher | ||
609 | * page for this CPU is. This is an optimization: when the Switcher | ||
610 | * saves the Guest registers, it saves them into the first page of this | ||
611 | * CPU's "struct lguest_pages": if we make sure the Guest's register | ||
612 | * page is already mapped there, we don't have to copy them out | ||
613 | * again. */ | ||
357 | regs_pte.pfn = __pa(lg->regs_page) >> PAGE_SHIFT; | 614 | regs_pte.pfn = __pa(lg->regs_page) >> PAGE_SHIFT; |
358 | regs_pte.flags = _PAGE_KERNEL; | 615 | regs_pte.flags = _PAGE_KERNEL; |
359 | switcher_pte_page[(unsigned long)pages/PAGE_SIZE%PTES_PER_PAGE] | 616 | switcher_pte_page[(unsigned long)pages/PAGE_SIZE%PTES_PER_PAGE] |
360 | = regs_pte; | 617 | = regs_pte; |
361 | } | 618 | } |
619 | /*:*/ | ||
362 | 620 | ||
363 | static void free_switcher_pte_pages(void) | 621 | static void free_switcher_pte_pages(void) |
364 | { | 622 | { |
@@ -368,6 +626,10 @@ static void free_switcher_pte_pages(void) | |||
368 | free_page((long)switcher_pte_page(i)); | 626 | free_page((long)switcher_pte_page(i)); |
369 | } | 627 | } |
370 | 628 | ||
629 | /*H:520 Setting up the Switcher PTE page for given CPU is fairly easy, given | ||
630 | * the CPU number and the "struct page"s for the Switcher code itself. | ||
631 | * | ||
632 | * Currently the Switcher is less than a page long, so "pages" is always 1. */ | ||
371 | static __init void populate_switcher_pte_page(unsigned int cpu, | 633 | static __init void populate_switcher_pte_page(unsigned int cpu, |
372 | struct page *switcher_page[], | 634 | struct page *switcher_page[], |
373 | unsigned int pages) | 635 | unsigned int pages) |
@@ -375,21 +637,26 @@ static __init void populate_switcher_pte_page(unsigned int cpu, | |||
375 | unsigned int i; | 637 | unsigned int i; |
376 | spte_t *pte = switcher_pte_page(cpu); | 638 | spte_t *pte = switcher_pte_page(cpu); |
377 | 639 | ||
640 | /* The first entries are easy: they map the Switcher code. */ | ||
378 | for (i = 0; i < pages; i++) { | 641 | for (i = 0; i < pages; i++) { |
379 | pte[i].pfn = page_to_pfn(switcher_page[i]); | 642 | pte[i].pfn = page_to_pfn(switcher_page[i]); |
380 | pte[i].flags = _PAGE_PRESENT|_PAGE_ACCESSED; | 643 | pte[i].flags = _PAGE_PRESENT|_PAGE_ACCESSED; |
381 | } | 644 | } |
382 | 645 | ||
383 | /* We only map this CPU's pages, so guest can't see others. */ | 646 | /* The only other thing we map is this CPU's pair of pages. */ |
384 | i = pages + cpu*2; | 647 | i = pages + cpu*2; |
385 | 648 | ||
386 | /* First page (regs) is rw, second (state) is ro. */ | 649 | /* First page (Guest registers) is writable from the Guest */ |
387 | pte[i].pfn = page_to_pfn(switcher_page[i]); | 650 | pte[i].pfn = page_to_pfn(switcher_page[i]); |
388 | pte[i].flags = _PAGE_PRESENT|_PAGE_ACCESSED|_PAGE_RW; | 651 | pte[i].flags = _PAGE_PRESENT|_PAGE_ACCESSED|_PAGE_RW; |
652 | /* The second page contains the "struct lguest_ro_state", and is | ||
653 | * read-only. */ | ||
389 | pte[i+1].pfn = page_to_pfn(switcher_page[i+1]); | 654 | pte[i+1].pfn = page_to_pfn(switcher_page[i+1]); |
390 | pte[i+1].flags = _PAGE_PRESENT|_PAGE_ACCESSED; | 655 | pte[i+1].flags = _PAGE_PRESENT|_PAGE_ACCESSED; |
391 | } | 656 | } |
392 | 657 | ||
658 | /*H:510 At boot or module load time, init_pagetables() allocates and populates | ||
659 | * the Switcher PTE page for each CPU. */ | ||
393 | __init int init_pagetables(struct page **switcher_page, unsigned int pages) | 660 | __init int init_pagetables(struct page **switcher_page, unsigned int pages) |
394 | { | 661 | { |
395 | unsigned int i; | 662 | unsigned int i; |
@@ -404,7 +671,9 @@ __init int init_pagetables(struct page **switcher_page, unsigned int pages) | |||
404 | } | 671 | } |
405 | return 0; | 672 | return 0; |
406 | } | 673 | } |
674 | /*:*/ | ||
407 | 675 | ||
676 | /* Cleaning up simply involves freeing the PTE page for each CPU. */ | ||
408 | void free_pagetables(void) | 677 | void free_pagetables(void) |
409 | { | 678 | { |
410 | free_switcher_pte_pages(); | 679 | free_switcher_pte_pages(); |
diff --git a/drivers/lguest/segments.c b/drivers/lguest/segments.c index 1b2cfe89dcd5..f675a41a80da 100644 --- a/drivers/lguest/segments.c +++ b/drivers/lguest/segments.c | |||
@@ -1,16 +1,68 @@ | |||
1 | /*P:600 The x86 architecture has segments, which involve a table of descriptors | ||
2 | * which can be used to do funky things with virtual address interpretation. | ||
3 | * We originally used to use segments so the Guest couldn't alter the | ||
4 | * Guest<->Host Switcher, and then we had to trim Guest segments, and restore | ||
5 | * for userspace per-thread segments, but trim again for on userspace->kernel | ||
6 | * transitions... This nightmarish creation was contained within this file, | ||
7 | * where we knew not to tread without heavy armament and a change of underwear. | ||
8 | * | ||
9 | * In these modern times, the segment handling code consists of simple sanity | ||
10 | * checks, and the worst you'll experience reading this code is butterfly-rash | ||
11 | * from frolicking through its parklike serenity. :*/ | ||
1 | #include "lg.h" | 12 | #include "lg.h" |
2 | 13 | ||
14 | /*H:600 | ||
15 | * We've almost completed the Host; there's just one file to go! | ||
16 | * | ||
17 | * Segments & The Global Descriptor Table | ||
18 | * | ||
19 | * (That title sounds like a bad Nerdcore group. Not to suggest that there are | ||
20 | * any good Nerdcore groups, but in high school a friend of mine had a band | ||
21 | * called Joe Fish and the Chips, so there are definitely worse band names). | ||
22 | * | ||
23 | * To refresh: the GDT is a table of 8-byte values describing segments. Once | ||
24 | * set up, these segments can be loaded into one of the 6 "segment registers". | ||
25 | * | ||
26 | * GDT entries are passed around as "struct desc_struct"s, which like IDT | ||
27 | * entries are split into two 32-bit members, "a" and "b". One day, someone | ||
28 | * will clean that up, and be declared a Hero. (No pressure, I'm just saying). | ||
29 | * | ||
30 | * Anyway, the GDT entry contains a base (the start address of the segment), a | ||
31 | * limit (the size of the segment - 1), and some flags. Sounds simple, and it | ||
32 | * would be, except those zany Intel engineers decided that it was too boring | ||
33 | * to put the base at one end, the limit at the other, and the flags in | ||
34 | * between. They decided to shotgun the bits at random throughout the 8 bytes, | ||
35 | * like so: | ||
36 | * | ||
37 | * 0 16 40 48 52 56 63 | ||
38 | * [ limit part 1 ][ base part 1 ][ flags ][li][fl][base ] | ||
39 | * mit ags part 2 | ||
40 | * part 2 | ||
41 | * | ||
42 | * As a result, this file contains a certain amount of magic numeracy. Let's | ||
43 | * begin. | ||
44 | */ | ||
45 | |||
46 | /* Is the descriptor the Guest wants us to put in OK? | ||
47 | * | ||
48 | * The flag which Intel says must be zero: must be zero. The descriptor must | ||
49 | * be present, (this is actually checked earlier but is here for thorougness), | ||
50 | * and the descriptor type must be 1 (a memory segment). */ | ||
3 | static int desc_ok(const struct desc_struct *gdt) | 51 | static int desc_ok(const struct desc_struct *gdt) |
4 | { | 52 | { |
5 | /* MBZ=0, P=1, DT=1 */ | ||
6 | return ((gdt->b & 0x00209000) == 0x00009000); | 53 | return ((gdt->b & 0x00209000) == 0x00009000); |
7 | } | 54 | } |
8 | 55 | ||
56 | /* Is the segment present? (Otherwise it can't be used by the Guest). */ | ||
9 | static int segment_present(const struct desc_struct *gdt) | 57 | static int segment_present(const struct desc_struct *gdt) |
10 | { | 58 | { |
11 | return gdt->b & 0x8000; | 59 | return gdt->b & 0x8000; |
12 | } | 60 | } |
13 | 61 | ||
62 | /* There are several entries we don't let the Guest set. The TSS entry is the | ||
63 | * "Task State Segment" which controls all kinds of delicate things. The | ||
64 | * LGUEST_CS and LGUEST_DS entries are reserved for the Switcher, and the | ||
65 | * the Guest can't be trusted to deal with double faults. */ | ||
14 | static int ignored_gdt(unsigned int num) | 66 | static int ignored_gdt(unsigned int num) |
15 | { | 67 | { |
16 | return (num == GDT_ENTRY_TSS | 68 | return (num == GDT_ENTRY_TSS |
@@ -19,9 +71,18 @@ static int ignored_gdt(unsigned int num) | |||
19 | || num == GDT_ENTRY_DOUBLEFAULT_TSS); | 71 | || num == GDT_ENTRY_DOUBLEFAULT_TSS); |
20 | } | 72 | } |
21 | 73 | ||
22 | /* We don't allow removal of CS, DS or SS; it doesn't make sense. */ | 74 | /* If the Guest asks us to remove an entry from the GDT, we have to be careful. |
75 | * If one of the segment registers is pointing at that entry the Switcher will | ||
76 | * crash when it tries to reload the segment registers for the Guest. | ||
77 | * | ||
78 | * It doesn't make much sense for the Guest to try to remove its own code, data | ||
79 | * or stack segments while they're in use: assume that's a Guest bug. If it's | ||
80 | * one of the lesser segment registers using the removed entry, we simply set | ||
81 | * that register to 0 (unusable). */ | ||
23 | static void check_segment_use(struct lguest *lg, unsigned int desc) | 82 | static void check_segment_use(struct lguest *lg, unsigned int desc) |
24 | { | 83 | { |
84 | /* GDT entries are 8 bytes long, so we divide to get the index and | ||
85 | * ignore the bottom bits. */ | ||
25 | if (lg->regs->gs / 8 == desc) | 86 | if (lg->regs->gs / 8 == desc) |
26 | lg->regs->gs = 0; | 87 | lg->regs->gs = 0; |
27 | if (lg->regs->fs / 8 == desc) | 88 | if (lg->regs->fs / 8 == desc) |
@@ -33,13 +94,21 @@ static void check_segment_use(struct lguest *lg, unsigned int desc) | |||
33 | || lg->regs->ss / 8 == desc) | 94 | || lg->regs->ss / 8 == desc) |
34 | kill_guest(lg, "Removed live GDT entry %u", desc); | 95 | kill_guest(lg, "Removed live GDT entry %u", desc); |
35 | } | 96 | } |
36 | 97 | /*:*/ | |
98 | /*M:009 We wouldn't need to check for removal of in-use segments if we handled | ||
99 | * faults in the Switcher. However, it's probably not a worthwhile | ||
100 | * optimization. :*/ | ||
101 | |||
102 | /*H:610 Once the GDT has been changed, we look through the changed entries and | ||
103 | * see if they're OK. If not, we'll call kill_guest() and the Guest will never | ||
104 | * get to use the invalid entries. */ | ||
37 | static void fixup_gdt_table(struct lguest *lg, unsigned start, unsigned end) | 105 | static void fixup_gdt_table(struct lguest *lg, unsigned start, unsigned end) |
38 | { | 106 | { |
39 | unsigned int i; | 107 | unsigned int i; |
40 | 108 | ||
41 | for (i = start; i < end; i++) { | 109 | for (i = start; i < end; i++) { |
42 | /* We never copy these ones to real gdt */ | 110 | /* We never copy these ones to real GDT, so we don't care what |
111 | * they say */ | ||
43 | if (ignored_gdt(i)) | 112 | if (ignored_gdt(i)) |
44 | continue; | 113 | continue; |
45 | 114 | ||
@@ -53,41 +122,57 @@ static void fixup_gdt_table(struct lguest *lg, unsigned start, unsigned end) | |||
53 | if (!desc_ok(&lg->gdt[i])) | 122 | if (!desc_ok(&lg->gdt[i])) |
54 | kill_guest(lg, "Bad GDT descriptor %i", i); | 123 | kill_guest(lg, "Bad GDT descriptor %i", i); |
55 | 124 | ||
56 | /* DPL 0 presumably means "for use by guest". */ | 125 | /* Segment descriptors contain a privilege level: the Guest is |
126 | * sometimes careless and leaves this as 0, even though it's | ||
127 | * running at privilege level 1. If so, we fix it here. */ | ||
57 | if ((lg->gdt[i].b & 0x00006000) == 0) | 128 | if ((lg->gdt[i].b & 0x00006000) == 0) |
58 | lg->gdt[i].b |= (GUEST_PL << 13); | 129 | lg->gdt[i].b |= (GUEST_PL << 13); |
59 | 130 | ||
60 | /* Set accessed bit, since gdt isn't writable. */ | 131 | /* Each descriptor has an "accessed" bit. If we don't set it |
132 | * now, the CPU will try to set it when the Guest first loads | ||
133 | * that entry into a segment register. But the GDT isn't | ||
134 | * writable by the Guest, so bad things can happen. */ | ||
61 | lg->gdt[i].b |= 0x00000100; | 135 | lg->gdt[i].b |= 0x00000100; |
62 | } | 136 | } |
63 | } | 137 | } |
64 | 138 | ||
139 | /* This routine is called at boot or modprobe time for each CPU to set up the | ||
140 | * "constant" GDT entries for Guests running on that CPU. */ | ||
65 | void setup_default_gdt_entries(struct lguest_ro_state *state) | 141 | void setup_default_gdt_entries(struct lguest_ro_state *state) |
66 | { | 142 | { |
67 | struct desc_struct *gdt = state->guest_gdt; | 143 | struct desc_struct *gdt = state->guest_gdt; |
68 | unsigned long tss = (unsigned long)&state->guest_tss; | 144 | unsigned long tss = (unsigned long)&state->guest_tss; |
69 | 145 | ||
70 | /* Hypervisor segments. */ | 146 | /* The hypervisor segments are full 0-4G segments, privilege level 0 */ |
71 | gdt[GDT_ENTRY_LGUEST_CS] = FULL_EXEC_SEGMENT; | 147 | gdt[GDT_ENTRY_LGUEST_CS] = FULL_EXEC_SEGMENT; |
72 | gdt[GDT_ENTRY_LGUEST_DS] = FULL_SEGMENT; | 148 | gdt[GDT_ENTRY_LGUEST_DS] = FULL_SEGMENT; |
73 | 149 | ||
74 | /* This is the one which we *cannot* copy from guest, since tss | 150 | /* The TSS segment refers to the TSS entry for this CPU, so we cannot |
75 | is depended on this lguest_ro_state, ie. this cpu. */ | 151 | * copy it from the Guest. Forgive the magic flags */ |
76 | gdt[GDT_ENTRY_TSS].a = 0x00000067 | (tss << 16); | 152 | gdt[GDT_ENTRY_TSS].a = 0x00000067 | (tss << 16); |
77 | gdt[GDT_ENTRY_TSS].b = 0x00008900 | (tss & 0xFF000000) | 153 | gdt[GDT_ENTRY_TSS].b = 0x00008900 | (tss & 0xFF000000) |
78 | | ((tss >> 16) & 0x000000FF); | 154 | | ((tss >> 16) & 0x000000FF); |
79 | } | 155 | } |
80 | 156 | ||
157 | /* This routine is called before the Guest is run for the first time. */ | ||
81 | void setup_guest_gdt(struct lguest *lg) | 158 | void setup_guest_gdt(struct lguest *lg) |
82 | { | 159 | { |
160 | /* Start with full 0-4G segments... */ | ||
83 | lg->gdt[GDT_ENTRY_KERNEL_CS] = FULL_EXEC_SEGMENT; | 161 | lg->gdt[GDT_ENTRY_KERNEL_CS] = FULL_EXEC_SEGMENT; |
84 | lg->gdt[GDT_ENTRY_KERNEL_DS] = FULL_SEGMENT; | 162 | lg->gdt[GDT_ENTRY_KERNEL_DS] = FULL_SEGMENT; |
163 | /* ...except the Guest is allowed to use them, so set the privilege | ||
164 | * level appropriately in the flags. */ | ||
85 | lg->gdt[GDT_ENTRY_KERNEL_CS].b |= (GUEST_PL << 13); | 165 | lg->gdt[GDT_ENTRY_KERNEL_CS].b |= (GUEST_PL << 13); |
86 | lg->gdt[GDT_ENTRY_KERNEL_DS].b |= (GUEST_PL << 13); | 166 | lg->gdt[GDT_ENTRY_KERNEL_DS].b |= (GUEST_PL << 13); |
87 | } | 167 | } |
88 | 168 | ||
89 | /* This is a fast version for the common case where only the three TLS entries | 169 | /* Like the IDT, we never simply use the GDT the Guest gives us. We set up the |
90 | * have changed. */ | 170 | * GDTs for each CPU, then we copy across the entries each time we want to run |
171 | * a different Guest on that CPU. */ | ||
172 | |||
173 | /* A partial GDT load, for the three "thead-local storage" entries. Otherwise | ||
174 | * it's just like load_guest_gdt(). So much, in fact, it would probably be | ||
175 | * neater to have a single hypercall to cover both. */ | ||
91 | void copy_gdt_tls(const struct lguest *lg, struct desc_struct *gdt) | 176 | void copy_gdt_tls(const struct lguest *lg, struct desc_struct *gdt) |
92 | { | 177 | { |
93 | unsigned int i; | 178 | unsigned int i; |
@@ -96,22 +181,31 @@ void copy_gdt_tls(const struct lguest *lg, struct desc_struct *gdt) | |||
96 | gdt[i] = lg->gdt[i]; | 181 | gdt[i] = lg->gdt[i]; |
97 | } | 182 | } |
98 | 183 | ||
184 | /* This is the full version */ | ||
99 | void copy_gdt(const struct lguest *lg, struct desc_struct *gdt) | 185 | void copy_gdt(const struct lguest *lg, struct desc_struct *gdt) |
100 | { | 186 | { |
101 | unsigned int i; | 187 | unsigned int i; |
102 | 188 | ||
189 | /* The default entries from setup_default_gdt_entries() are not | ||
190 | * replaced. See ignored_gdt() above. */ | ||
103 | for (i = 0; i < GDT_ENTRIES; i++) | 191 | for (i = 0; i < GDT_ENTRIES; i++) |
104 | if (!ignored_gdt(i)) | 192 | if (!ignored_gdt(i)) |
105 | gdt[i] = lg->gdt[i]; | 193 | gdt[i] = lg->gdt[i]; |
106 | } | 194 | } |
107 | 195 | ||
196 | /* This is where the Guest asks us to load a new GDT (LHCALL_LOAD_GDT). */ | ||
108 | void load_guest_gdt(struct lguest *lg, unsigned long table, u32 num) | 197 | void load_guest_gdt(struct lguest *lg, unsigned long table, u32 num) |
109 | { | 198 | { |
199 | /* We assume the Guest has the same number of GDT entries as the | ||
200 | * Host, otherwise we'd have to dynamically allocate the Guest GDT. */ | ||
110 | if (num > ARRAY_SIZE(lg->gdt)) | 201 | if (num > ARRAY_SIZE(lg->gdt)) |
111 | kill_guest(lg, "too many gdt entries %i", num); | 202 | kill_guest(lg, "too many gdt entries %i", num); |
112 | 203 | ||
204 | /* We read the whole thing in, then fix it up. */ | ||
113 | lgread(lg, lg->gdt, table, num * sizeof(lg->gdt[0])); | 205 | lgread(lg, lg->gdt, table, num * sizeof(lg->gdt[0])); |
114 | fixup_gdt_table(lg, 0, ARRAY_SIZE(lg->gdt)); | 206 | fixup_gdt_table(lg, 0, ARRAY_SIZE(lg->gdt)); |
207 | /* Mark that the GDT changed so the core knows it has to copy it again, | ||
208 | * even if the Guest is run on the same CPU. */ | ||
115 | lg->changed |= CHANGED_GDT; | 209 | lg->changed |= CHANGED_GDT; |
116 | } | 210 | } |
117 | 211 | ||
@@ -123,3 +217,13 @@ void guest_load_tls(struct lguest *lg, unsigned long gtls) | |||
123 | fixup_gdt_table(lg, GDT_ENTRY_TLS_MIN, GDT_ENTRY_TLS_MAX+1); | 217 | fixup_gdt_table(lg, GDT_ENTRY_TLS_MIN, GDT_ENTRY_TLS_MAX+1); |
124 | lg->changed |= CHANGED_GDT_TLS; | 218 | lg->changed |= CHANGED_GDT_TLS; |
125 | } | 219 | } |
220 | |||
221 | /* | ||
222 | * With this, we have finished the Host. | ||
223 | * | ||
224 | * Five of the seven parts of our task are complete. You have made it through | ||
225 | * the Bit of Despair (I think that's somewhere in the page table code, | ||
226 | * myself). | ||
227 | * | ||
228 | * Next, we examine "make Switcher". It's short, but intense. | ||
229 | */ | ||
diff --git a/drivers/lguest/switcher.S b/drivers/lguest/switcher.S index eadd4cc299d2..d418179ea6b5 100644 --- a/drivers/lguest/switcher.S +++ b/drivers/lguest/switcher.S | |||
@@ -1,45 +1,136 @@ | |||
1 | /* This code sits at 0xFFC00000 to do the low-level guest<->host switch. | 1 | /*P:900 This is the Switcher: code which sits at 0xFFC00000 to do the low-level |
2 | * Guest<->Host switch. It is as simple as it can be made, but it's naturally | ||
3 | * very specific to x86. | ||
4 | * | ||
5 | * You have now completed Preparation. If this has whet your appetite; if you | ||
6 | * are feeling invigorated and refreshed then the next, more challenging stage | ||
7 | * can be found in "make Guest". :*/ | ||
2 | 8 | ||
3 | There is are two pages above us for this CPU (struct lguest_pages). | 9 | /*S:100 |
4 | The second page (struct lguest_ro_state) becomes read-only after the | 10 | * Welcome to the Switcher itself! |
5 | context switch. The first page (the stack for traps) remains writable, | 11 | * |
6 | but while we're in here, the guest cannot be running. | 12 | * This file contains the low-level code which changes the CPU to run the Guest |
7 | */ | 13 | * code, and returns to the Host when something happens. Understand this, and |
14 | * you understand the heart of our journey. | ||
15 | * | ||
16 | * Because this is in assembler rather than C, our tale switches from prose to | ||
17 | * verse. First I tried limericks: | ||
18 | * | ||
19 | * There once was an eax reg, | ||
20 | * To which our pointer was fed, | ||
21 | * It needed an add, | ||
22 | * Which asm-offsets.h had | ||
23 | * But this limerick is hurting my head. | ||
24 | * | ||
25 | * Next I tried haikus, but fitting the required reference to the seasons in | ||
26 | * every stanza was quickly becoming tiresome: | ||
27 | * | ||
28 | * The %eax reg | ||
29 | * Holds "struct lguest_pages" now: | ||
30 | * Cherry blossoms fall. | ||
31 | * | ||
32 | * Then I started with Heroic Verse, but the rhyming requirement leeched away | ||
33 | * the content density and led to some uniquely awful oblique rhymes: | ||
34 | * | ||
35 | * These constants are coming from struct offsets | ||
36 | * For use within the asm switcher text. | ||
37 | * | ||
38 | * Finally, I settled for something between heroic hexameter, and normal prose | ||
39 | * with inappropriate linebreaks. Anyway, it aint no Shakespeare. | ||
40 | */ | ||
41 | |||
42 | // Not all kernel headers work from assembler | ||
43 | // But these ones are needed: the ENTRY() define | ||
44 | // And constants extracted from struct offsets | ||
45 | // To avoid magic numbers and breakage: | ||
46 | // Should they change the compiler can't save us | ||
47 | // Down here in the depths of assembler code. | ||
8 | #include <linux/linkage.h> | 48 | #include <linux/linkage.h> |
9 | #include <asm/asm-offsets.h> | 49 | #include <asm/asm-offsets.h> |
10 | #include "lg.h" | 50 | #include "lg.h" |
11 | 51 | ||
52 | // We mark the start of the code to copy | ||
53 | // It's placed in .text tho it's never run here | ||
54 | // You'll see the trick macro at the end | ||
55 | // Which interleaves data and text to effect. | ||
12 | .text | 56 | .text |
13 | ENTRY(start_switcher_text) | 57 | ENTRY(start_switcher_text) |
14 | 58 | ||
15 | /* %eax points to lguest pages for this CPU. %ebx contains cr3 value. | 59 | // When we reach switch_to_guest we have just left |
16 | All normal registers can be clobbered! */ | 60 | // The safe and comforting shores of C code |
61 | // %eax has the "struct lguest_pages" to use | ||
62 | // Where we save state and still see it from the Guest | ||
63 | // And %ebx holds the Guest shadow pagetable: | ||
64 | // Once set we have truly left Host behind. | ||
17 | ENTRY(switch_to_guest) | 65 | ENTRY(switch_to_guest) |
18 | /* Save host segments on host stack. */ | 66 | // We told gcc all its regs could fade, |
67 | // Clobbered by our journey into the Guest | ||
68 | // We could have saved them, if we tried | ||
69 | // But time is our master and cycles count. | ||
70 | |||
71 | // Segment registers must be saved for the Host | ||
72 | // We push them on the Host stack for later | ||
19 | pushl %es | 73 | pushl %es |
20 | pushl %ds | 74 | pushl %ds |
21 | pushl %gs | 75 | pushl %gs |
22 | pushl %fs | 76 | pushl %fs |
23 | /* With CONFIG_FRAME_POINTER, gcc doesn't let us clobber this! */ | 77 | // But the compiler is fickle, and heeds |
78 | // No warning of %ebp clobbers | ||
79 | // When frame pointers are used. That register | ||
80 | // Must be saved and restored or chaos strikes. | ||
24 | pushl %ebp | 81 | pushl %ebp |
25 | /* Save host stack. */ | 82 | // The Host's stack is done, now save it away |
83 | // In our "struct lguest_pages" at offset | ||
84 | // Distilled into asm-offsets.h | ||
26 | movl %esp, LGUEST_PAGES_host_sp(%eax) | 85 | movl %esp, LGUEST_PAGES_host_sp(%eax) |
27 | /* Switch to guest stack: if we get NMI we expect to be there. */ | 86 | |
87 | // All saved and there's now five steps before us: | ||
88 | // Stack, GDT, IDT, TSS | ||
89 | // And last of all the page tables are flipped. | ||
90 | |||
91 | // Yet beware that our stack pointer must be | ||
92 | // Always valid lest an NMI hits | ||
93 | // %edx does the duty here as we juggle | ||
94 | // %eax is lguest_pages: our stack lies within. | ||
28 | movl %eax, %edx | 95 | movl %eax, %edx |
29 | addl $LGUEST_PAGES_regs, %edx | 96 | addl $LGUEST_PAGES_regs, %edx |
30 | movl %edx, %esp | 97 | movl %edx, %esp |
31 | /* Switch to guest's GDT, IDT. */ | 98 | |
99 | // The Guest's GDT we so carefully | ||
100 | // Placed in the "struct lguest_pages" before | ||
32 | lgdt LGUEST_PAGES_guest_gdt_desc(%eax) | 101 | lgdt LGUEST_PAGES_guest_gdt_desc(%eax) |
102 | |||
103 | // The Guest's IDT we did partially | ||
104 | // Move to the "struct lguest_pages" as well. | ||
33 | lidt LGUEST_PAGES_guest_idt_desc(%eax) | 105 | lidt LGUEST_PAGES_guest_idt_desc(%eax) |
34 | /* Switch to guest's TSS while GDT still writable. */ | 106 | |
107 | // The TSS entry which controls traps | ||
108 | // Must be loaded up with "ltr" now: | ||
109 | // For after we switch over our page tables | ||
110 | // It (as the rest) will be writable no more. | ||
111 | // (The GDT entry TSS needs | ||
112 | // Changes type when we load it: damn Intel!) | ||
35 | movl $(GDT_ENTRY_TSS*8), %edx | 113 | movl $(GDT_ENTRY_TSS*8), %edx |
36 | ltr %dx | 114 | ltr %dx |
37 | /* Set host's TSS GDT entry to available (clear byte 5 bit 2). */ | 115 | |
116 | // Look back now, before we take this last step! | ||
117 | // The Host's TSS entry was also marked used; | ||
118 | // Let's clear it again, ere we return. | ||
119 | // The GDT descriptor of the Host | ||
120 | // Points to the table after two "size" bytes | ||
38 | movl (LGUEST_PAGES_host_gdt_desc+2)(%eax), %edx | 121 | movl (LGUEST_PAGES_host_gdt_desc+2)(%eax), %edx |
122 | // Clear the type field of "used" (byte 5, bit 2) | ||
39 | andb $0xFD, (GDT_ENTRY_TSS*8 + 5)(%edx) | 123 | andb $0xFD, (GDT_ENTRY_TSS*8 + 5)(%edx) |
40 | /* Switch to guest page tables: lguest_pages->state now read-only. */ | 124 | |
125 | // Once our page table's switched, the Guest is live! | ||
126 | // The Host fades as we run this final step. | ||
127 | // Our "struct lguest_pages" is now read-only. | ||
41 | movl %ebx, %cr3 | 128 | movl %ebx, %cr3 |
42 | /* Restore guest regs */ | 129 | |
130 | // The page table change did one tricky thing: | ||
131 | // The Guest's register page has been mapped | ||
132 | // Writable onto our %esp (stack) -- | ||
133 | // We can simply pop off all Guest regs. | ||
43 | popl %ebx | 134 | popl %ebx |
44 | popl %ecx | 135 | popl %ecx |
45 | popl %edx | 136 | popl %edx |
@@ -51,12 +142,27 @@ ENTRY(switch_to_guest) | |||
51 | popl %fs | 142 | popl %fs |
52 | popl %ds | 143 | popl %ds |
53 | popl %es | 144 | popl %es |
54 | /* Skip error code and trap number */ | 145 | |
146 | // Near the base of the stack lurk two strange fields | ||
147 | // Which we fill as we exit the Guest | ||
148 | // These are the trap number and its error | ||
149 | // We can simply step past them on our way. | ||
55 | addl $8, %esp | 150 | addl $8, %esp |
151 | |||
152 | // The last five stack slots hold return address | ||
153 | // And everything needed to change privilege | ||
154 | // Into the Guest privilege level of 1, | ||
155 | // And the stack where the Guest had last left it. | ||
156 | // Interrupts are turned back on: we are Guest. | ||
56 | iret | 157 | iret |
57 | 158 | ||
159 | // There are two paths where we switch to the Host | ||
160 | // So we put the routine in a macro. | ||
161 | // We are on our way home, back to the Host | ||
162 | // Interrupted out of the Guest, we come here. | ||
58 | #define SWITCH_TO_HOST \ | 163 | #define SWITCH_TO_HOST \ |
59 | /* Save guest state */ \ | 164 | /* We save the Guest state: all registers first \ |
165 | * Laid out just as "struct lguest_regs" defines */ \ | ||
60 | pushl %es; \ | 166 | pushl %es; \ |
61 | pushl %ds; \ | 167 | pushl %ds; \ |
62 | pushl %fs; \ | 168 | pushl %fs; \ |
@@ -68,58 +174,119 @@ ENTRY(switch_to_guest) | |||
68 | pushl %edx; \ | 174 | pushl %edx; \ |
69 | pushl %ecx; \ | 175 | pushl %ecx; \ |
70 | pushl %ebx; \ | 176 | pushl %ebx; \ |
71 | /* Load lguest ds segment for convenience. */ \ | 177 | /* Our stack and our code are using segments \ |
178 | * Set in the TSS and IDT \ | ||
179 | * Yet if we were to touch data we'd use \ | ||
180 | * Whatever data segment the Guest had. \ | ||
181 | * Load the lguest ds segment for now. */ \ | ||
72 | movl $(LGUEST_DS), %eax; \ | 182 | movl $(LGUEST_DS), %eax; \ |
73 | movl %eax, %ds; \ | 183 | movl %eax, %ds; \ |
74 | /* Figure out where we are, based on stack (at top of regs). */ \ | 184 | /* So where are we? Which CPU, which struct? \ |
185 | * The stack is our clue: our TSS sets \ | ||
186 | * It at the end of "struct lguest_pages" \ | ||
187 | * And we then pushed and pushed and pushed Guest regs: \ | ||
188 | * Now stack points atop the "struct lguest_regs". \ | ||
189 | * Subtract that offset, and we find our struct. */ \ | ||
75 | movl %esp, %eax; \ | 190 | movl %esp, %eax; \ |
76 | subl $LGUEST_PAGES_regs, %eax; \ | 191 | subl $LGUEST_PAGES_regs, %eax; \ |
77 | /* Put trap number in %ebx before we switch cr3 and lose it. */ \ | 192 | /* Save our trap number: the switch will obscure it \ |
193 | * (The Guest regs are not mapped here in the Host) \ | ||
194 | * %ebx holds it safe for deliver_to_host */ \ | ||
78 | movl LGUEST_PAGES_regs_trapnum(%eax), %ebx; \ | 195 | movl LGUEST_PAGES_regs_trapnum(%eax), %ebx; \ |
79 | /* Switch to host page tables (host GDT, IDT and stack are in host \ | 196 | /* The Host GDT, IDT and stack! \ |
80 | mem, so need this first) */ \ | 197 | * All these lie safely hidden from the Guest: \ |
198 | * We must return to the Host page tables \ | ||
199 | * (Hence that was saved in struct lguest_pages) */ \ | ||
81 | movl LGUEST_PAGES_host_cr3(%eax), %edx; \ | 200 | movl LGUEST_PAGES_host_cr3(%eax), %edx; \ |
82 | movl %edx, %cr3; \ | 201 | movl %edx, %cr3; \ |
83 | /* Set guest's TSS to available (clear byte 5 bit 2). */ \ | 202 | /* As before, when we looked back at the Host \ |
203 | * As we left and marked TSS unused \ | ||
204 | * So must we now for the Guest left behind. */ \ | ||
84 | andb $0xFD, (LGUEST_PAGES_guest_gdt+GDT_ENTRY_TSS*8+5)(%eax); \ | 205 | andb $0xFD, (LGUEST_PAGES_guest_gdt+GDT_ENTRY_TSS*8+5)(%eax); \ |
85 | /* Switch to host's GDT & IDT. */ \ | 206 | /* Switch to Host's GDT, IDT. */ \ |
86 | lgdt LGUEST_PAGES_host_gdt_desc(%eax); \ | 207 | lgdt LGUEST_PAGES_host_gdt_desc(%eax); \ |
87 | lidt LGUEST_PAGES_host_idt_desc(%eax); \ | 208 | lidt LGUEST_PAGES_host_idt_desc(%eax); \ |
88 | /* Switch to host's stack. */ \ | 209 | /* Restore the Host's stack where it's saved regs lie */ \ |
89 | movl LGUEST_PAGES_host_sp(%eax), %esp; \ | 210 | movl LGUEST_PAGES_host_sp(%eax), %esp; \ |
90 | /* Switch to host's TSS */ \ | 211 | /* Last the TSS: our Host is complete */ \ |
91 | movl $(GDT_ENTRY_TSS*8), %edx; \ | 212 | movl $(GDT_ENTRY_TSS*8), %edx; \ |
92 | ltr %dx; \ | 213 | ltr %dx; \ |
214 | /* Restore now the regs saved right at the first. */ \ | ||
93 | popl %ebp; \ | 215 | popl %ebp; \ |
94 | popl %fs; \ | 216 | popl %fs; \ |
95 | popl %gs; \ | 217 | popl %gs; \ |
96 | popl %ds; \ | 218 | popl %ds; \ |
97 | popl %es | 219 | popl %es |
98 | 220 | ||
99 | /* Return to run_guest_once. */ | 221 | // Here's where we come when the Guest has just trapped: |
222 | // (Which trap we'll see has been pushed on the stack). | ||
223 | // We need only switch back, and the Host will decode | ||
224 | // Why we came home, and what needs to be done. | ||
100 | return_to_host: | 225 | return_to_host: |
101 | SWITCH_TO_HOST | 226 | SWITCH_TO_HOST |
102 | iret | 227 | iret |
103 | 228 | ||
229 | // An interrupt, with some cause external | ||
230 | // Has ajerked us rudely from the Guest's code | ||
231 | // Again we must return home to the Host | ||
104 | deliver_to_host: | 232 | deliver_to_host: |
105 | SWITCH_TO_HOST | 233 | SWITCH_TO_HOST |
106 | /* Decode IDT and jump to hosts' irq handler. When that does iret, it | 234 | // But now we must go home via that place |
107 | * will return to run_guest_once. This is a feature. */ | 235 | // Where that interrupt was supposed to go |
236 | // Had we not been ensconced, running the Guest. | ||
237 | // Here we see the cleverness of our stack: | ||
238 | // The Host stack is formed like an interrupt | ||
239 | // With EIP, CS and EFLAGS layered. | ||
240 | // Interrupt handlers end with "iret" | ||
241 | // And that will take us home at long long last. | ||
242 | |||
243 | // But first we must find the handler to call! | ||
244 | // The IDT descriptor for the Host | ||
245 | // Has two bytes for size, and four for address: | ||
246 | // %edx will hold it for us for now. | ||
108 | movl (LGUEST_PAGES_host_idt_desc+2)(%eax), %edx | 247 | movl (LGUEST_PAGES_host_idt_desc+2)(%eax), %edx |
248 | // We now know the table address we need, | ||
249 | // And saved the trap's number inside %ebx. | ||
250 | // Yet the pointer to the handler is smeared | ||
251 | // Across the bits of the table entry. | ||
252 | // What oracle can tell us how to extract | ||
253 | // From such a convoluted encoding? | ||
254 | // I consulted gcc, and it gave | ||
255 | // These instructions, which I gladly credit: | ||
109 | leal (%edx,%ebx,8), %eax | 256 | leal (%edx,%ebx,8), %eax |
110 | movzwl (%eax),%edx | 257 | movzwl (%eax),%edx |
111 | movl 4(%eax), %eax | 258 | movl 4(%eax), %eax |
112 | xorw %ax, %ax | 259 | xorw %ax, %ax |
113 | orl %eax, %edx | 260 | orl %eax, %edx |
261 | // Now the address of the handler's in %edx | ||
262 | // We call it now: its "iret" takes us home. | ||
114 | jmp *%edx | 263 | jmp *%edx |
115 | 264 | ||
116 | /* Real hardware interrupts are delivered straight to the host. Others | 265 | // Every interrupt can come to us here |
117 | cause us to return to run_guest_once so it can decide what to do. Note | 266 | // But we must truly tell each apart. |
118 | that some of these are overridden by the guest to deliver directly, and | 267 | // They number two hundred and fifty six |
119 | never enter here (see load_guest_idt_entry). */ | 268 | // And each must land in a different spot, |
269 | // Push its number on stack, and join the stream. | ||
270 | |||
271 | // And worse, a mere six of the traps stand apart | ||
272 | // And push on their stack an addition: | ||
273 | // An error number, thirty two bits long | ||
274 | // So we punish the other two fifty | ||
275 | // And make them push a zero so they match. | ||
276 | |||
277 | // Yet two fifty six entries is long | ||
278 | // And all will look most the same as the last | ||
279 | // So we create a macro which can make | ||
280 | // As many entries as we need to fill. | ||
281 | |||
282 | // Note the change to .data then .text: | ||
283 | // We plant the address of each entry | ||
284 | // Into a (data) table for the Host | ||
285 | // To know where each Guest interrupt should go. | ||
120 | .macro IRQ_STUB N TARGET | 286 | .macro IRQ_STUB N TARGET |
121 | .data; .long 1f; .text; 1: | 287 | .data; .long 1f; .text; 1: |
122 | /* Make an error number for most traps, which don't have one. */ | 288 | // Trap eight, ten through fourteen and seventeen |
289 | // Supply an error number. Else zero. | ||
123 | .if (\N <> 8) && (\N < 10 || \N > 14) && (\N <> 17) | 290 | .if (\N <> 8) && (\N < 10 || \N > 14) && (\N <> 17) |
124 | pushl $0 | 291 | pushl $0 |
125 | .endif | 292 | .endif |
@@ -128,6 +295,8 @@ deliver_to_host: | |||
128 | ALIGN | 295 | ALIGN |
129 | .endm | 296 | .endm |
130 | 297 | ||
298 | // This macro creates numerous entries | ||
299 | // Using GAS macros which out-power C's. | ||
131 | .macro IRQ_STUBS FIRST LAST TARGET | 300 | .macro IRQ_STUBS FIRST LAST TARGET |
132 | irq=\FIRST | 301 | irq=\FIRST |
133 | .rept \LAST-\FIRST+1 | 302 | .rept \LAST-\FIRST+1 |
@@ -136,24 +305,43 @@ deliver_to_host: | |||
136 | .endr | 305 | .endr |
137 | .endm | 306 | .endm |
138 | 307 | ||
139 | /* We intercept every interrupt, because we may need to switch back to | 308 | // Here's the marker for our pointer table |
140 | * host. Unfortunately we can't tell them apart except by entry | 309 | // Laid in the data section just before |
141 | * point, so we need 256 entry points. | 310 | // Each macro places the address of code |
142 | */ | 311 | // Forming an array: each one points to text |
312 | // Which handles interrupt in its turn. | ||
143 | .data | 313 | .data |
144 | .global default_idt_entries | 314 | .global default_idt_entries |
145 | default_idt_entries: | 315 | default_idt_entries: |
146 | .text | 316 | .text |
147 | IRQ_STUBS 0 1 return_to_host /* First two traps */ | 317 | // The first two traps go straight back to the Host |
148 | IRQ_STUB 2 handle_nmi /* NMI */ | 318 | IRQ_STUBS 0 1 return_to_host |
149 | IRQ_STUBS 3 31 return_to_host /* Rest of traps */ | 319 | // We'll say nothing, yet, about NMI |
150 | IRQ_STUBS 32 127 deliver_to_host /* Real interrupts */ | 320 | IRQ_STUB 2 handle_nmi |
151 | IRQ_STUB 128 return_to_host /* System call (overridden) */ | 321 | // Other traps also return to the Host |
152 | IRQ_STUBS 129 255 deliver_to_host /* Other real interrupts */ | 322 | IRQ_STUBS 3 31 return_to_host |
153 | 323 | // All interrupts go via their handlers | |
154 | /* We ignore NMI and return. */ | 324 | IRQ_STUBS 32 127 deliver_to_host |
325 | // 'Cept system calls coming from userspace | ||
326 | // Are to go to the Guest, never the Host. | ||
327 | IRQ_STUB 128 return_to_host | ||
328 | IRQ_STUBS 129 255 deliver_to_host | ||
329 | |||
330 | // The NMI, what a fabulous beast | ||
331 | // Which swoops in and stops us no matter that | ||
332 | // We're suspended between heaven and hell, | ||
333 | // (Or more likely between the Host and Guest) | ||
334 | // When in it comes! We are dazed and confused | ||
335 | // So we do the simplest thing which one can. | ||
336 | // Though we've pushed the trap number and zero | ||
337 | // We discard them, return, and hope we live. | ||
155 | handle_nmi: | 338 | handle_nmi: |
156 | addl $8, %esp | 339 | addl $8, %esp |
157 | iret | 340 | iret |
158 | 341 | ||
342 | // We are done; all that's left is Mastery | ||
343 | // And "make Mastery" is a journey long | ||
344 | // Designed to make your fingers itch to code. | ||
345 | |||
346 | // Here ends the text, the file and poem. | ||
159 | ENTRY(end_switcher_text) | 347 | ENTRY(end_switcher_text) |
diff --git a/drivers/md/dm-table.c b/drivers/md/dm-table.c index 2fc199b0016b..2bcde5798b5a 100644 --- a/drivers/md/dm-table.c +++ b/drivers/md/dm-table.c | |||
@@ -526,7 +526,7 @@ static int __table_get_device(struct dm_table *t, struct dm_target *ti, | |||
526 | 526 | ||
527 | void dm_set_device_limits(struct dm_target *ti, struct block_device *bdev) | 527 | void dm_set_device_limits(struct dm_target *ti, struct block_device *bdev) |
528 | { | 528 | { |
529 | request_queue_t *q = bdev_get_queue(bdev); | 529 | struct request_queue *q = bdev_get_queue(bdev); |
530 | struct io_restrictions *rs = &ti->limits; | 530 | struct io_restrictions *rs = &ti->limits; |
531 | 531 | ||
532 | /* | 532 | /* |
@@ -979,7 +979,7 @@ int dm_table_any_congested(struct dm_table *t, int bdi_bits) | |||
979 | devices = dm_table_get_devices(t); | 979 | devices = dm_table_get_devices(t); |
980 | for (d = devices->next; d != devices; d = d->next) { | 980 | for (d = devices->next; d != devices; d = d->next) { |
981 | struct dm_dev *dd = list_entry(d, struct dm_dev, list); | 981 | struct dm_dev *dd = list_entry(d, struct dm_dev, list); |
982 | request_queue_t *q = bdev_get_queue(dd->bdev); | 982 | struct request_queue *q = bdev_get_queue(dd->bdev); |
983 | r |= bdi_congested(&q->backing_dev_info, bdi_bits); | 983 | r |= bdi_congested(&q->backing_dev_info, bdi_bits); |
984 | } | 984 | } |
985 | 985 | ||
@@ -992,7 +992,7 @@ void dm_table_unplug_all(struct dm_table *t) | |||
992 | 992 | ||
993 | for (d = devices->next; d != devices; d = d->next) { | 993 | for (d = devices->next; d != devices; d = d->next) { |
994 | struct dm_dev *dd = list_entry(d, struct dm_dev, list); | 994 | struct dm_dev *dd = list_entry(d, struct dm_dev, list); |
995 | request_queue_t *q = bdev_get_queue(dd->bdev); | 995 | struct request_queue *q = bdev_get_queue(dd->bdev); |
996 | 996 | ||
997 | if (q->unplug_fn) | 997 | if (q->unplug_fn) |
998 | q->unplug_fn(q); | 998 | q->unplug_fn(q); |
@@ -1011,7 +1011,7 @@ int dm_table_flush_all(struct dm_table *t) | |||
1011 | 1011 | ||
1012 | for (d = devices->next; d != devices; d = d->next) { | 1012 | for (d = devices->next; d != devices; d = d->next) { |
1013 | struct dm_dev *dd = list_entry(d, struct dm_dev, list); | 1013 | struct dm_dev *dd = list_entry(d, struct dm_dev, list); |
1014 | request_queue_t *q = bdev_get_queue(dd->bdev); | 1014 | struct request_queue *q = bdev_get_queue(dd->bdev); |
1015 | int err; | 1015 | int err; |
1016 | 1016 | ||
1017 | if (!q->issue_flush_fn) | 1017 | if (!q->issue_flush_fn) |
diff --git a/drivers/md/dm.c b/drivers/md/dm.c index 846614e676c6..141ff9fa296e 100644 --- a/drivers/md/dm.c +++ b/drivers/md/dm.c | |||
@@ -80,7 +80,7 @@ struct mapped_device { | |||
80 | 80 | ||
81 | unsigned long flags; | 81 | unsigned long flags; |
82 | 82 | ||
83 | request_queue_t *queue; | 83 | struct request_queue *queue; |
84 | struct gendisk *disk; | 84 | struct gendisk *disk; |
85 | char name[16]; | 85 | char name[16]; |
86 | 86 | ||
@@ -792,7 +792,7 @@ static void __split_bio(struct mapped_device *md, struct bio *bio) | |||
792 | * The request function that just remaps the bio built up by | 792 | * The request function that just remaps the bio built up by |
793 | * dm_merge_bvec. | 793 | * dm_merge_bvec. |
794 | */ | 794 | */ |
795 | static int dm_request(request_queue_t *q, struct bio *bio) | 795 | static int dm_request(struct request_queue *q, struct bio *bio) |
796 | { | 796 | { |
797 | int r; | 797 | int r; |
798 | int rw = bio_data_dir(bio); | 798 | int rw = bio_data_dir(bio); |
@@ -844,7 +844,7 @@ static int dm_request(request_queue_t *q, struct bio *bio) | |||
844 | return 0; | 844 | return 0; |
845 | } | 845 | } |
846 | 846 | ||
847 | static int dm_flush_all(request_queue_t *q, struct gendisk *disk, | 847 | static int dm_flush_all(struct request_queue *q, struct gendisk *disk, |
848 | sector_t *error_sector) | 848 | sector_t *error_sector) |
849 | { | 849 | { |
850 | struct mapped_device *md = q->queuedata; | 850 | struct mapped_device *md = q->queuedata; |
@@ -859,7 +859,7 @@ static int dm_flush_all(request_queue_t *q, struct gendisk *disk, | |||
859 | return ret; | 859 | return ret; |
860 | } | 860 | } |
861 | 861 | ||
862 | static void dm_unplug_all(request_queue_t *q) | 862 | static void dm_unplug_all(struct request_queue *q) |
863 | { | 863 | { |
864 | struct mapped_device *md = q->queuedata; | 864 | struct mapped_device *md = q->queuedata; |
865 | struct dm_table *map = dm_get_table(md); | 865 | struct dm_table *map = dm_get_table(md); |
@@ -1110,7 +1110,7 @@ static void __set_size(struct mapped_device *md, sector_t size) | |||
1110 | 1110 | ||
1111 | static int __bind(struct mapped_device *md, struct dm_table *t) | 1111 | static int __bind(struct mapped_device *md, struct dm_table *t) |
1112 | { | 1112 | { |
1113 | request_queue_t *q = md->queue; | 1113 | struct request_queue *q = md->queue; |
1114 | sector_t size; | 1114 | sector_t size; |
1115 | 1115 | ||
1116 | size = dm_table_get_size(t); | 1116 | size = dm_table_get_size(t); |
diff --git a/drivers/md/faulty.c b/drivers/md/faulty.c index 4ebd0f2a75ec..cb059cf14c2e 100644 --- a/drivers/md/faulty.c +++ b/drivers/md/faulty.c | |||
@@ -167,7 +167,7 @@ static void add_sector(conf_t *conf, sector_t start, int mode) | |||
167 | conf->nfaults = n+1; | 167 | conf->nfaults = n+1; |
168 | } | 168 | } |
169 | 169 | ||
170 | static int make_request(request_queue_t *q, struct bio *bio) | 170 | static int make_request(struct request_queue *q, struct bio *bio) |
171 | { | 171 | { |
172 | mddev_t *mddev = q->queuedata; | 172 | mddev_t *mddev = q->queuedata; |
173 | conf_t *conf = (conf_t*)mddev->private; | 173 | conf_t *conf = (conf_t*)mddev->private; |
diff --git a/drivers/md/linear.c b/drivers/md/linear.c index 192741083196..17f795c3e0ab 100644 --- a/drivers/md/linear.c +++ b/drivers/md/linear.c | |||
@@ -55,7 +55,7 @@ static inline dev_info_t *which_dev(mddev_t *mddev, sector_t sector) | |||
55 | * | 55 | * |
56 | * Return amount of bytes we can take at this offset | 56 | * Return amount of bytes we can take at this offset |
57 | */ | 57 | */ |
58 | static int linear_mergeable_bvec(request_queue_t *q, struct bio *bio, struct bio_vec *biovec) | 58 | static int linear_mergeable_bvec(struct request_queue *q, struct bio *bio, struct bio_vec *biovec) |
59 | { | 59 | { |
60 | mddev_t *mddev = q->queuedata; | 60 | mddev_t *mddev = q->queuedata; |
61 | dev_info_t *dev0; | 61 | dev_info_t *dev0; |
@@ -79,20 +79,20 @@ static int linear_mergeable_bvec(request_queue_t *q, struct bio *bio, struct bio | |||
79 | return maxsectors << 9; | 79 | return maxsectors << 9; |
80 | } | 80 | } |
81 | 81 | ||
82 | static void linear_unplug(request_queue_t *q) | 82 | static void linear_unplug(struct request_queue *q) |
83 | { | 83 | { |
84 | mddev_t *mddev = q->queuedata; | 84 | mddev_t *mddev = q->queuedata; |
85 | linear_conf_t *conf = mddev_to_conf(mddev); | 85 | linear_conf_t *conf = mddev_to_conf(mddev); |
86 | int i; | 86 | int i; |
87 | 87 | ||
88 | for (i=0; i < mddev->raid_disks; i++) { | 88 | for (i=0; i < mddev->raid_disks; i++) { |
89 | request_queue_t *r_queue = bdev_get_queue(conf->disks[i].rdev->bdev); | 89 | struct request_queue *r_queue = bdev_get_queue(conf->disks[i].rdev->bdev); |
90 | if (r_queue->unplug_fn) | 90 | if (r_queue->unplug_fn) |
91 | r_queue->unplug_fn(r_queue); | 91 | r_queue->unplug_fn(r_queue); |
92 | } | 92 | } |
93 | } | 93 | } |
94 | 94 | ||
95 | static int linear_issue_flush(request_queue_t *q, struct gendisk *disk, | 95 | static int linear_issue_flush(struct request_queue *q, struct gendisk *disk, |
96 | sector_t *error_sector) | 96 | sector_t *error_sector) |
97 | { | 97 | { |
98 | mddev_t *mddev = q->queuedata; | 98 | mddev_t *mddev = q->queuedata; |
@@ -101,7 +101,7 @@ static int linear_issue_flush(request_queue_t *q, struct gendisk *disk, | |||
101 | 101 | ||
102 | for (i=0; i < mddev->raid_disks && ret == 0; i++) { | 102 | for (i=0; i < mddev->raid_disks && ret == 0; i++) { |
103 | struct block_device *bdev = conf->disks[i].rdev->bdev; | 103 | struct block_device *bdev = conf->disks[i].rdev->bdev; |
104 | request_queue_t *r_queue = bdev_get_queue(bdev); | 104 | struct request_queue *r_queue = bdev_get_queue(bdev); |
105 | 105 | ||
106 | if (!r_queue->issue_flush_fn) | 106 | if (!r_queue->issue_flush_fn) |
107 | ret = -EOPNOTSUPP; | 107 | ret = -EOPNOTSUPP; |
@@ -118,7 +118,7 @@ static int linear_congested(void *data, int bits) | |||
118 | int i, ret = 0; | 118 | int i, ret = 0; |
119 | 119 | ||
120 | for (i = 0; i < mddev->raid_disks && !ret ; i++) { | 120 | for (i = 0; i < mddev->raid_disks && !ret ; i++) { |
121 | request_queue_t *q = bdev_get_queue(conf->disks[i].rdev->bdev); | 121 | struct request_queue *q = bdev_get_queue(conf->disks[i].rdev->bdev); |
122 | ret |= bdi_congested(&q->backing_dev_info, bits); | 122 | ret |= bdi_congested(&q->backing_dev_info, bits); |
123 | } | 123 | } |
124 | return ret; | 124 | return ret; |
@@ -330,7 +330,7 @@ static int linear_stop (mddev_t *mddev) | |||
330 | return 0; | 330 | return 0; |
331 | } | 331 | } |
332 | 332 | ||
333 | static int linear_make_request (request_queue_t *q, struct bio *bio) | 333 | static int linear_make_request (struct request_queue *q, struct bio *bio) |
334 | { | 334 | { |
335 | const int rw = bio_data_dir(bio); | 335 | const int rw = bio_data_dir(bio); |
336 | mddev_t *mddev = q->queuedata; | 336 | mddev_t *mddev = q->queuedata; |
diff --git a/drivers/md/md.c b/drivers/md/md.c index 65ddc887dfd7..f883b7e37f3d 100644 --- a/drivers/md/md.c +++ b/drivers/md/md.c | |||
@@ -211,7 +211,7 @@ static DEFINE_SPINLOCK(all_mddevs_lock); | |||
211 | ) | 211 | ) |
212 | 212 | ||
213 | 213 | ||
214 | static int md_fail_request (request_queue_t *q, struct bio *bio) | 214 | static int md_fail_request (struct request_queue *q, struct bio *bio) |
215 | { | 215 | { |
216 | bio_io_error(bio, bio->bi_size); | 216 | bio_io_error(bio, bio->bi_size); |
217 | return 0; | 217 | return 0; |
diff --git a/drivers/md/multipath.c b/drivers/md/multipath.c index 14da37fee37b..1e2af43a73b9 100644 --- a/drivers/md/multipath.c +++ b/drivers/md/multipath.c | |||
@@ -125,7 +125,7 @@ static void unplug_slaves(mddev_t *mddev) | |||
125 | mdk_rdev_t *rdev = rcu_dereference(conf->multipaths[i].rdev); | 125 | mdk_rdev_t *rdev = rcu_dereference(conf->multipaths[i].rdev); |
126 | if (rdev && !test_bit(Faulty, &rdev->flags) | 126 | if (rdev && !test_bit(Faulty, &rdev->flags) |
127 | && atomic_read(&rdev->nr_pending)) { | 127 | && atomic_read(&rdev->nr_pending)) { |
128 | request_queue_t *r_queue = bdev_get_queue(rdev->bdev); | 128 | struct request_queue *r_queue = bdev_get_queue(rdev->bdev); |
129 | 129 | ||
130 | atomic_inc(&rdev->nr_pending); | 130 | atomic_inc(&rdev->nr_pending); |
131 | rcu_read_unlock(); | 131 | rcu_read_unlock(); |
@@ -140,13 +140,13 @@ static void unplug_slaves(mddev_t *mddev) | |||
140 | rcu_read_unlock(); | 140 | rcu_read_unlock(); |
141 | } | 141 | } |
142 | 142 | ||
143 | static void multipath_unplug(request_queue_t *q) | 143 | static void multipath_unplug(struct request_queue *q) |
144 | { | 144 | { |
145 | unplug_slaves(q->queuedata); | 145 | unplug_slaves(q->queuedata); |
146 | } | 146 | } |
147 | 147 | ||
148 | 148 | ||
149 | static int multipath_make_request (request_queue_t *q, struct bio * bio) | 149 | static int multipath_make_request (struct request_queue *q, struct bio * bio) |
150 | { | 150 | { |
151 | mddev_t *mddev = q->queuedata; | 151 | mddev_t *mddev = q->queuedata; |
152 | multipath_conf_t *conf = mddev_to_conf(mddev); | 152 | multipath_conf_t *conf = mddev_to_conf(mddev); |
@@ -199,7 +199,7 @@ static void multipath_status (struct seq_file *seq, mddev_t *mddev) | |||
199 | seq_printf (seq, "]"); | 199 | seq_printf (seq, "]"); |
200 | } | 200 | } |
201 | 201 | ||
202 | static int multipath_issue_flush(request_queue_t *q, struct gendisk *disk, | 202 | static int multipath_issue_flush(struct request_queue *q, struct gendisk *disk, |
203 | sector_t *error_sector) | 203 | sector_t *error_sector) |
204 | { | 204 | { |
205 | mddev_t *mddev = q->queuedata; | 205 | mddev_t *mddev = q->queuedata; |
@@ -211,7 +211,7 @@ static int multipath_issue_flush(request_queue_t *q, struct gendisk *disk, | |||
211 | mdk_rdev_t *rdev = rcu_dereference(conf->multipaths[i].rdev); | 211 | mdk_rdev_t *rdev = rcu_dereference(conf->multipaths[i].rdev); |
212 | if (rdev && !test_bit(Faulty, &rdev->flags)) { | 212 | if (rdev && !test_bit(Faulty, &rdev->flags)) { |
213 | struct block_device *bdev = rdev->bdev; | 213 | struct block_device *bdev = rdev->bdev; |
214 | request_queue_t *r_queue = bdev_get_queue(bdev); | 214 | struct request_queue *r_queue = bdev_get_queue(bdev); |
215 | 215 | ||
216 | if (!r_queue->issue_flush_fn) | 216 | if (!r_queue->issue_flush_fn) |
217 | ret = -EOPNOTSUPP; | 217 | ret = -EOPNOTSUPP; |
@@ -238,7 +238,7 @@ static int multipath_congested(void *data, int bits) | |||
238 | for (i = 0; i < mddev->raid_disks ; i++) { | 238 | for (i = 0; i < mddev->raid_disks ; i++) { |
239 | mdk_rdev_t *rdev = rcu_dereference(conf->multipaths[i].rdev); | 239 | mdk_rdev_t *rdev = rcu_dereference(conf->multipaths[i].rdev); |
240 | if (rdev && !test_bit(Faulty, &rdev->flags)) { | 240 | if (rdev && !test_bit(Faulty, &rdev->flags)) { |
241 | request_queue_t *q = bdev_get_queue(rdev->bdev); | 241 | struct request_queue *q = bdev_get_queue(rdev->bdev); |
242 | 242 | ||
243 | ret |= bdi_congested(&q->backing_dev_info, bits); | 243 | ret |= bdi_congested(&q->backing_dev_info, bits); |
244 | /* Just like multipath_map, we just check the | 244 | /* Just like multipath_map, we just check the |
diff --git a/drivers/md/raid0.c b/drivers/md/raid0.c index 2c404f73a377..b8216bc6db45 100644 --- a/drivers/md/raid0.c +++ b/drivers/md/raid0.c | |||
@@ -25,7 +25,7 @@ | |||
25 | #define MD_DRIVER | 25 | #define MD_DRIVER |
26 | #define MD_PERSONALITY | 26 | #define MD_PERSONALITY |
27 | 27 | ||
28 | static void raid0_unplug(request_queue_t *q) | 28 | static void raid0_unplug(struct request_queue *q) |
29 | { | 29 | { |
30 | mddev_t *mddev = q->queuedata; | 30 | mddev_t *mddev = q->queuedata; |
31 | raid0_conf_t *conf = mddev_to_conf(mddev); | 31 | raid0_conf_t *conf = mddev_to_conf(mddev); |
@@ -33,14 +33,14 @@ static void raid0_unplug(request_queue_t *q) | |||
33 | int i; | 33 | int i; |
34 | 34 | ||
35 | for (i=0; i<mddev->raid_disks; i++) { | 35 | for (i=0; i<mddev->raid_disks; i++) { |
36 | request_queue_t *r_queue = bdev_get_queue(devlist[i]->bdev); | 36 | struct request_queue *r_queue = bdev_get_queue(devlist[i]->bdev); |
37 | 37 | ||
38 | if (r_queue->unplug_fn) | 38 | if (r_queue->unplug_fn) |
39 | r_queue->unplug_fn(r_queue); | 39 | r_queue->unplug_fn(r_queue); |
40 | } | 40 | } |
41 | } | 41 | } |
42 | 42 | ||
43 | static int raid0_issue_flush(request_queue_t *q, struct gendisk *disk, | 43 | static int raid0_issue_flush(struct request_queue *q, struct gendisk *disk, |
44 | sector_t *error_sector) | 44 | sector_t *error_sector) |
45 | { | 45 | { |
46 | mddev_t *mddev = q->queuedata; | 46 | mddev_t *mddev = q->queuedata; |
@@ -50,7 +50,7 @@ static int raid0_issue_flush(request_queue_t *q, struct gendisk *disk, | |||
50 | 50 | ||
51 | for (i=0; i<mddev->raid_disks && ret == 0; i++) { | 51 | for (i=0; i<mddev->raid_disks && ret == 0; i++) { |
52 | struct block_device *bdev = devlist[i]->bdev; | 52 | struct block_device *bdev = devlist[i]->bdev; |
53 | request_queue_t *r_queue = bdev_get_queue(bdev); | 53 | struct request_queue *r_queue = bdev_get_queue(bdev); |
54 | 54 | ||
55 | if (!r_queue->issue_flush_fn) | 55 | if (!r_queue->issue_flush_fn) |
56 | ret = -EOPNOTSUPP; | 56 | ret = -EOPNOTSUPP; |
@@ -68,7 +68,7 @@ static int raid0_congested(void *data, int bits) | |||
68 | int i, ret = 0; | 68 | int i, ret = 0; |
69 | 69 | ||
70 | for (i = 0; i < mddev->raid_disks && !ret ; i++) { | 70 | for (i = 0; i < mddev->raid_disks && !ret ; i++) { |
71 | request_queue_t *q = bdev_get_queue(devlist[i]->bdev); | 71 | struct request_queue *q = bdev_get_queue(devlist[i]->bdev); |
72 | 72 | ||
73 | ret |= bdi_congested(&q->backing_dev_info, bits); | 73 | ret |= bdi_congested(&q->backing_dev_info, bits); |
74 | } | 74 | } |
@@ -268,7 +268,7 @@ static int create_strip_zones (mddev_t *mddev) | |||
268 | * | 268 | * |
269 | * Return amount of bytes we can accept at this offset | 269 | * Return amount of bytes we can accept at this offset |
270 | */ | 270 | */ |
271 | static int raid0_mergeable_bvec(request_queue_t *q, struct bio *bio, struct bio_vec *biovec) | 271 | static int raid0_mergeable_bvec(struct request_queue *q, struct bio *bio, struct bio_vec *biovec) |
272 | { | 272 | { |
273 | mddev_t *mddev = q->queuedata; | 273 | mddev_t *mddev = q->queuedata; |
274 | sector_t sector = bio->bi_sector + get_start_sect(bio->bi_bdev); | 274 | sector_t sector = bio->bi_sector + get_start_sect(bio->bi_bdev); |
@@ -408,7 +408,7 @@ static int raid0_stop (mddev_t *mddev) | |||
408 | return 0; | 408 | return 0; |
409 | } | 409 | } |
410 | 410 | ||
411 | static int raid0_make_request (request_queue_t *q, struct bio *bio) | 411 | static int raid0_make_request (struct request_queue *q, struct bio *bio) |
412 | { | 412 | { |
413 | mddev_t *mddev = q->queuedata; | 413 | mddev_t *mddev = q->queuedata; |
414 | unsigned int sect_in_chunk, chunksize_bits, chunk_size, chunk_sects; | 414 | unsigned int sect_in_chunk, chunksize_bits, chunk_size, chunk_sects; |
diff --git a/drivers/md/raid1.c b/drivers/md/raid1.c index 00c78b77b13d..650991bddd8e 100644 --- a/drivers/md/raid1.c +++ b/drivers/md/raid1.c | |||
@@ -552,7 +552,7 @@ static void unplug_slaves(mddev_t *mddev) | |||
552 | for (i=0; i<mddev->raid_disks; i++) { | 552 | for (i=0; i<mddev->raid_disks; i++) { |
553 | mdk_rdev_t *rdev = rcu_dereference(conf->mirrors[i].rdev); | 553 | mdk_rdev_t *rdev = rcu_dereference(conf->mirrors[i].rdev); |
554 | if (rdev && !test_bit(Faulty, &rdev->flags) && atomic_read(&rdev->nr_pending)) { | 554 | if (rdev && !test_bit(Faulty, &rdev->flags) && atomic_read(&rdev->nr_pending)) { |
555 | request_queue_t *r_queue = bdev_get_queue(rdev->bdev); | 555 | struct request_queue *r_queue = bdev_get_queue(rdev->bdev); |
556 | 556 | ||
557 | atomic_inc(&rdev->nr_pending); | 557 | atomic_inc(&rdev->nr_pending); |
558 | rcu_read_unlock(); | 558 | rcu_read_unlock(); |
@@ -567,7 +567,7 @@ static void unplug_slaves(mddev_t *mddev) | |||
567 | rcu_read_unlock(); | 567 | rcu_read_unlock(); |
568 | } | 568 | } |
569 | 569 | ||
570 | static void raid1_unplug(request_queue_t *q) | 570 | static void raid1_unplug(struct request_queue *q) |
571 | { | 571 | { |
572 | mddev_t *mddev = q->queuedata; | 572 | mddev_t *mddev = q->queuedata; |
573 | 573 | ||
@@ -575,7 +575,7 @@ static void raid1_unplug(request_queue_t *q) | |||
575 | md_wakeup_thread(mddev->thread); | 575 | md_wakeup_thread(mddev->thread); |
576 | } | 576 | } |
577 | 577 | ||
578 | static int raid1_issue_flush(request_queue_t *q, struct gendisk *disk, | 578 | static int raid1_issue_flush(struct request_queue *q, struct gendisk *disk, |
579 | sector_t *error_sector) | 579 | sector_t *error_sector) |
580 | { | 580 | { |
581 | mddev_t *mddev = q->queuedata; | 581 | mddev_t *mddev = q->queuedata; |
@@ -587,7 +587,7 @@ static int raid1_issue_flush(request_queue_t *q, struct gendisk *disk, | |||
587 | mdk_rdev_t *rdev = rcu_dereference(conf->mirrors[i].rdev); | 587 | mdk_rdev_t *rdev = rcu_dereference(conf->mirrors[i].rdev); |
588 | if (rdev && !test_bit(Faulty, &rdev->flags)) { | 588 | if (rdev && !test_bit(Faulty, &rdev->flags)) { |
589 | struct block_device *bdev = rdev->bdev; | 589 | struct block_device *bdev = rdev->bdev; |
590 | request_queue_t *r_queue = bdev_get_queue(bdev); | 590 | struct request_queue *r_queue = bdev_get_queue(bdev); |
591 | 591 | ||
592 | if (!r_queue->issue_flush_fn) | 592 | if (!r_queue->issue_flush_fn) |
593 | ret = -EOPNOTSUPP; | 593 | ret = -EOPNOTSUPP; |
@@ -615,7 +615,7 @@ static int raid1_congested(void *data, int bits) | |||
615 | for (i = 0; i < mddev->raid_disks; i++) { | 615 | for (i = 0; i < mddev->raid_disks; i++) { |
616 | mdk_rdev_t *rdev = rcu_dereference(conf->mirrors[i].rdev); | 616 | mdk_rdev_t *rdev = rcu_dereference(conf->mirrors[i].rdev); |
617 | if (rdev && !test_bit(Faulty, &rdev->flags)) { | 617 | if (rdev && !test_bit(Faulty, &rdev->flags)) { |
618 | request_queue_t *q = bdev_get_queue(rdev->bdev); | 618 | struct request_queue *q = bdev_get_queue(rdev->bdev); |
619 | 619 | ||
620 | /* Note the '|| 1' - when read_balance prefers | 620 | /* Note the '|| 1' - when read_balance prefers |
621 | * non-congested targets, it can be removed | 621 | * non-congested targets, it can be removed |
@@ -765,7 +765,7 @@ do_sync_io: | |||
765 | return NULL; | 765 | return NULL; |
766 | } | 766 | } |
767 | 767 | ||
768 | static int make_request(request_queue_t *q, struct bio * bio) | 768 | static int make_request(struct request_queue *q, struct bio * bio) |
769 | { | 769 | { |
770 | mddev_t *mddev = q->queuedata; | 770 | mddev_t *mddev = q->queuedata; |
771 | conf_t *conf = mddev_to_conf(mddev); | 771 | conf_t *conf = mddev_to_conf(mddev); |
diff --git a/drivers/md/raid10.c b/drivers/md/raid10.c index a95ada1cfac4..f730a144baf1 100644 --- a/drivers/md/raid10.c +++ b/drivers/md/raid10.c | |||
@@ -453,7 +453,7 @@ static sector_t raid10_find_virt(conf_t *conf, sector_t sector, int dev) | |||
453 | * If near_copies == raid_disk, there are no striping issues, | 453 | * If near_copies == raid_disk, there are no striping issues, |
454 | * but in that case, the function isn't called at all. | 454 | * but in that case, the function isn't called at all. |
455 | */ | 455 | */ |
456 | static int raid10_mergeable_bvec(request_queue_t *q, struct bio *bio, | 456 | static int raid10_mergeable_bvec(struct request_queue *q, struct bio *bio, |
457 | struct bio_vec *bio_vec) | 457 | struct bio_vec *bio_vec) |
458 | { | 458 | { |
459 | mddev_t *mddev = q->queuedata; | 459 | mddev_t *mddev = q->queuedata; |
@@ -595,7 +595,7 @@ static void unplug_slaves(mddev_t *mddev) | |||
595 | for (i=0; i<mddev->raid_disks; i++) { | 595 | for (i=0; i<mddev->raid_disks; i++) { |
596 | mdk_rdev_t *rdev = rcu_dereference(conf->mirrors[i].rdev); | 596 | mdk_rdev_t *rdev = rcu_dereference(conf->mirrors[i].rdev); |
597 | if (rdev && !test_bit(Faulty, &rdev->flags) && atomic_read(&rdev->nr_pending)) { | 597 | if (rdev && !test_bit(Faulty, &rdev->flags) && atomic_read(&rdev->nr_pending)) { |
598 | request_queue_t *r_queue = bdev_get_queue(rdev->bdev); | 598 | struct request_queue *r_queue = bdev_get_queue(rdev->bdev); |
599 | 599 | ||
600 | atomic_inc(&rdev->nr_pending); | 600 | atomic_inc(&rdev->nr_pending); |
601 | rcu_read_unlock(); | 601 | rcu_read_unlock(); |
@@ -610,7 +610,7 @@ static void unplug_slaves(mddev_t *mddev) | |||
610 | rcu_read_unlock(); | 610 | rcu_read_unlock(); |
611 | } | 611 | } |
612 | 612 | ||
613 | static void raid10_unplug(request_queue_t *q) | 613 | static void raid10_unplug(struct request_queue *q) |
614 | { | 614 | { |
615 | mddev_t *mddev = q->queuedata; | 615 | mddev_t *mddev = q->queuedata; |
616 | 616 | ||
@@ -618,7 +618,7 @@ static void raid10_unplug(request_queue_t *q) | |||
618 | md_wakeup_thread(mddev->thread); | 618 | md_wakeup_thread(mddev->thread); |
619 | } | 619 | } |
620 | 620 | ||
621 | static int raid10_issue_flush(request_queue_t *q, struct gendisk *disk, | 621 | static int raid10_issue_flush(struct request_queue *q, struct gendisk *disk, |
622 | sector_t *error_sector) | 622 | sector_t *error_sector) |
623 | { | 623 | { |
624 | mddev_t *mddev = q->queuedata; | 624 | mddev_t *mddev = q->queuedata; |
@@ -630,7 +630,7 @@ static int raid10_issue_flush(request_queue_t *q, struct gendisk *disk, | |||
630 | mdk_rdev_t *rdev = rcu_dereference(conf->mirrors[i].rdev); | 630 | mdk_rdev_t *rdev = rcu_dereference(conf->mirrors[i].rdev); |
631 | if (rdev && !test_bit(Faulty, &rdev->flags)) { | 631 | if (rdev && !test_bit(Faulty, &rdev->flags)) { |
632 | struct block_device *bdev = rdev->bdev; | 632 | struct block_device *bdev = rdev->bdev; |
633 | request_queue_t *r_queue = bdev_get_queue(bdev); | 633 | struct request_queue *r_queue = bdev_get_queue(bdev); |
634 | 634 | ||
635 | if (!r_queue->issue_flush_fn) | 635 | if (!r_queue->issue_flush_fn) |
636 | ret = -EOPNOTSUPP; | 636 | ret = -EOPNOTSUPP; |
@@ -658,7 +658,7 @@ static int raid10_congested(void *data, int bits) | |||
658 | for (i = 0; i < mddev->raid_disks && ret == 0; i++) { | 658 | for (i = 0; i < mddev->raid_disks && ret == 0; i++) { |
659 | mdk_rdev_t *rdev = rcu_dereference(conf->mirrors[i].rdev); | 659 | mdk_rdev_t *rdev = rcu_dereference(conf->mirrors[i].rdev); |
660 | if (rdev && !test_bit(Faulty, &rdev->flags)) { | 660 | if (rdev && !test_bit(Faulty, &rdev->flags)) { |
661 | request_queue_t *q = bdev_get_queue(rdev->bdev); | 661 | struct request_queue *q = bdev_get_queue(rdev->bdev); |
662 | 662 | ||
663 | ret |= bdi_congested(&q->backing_dev_info, bits); | 663 | ret |= bdi_congested(&q->backing_dev_info, bits); |
664 | } | 664 | } |
@@ -772,7 +772,7 @@ static void unfreeze_array(conf_t *conf) | |||
772 | spin_unlock_irq(&conf->resync_lock); | 772 | spin_unlock_irq(&conf->resync_lock); |
773 | } | 773 | } |
774 | 774 | ||
775 | static int make_request(request_queue_t *q, struct bio * bio) | 775 | static int make_request(struct request_queue *q, struct bio * bio) |
776 | { | 776 | { |
777 | mddev_t *mddev = q->queuedata; | 777 | mddev_t *mddev = q->queuedata; |
778 | conf_t *conf = mddev_to_conf(mddev); | 778 | conf_t *conf = mddev_to_conf(mddev); |
diff --git a/drivers/md/raid5.c b/drivers/md/raid5.c index d90ee145effe..2aff4be35dc4 100644 --- a/drivers/md/raid5.c +++ b/drivers/md/raid5.c | |||
@@ -289,7 +289,7 @@ static struct stripe_head *__find_stripe(raid5_conf_t *conf, sector_t sector, in | |||
289 | } | 289 | } |
290 | 290 | ||
291 | static void unplug_slaves(mddev_t *mddev); | 291 | static void unplug_slaves(mddev_t *mddev); |
292 | static void raid5_unplug_device(request_queue_t *q); | 292 | static void raid5_unplug_device(struct request_queue *q); |
293 | 293 | ||
294 | static struct stripe_head *get_active_stripe(raid5_conf_t *conf, sector_t sector, int disks, | 294 | static struct stripe_head *get_active_stripe(raid5_conf_t *conf, sector_t sector, int disks, |
295 | int pd_idx, int noblock) | 295 | int pd_idx, int noblock) |
@@ -3182,7 +3182,7 @@ static void unplug_slaves(mddev_t *mddev) | |||
3182 | for (i=0; i<mddev->raid_disks; i++) { | 3182 | for (i=0; i<mddev->raid_disks; i++) { |
3183 | mdk_rdev_t *rdev = rcu_dereference(conf->disks[i].rdev); | 3183 | mdk_rdev_t *rdev = rcu_dereference(conf->disks[i].rdev); |
3184 | if (rdev && !test_bit(Faulty, &rdev->flags) && atomic_read(&rdev->nr_pending)) { | 3184 | if (rdev && !test_bit(Faulty, &rdev->flags) && atomic_read(&rdev->nr_pending)) { |
3185 | request_queue_t *r_queue = bdev_get_queue(rdev->bdev); | 3185 | struct request_queue *r_queue = bdev_get_queue(rdev->bdev); |
3186 | 3186 | ||
3187 | atomic_inc(&rdev->nr_pending); | 3187 | atomic_inc(&rdev->nr_pending); |
3188 | rcu_read_unlock(); | 3188 | rcu_read_unlock(); |
@@ -3197,7 +3197,7 @@ static void unplug_slaves(mddev_t *mddev) | |||
3197 | rcu_read_unlock(); | 3197 | rcu_read_unlock(); |
3198 | } | 3198 | } |
3199 | 3199 | ||
3200 | static void raid5_unplug_device(request_queue_t *q) | 3200 | static void raid5_unplug_device(struct request_queue *q) |
3201 | { | 3201 | { |
3202 | mddev_t *mddev = q->queuedata; | 3202 | mddev_t *mddev = q->queuedata; |
3203 | raid5_conf_t *conf = mddev_to_conf(mddev); | 3203 | raid5_conf_t *conf = mddev_to_conf(mddev); |
@@ -3216,7 +3216,7 @@ static void raid5_unplug_device(request_queue_t *q) | |||
3216 | unplug_slaves(mddev); | 3216 | unplug_slaves(mddev); |
3217 | } | 3217 | } |
3218 | 3218 | ||
3219 | static int raid5_issue_flush(request_queue_t *q, struct gendisk *disk, | 3219 | static int raid5_issue_flush(struct request_queue *q, struct gendisk *disk, |
3220 | sector_t *error_sector) | 3220 | sector_t *error_sector) |
3221 | { | 3221 | { |
3222 | mddev_t *mddev = q->queuedata; | 3222 | mddev_t *mddev = q->queuedata; |
@@ -3228,7 +3228,7 @@ static int raid5_issue_flush(request_queue_t *q, struct gendisk *disk, | |||
3228 | mdk_rdev_t *rdev = rcu_dereference(conf->disks[i].rdev); | 3228 | mdk_rdev_t *rdev = rcu_dereference(conf->disks[i].rdev); |
3229 | if (rdev && !test_bit(Faulty, &rdev->flags)) { | 3229 | if (rdev && !test_bit(Faulty, &rdev->flags)) { |
3230 | struct block_device *bdev = rdev->bdev; | 3230 | struct block_device *bdev = rdev->bdev; |
3231 | request_queue_t *r_queue = bdev_get_queue(bdev); | 3231 | struct request_queue *r_queue = bdev_get_queue(bdev); |
3232 | 3232 | ||
3233 | if (!r_queue->issue_flush_fn) | 3233 | if (!r_queue->issue_flush_fn) |
3234 | ret = -EOPNOTSUPP; | 3234 | ret = -EOPNOTSUPP; |
@@ -3267,7 +3267,7 @@ static int raid5_congested(void *data, int bits) | |||
3267 | /* We want read requests to align with chunks where possible, | 3267 | /* We want read requests to align with chunks where possible, |
3268 | * but write requests don't need to. | 3268 | * but write requests don't need to. |
3269 | */ | 3269 | */ |
3270 | static int raid5_mergeable_bvec(request_queue_t *q, struct bio *bio, struct bio_vec *biovec) | 3270 | static int raid5_mergeable_bvec(struct request_queue *q, struct bio *bio, struct bio_vec *biovec) |
3271 | { | 3271 | { |
3272 | mddev_t *mddev = q->queuedata; | 3272 | mddev_t *mddev = q->queuedata; |
3273 | sector_t sector = bio->bi_sector + get_start_sect(bio->bi_bdev); | 3273 | sector_t sector = bio->bi_sector + get_start_sect(bio->bi_bdev); |
@@ -3377,7 +3377,7 @@ static int raid5_align_endio(struct bio *bi, unsigned int bytes, int error) | |||
3377 | 3377 | ||
3378 | static int bio_fits_rdev(struct bio *bi) | 3378 | static int bio_fits_rdev(struct bio *bi) |
3379 | { | 3379 | { |
3380 | request_queue_t *q = bdev_get_queue(bi->bi_bdev); | 3380 | struct request_queue *q = bdev_get_queue(bi->bi_bdev); |
3381 | 3381 | ||
3382 | if ((bi->bi_size>>9) > q->max_sectors) | 3382 | if ((bi->bi_size>>9) > q->max_sectors) |
3383 | return 0; | 3383 | return 0; |
@@ -3396,7 +3396,7 @@ static int bio_fits_rdev(struct bio *bi) | |||
3396 | } | 3396 | } |
3397 | 3397 | ||
3398 | 3398 | ||
3399 | static int chunk_aligned_read(request_queue_t *q, struct bio * raid_bio) | 3399 | static int chunk_aligned_read(struct request_queue *q, struct bio * raid_bio) |
3400 | { | 3400 | { |
3401 | mddev_t *mddev = q->queuedata; | 3401 | mddev_t *mddev = q->queuedata; |
3402 | raid5_conf_t *conf = mddev_to_conf(mddev); | 3402 | raid5_conf_t *conf = mddev_to_conf(mddev); |
@@ -3466,7 +3466,7 @@ static int chunk_aligned_read(request_queue_t *q, struct bio * raid_bio) | |||
3466 | } | 3466 | } |
3467 | 3467 | ||
3468 | 3468 | ||
3469 | static int make_request(request_queue_t *q, struct bio * bi) | 3469 | static int make_request(struct request_queue *q, struct bio * bi) |
3470 | { | 3470 | { |
3471 | mddev_t *mddev = q->queuedata; | 3471 | mddev_t *mddev = q->queuedata; |
3472 | raid5_conf_t *conf = mddev_to_conf(mddev); | 3472 | raid5_conf_t *conf = mddev_to_conf(mddev); |
diff --git a/drivers/media/video/Kconfig b/drivers/media/video/Kconfig index 9dcbffd0aa15..e204e7b4028a 100644 --- a/drivers/media/video/Kconfig +++ b/drivers/media/video/Kconfig | |||
@@ -509,7 +509,7 @@ config VIDEO_VINO | |||
509 | 509 | ||
510 | config VIDEO_STRADIS | 510 | config VIDEO_STRADIS |
511 | tristate "Stradis 4:2:2 MPEG-2 video driver (EXPERIMENTAL)" | 511 | tristate "Stradis 4:2:2 MPEG-2 video driver (EXPERIMENTAL)" |
512 | depends on EXPERIMENTAL && PCI && VIDEO_V4L1 && !PPC64 | 512 | depends on EXPERIMENTAL && PCI && VIDEO_V4L1 && VIRT_TO_BUS |
513 | help | 513 | help |
514 | Say Y here to enable support for the Stradis 4:2:2 MPEG-2 video | 514 | Say Y here to enable support for the Stradis 4:2:2 MPEG-2 video |
515 | driver for PCI. There is a product page at | 515 | driver for PCI. There is a product page at |
@@ -520,7 +520,7 @@ config VIDEO_ZORAN_ZR36060 | |||
520 | 520 | ||
521 | config VIDEO_ZORAN | 521 | config VIDEO_ZORAN |
522 | tristate "Zoran ZR36057/36067 Video For Linux" | 522 | tristate "Zoran ZR36057/36067 Video For Linux" |
523 | depends on PCI && I2C_ALGOBIT && VIDEO_V4L1 && !PPC64 | 523 | depends on PCI && I2C_ALGOBIT && VIDEO_V4L1 && VIRT_TO_BUS |
524 | help | 524 | help |
525 | Say Y for support for MJPEG capture cards based on the Zoran | 525 | Say Y for support for MJPEG capture cards based on the Zoran |
526 | 36057/36067 PCI controller chipset. This includes the Iomega | 526 | 36057/36067 PCI controller chipset. This includes the Iomega |
diff --git a/drivers/message/i2o/i2o_block.c b/drivers/message/i2o/i2o_block.c index 988c8ce47f58..5e1c99f83ab5 100644 --- a/drivers/message/i2o/i2o_block.c +++ b/drivers/message/i2o/i2o_block.c | |||
@@ -159,7 +159,7 @@ static int i2o_block_device_flush(struct i2o_device *dev) | |||
159 | * Returns 0 on success or negative error code on failure. | 159 | * Returns 0 on success or negative error code on failure. |
160 | */ | 160 | */ |
161 | 161 | ||
162 | static int i2o_block_issue_flush(request_queue_t * queue, struct gendisk *disk, | 162 | static int i2o_block_issue_flush(struct request_queue * queue, struct gendisk *disk, |
163 | sector_t * error_sector) | 163 | sector_t * error_sector) |
164 | { | 164 | { |
165 | struct i2o_block_device *i2o_blk_dev = queue->queuedata; | 165 | struct i2o_block_device *i2o_blk_dev = queue->queuedata; |
@@ -445,7 +445,7 @@ static void i2o_block_end_request(struct request *req, int uptodate, | |||
445 | { | 445 | { |
446 | struct i2o_block_request *ireq = req->special; | 446 | struct i2o_block_request *ireq = req->special; |
447 | struct i2o_block_device *dev = ireq->i2o_blk_dev; | 447 | struct i2o_block_device *dev = ireq->i2o_blk_dev; |
448 | request_queue_t *q = req->q; | 448 | struct request_queue *q = req->q; |
449 | unsigned long flags; | 449 | unsigned long flags; |
450 | 450 | ||
451 | if (end_that_request_chunk(req, uptodate, nr_bytes)) { | 451 | if (end_that_request_chunk(req, uptodate, nr_bytes)) { |
diff --git a/drivers/misc/asus-laptop.c b/drivers/misc/asus-laptop.c index 6b89854bd3ff..d0fc4fd212e6 100644 --- a/drivers/misc/asus-laptop.c +++ b/drivers/misc/asus-laptop.c | |||
@@ -53,7 +53,6 @@ | |||
53 | #define ASUS_HOTK_NAME "Asus Laptop Support" | 53 | #define ASUS_HOTK_NAME "Asus Laptop Support" |
54 | #define ASUS_HOTK_CLASS "hotkey" | 54 | #define ASUS_HOTK_CLASS "hotkey" |
55 | #define ASUS_HOTK_DEVICE_NAME "Hotkey" | 55 | #define ASUS_HOTK_DEVICE_NAME "Hotkey" |
56 | #define ASUS_HOTK_HID "ATK0100" | ||
57 | #define ASUS_HOTK_FILE "asus-laptop" | 56 | #define ASUS_HOTK_FILE "asus-laptop" |
58 | #define ASUS_HOTK_PREFIX "\\_SB.ATKD." | 57 | #define ASUS_HOTK_PREFIX "\\_SB.ATKD." |
59 | 58 | ||
@@ -197,12 +196,18 @@ static struct asus_hotk *hotk; | |||
197 | /* | 196 | /* |
198 | * The hotkey driver declaration | 197 | * The hotkey driver declaration |
199 | */ | 198 | */ |
199 | static const struct acpi_device_id asus_device_ids[] = { | ||
200 | {"ATK0100", 0}, | ||
201 | {"", 0}, | ||
202 | }; | ||
203 | MODULE_DEVICE_TABLE(acpi, asus_device_ids); | ||
204 | |||
200 | static int asus_hotk_add(struct acpi_device *device); | 205 | static int asus_hotk_add(struct acpi_device *device); |
201 | static int asus_hotk_remove(struct acpi_device *device, int type); | 206 | static int asus_hotk_remove(struct acpi_device *device, int type); |
202 | static struct acpi_driver asus_hotk_driver = { | 207 | static struct acpi_driver asus_hotk_driver = { |
203 | .name = ASUS_HOTK_NAME, | 208 | .name = ASUS_HOTK_NAME, |
204 | .class = ASUS_HOTK_CLASS, | 209 | .class = ASUS_HOTK_CLASS, |
205 | .ids = ASUS_HOTK_HID, | 210 | .ids = asus_device_ids, |
206 | .ops = { | 211 | .ops = { |
207 | .add = asus_hotk_add, | 212 | .add = asus_hotk_add, |
208 | .remove = asus_hotk_remove, | 213 | .remove = asus_hotk_remove, |
diff --git a/drivers/misc/sony-laptop.c b/drivers/misc/sony-laptop.c index 303e48ca0e8a..14ee06c8f127 100644 --- a/drivers/misc/sony-laptop.c +++ b/drivers/misc/sony-laptop.c | |||
@@ -1124,10 +1124,22 @@ static int sony_nc_remove(struct acpi_device *device, int type) | |||
1124 | return 0; | 1124 | return 0; |
1125 | } | 1125 | } |
1126 | 1126 | ||
1127 | static const struct acpi_device_id sony_device_ids[] = { | ||
1128 | {SONY_NC_HID, 0}, | ||
1129 | {SONY_PIC_HID, 0}, | ||
1130 | {"", 0}, | ||
1131 | }; | ||
1132 | MODULE_DEVICE_TABLE(acpi, sony_device_ids); | ||
1133 | |||
1134 | static const struct acpi_device_id sony_nc_device_ids[] = { | ||
1135 | {SONY_NC_HID, 0}, | ||
1136 | {"", 0}, | ||
1137 | }; | ||
1138 | |||
1127 | static struct acpi_driver sony_nc_driver = { | 1139 | static struct acpi_driver sony_nc_driver = { |
1128 | .name = SONY_NC_DRIVER_NAME, | 1140 | .name = SONY_NC_DRIVER_NAME, |
1129 | .class = SONY_NC_CLASS, | 1141 | .class = SONY_NC_CLASS, |
1130 | .ids = SONY_NC_HID, | 1142 | .ids = sony_nc_device_ids, |
1131 | .owner = THIS_MODULE, | 1143 | .owner = THIS_MODULE, |
1132 | .ops = { | 1144 | .ops = { |
1133 | .add = sony_nc_add, | 1145 | .add = sony_nc_add, |
@@ -2470,10 +2482,15 @@ static int sony_pic_resume(struct acpi_device *device) | |||
2470 | return 0; | 2482 | return 0; |
2471 | } | 2483 | } |
2472 | 2484 | ||
2485 | static const struct acpi_device_id sony_pic_device_ids[] = { | ||
2486 | {SONY_PIC_HID, 0}, | ||
2487 | {"", 0}, | ||
2488 | }; | ||
2489 | |||
2473 | static struct acpi_driver sony_pic_driver = { | 2490 | static struct acpi_driver sony_pic_driver = { |
2474 | .name = SONY_PIC_DRIVER_NAME, | 2491 | .name = SONY_PIC_DRIVER_NAME, |
2475 | .class = SONY_PIC_CLASS, | 2492 | .class = SONY_PIC_CLASS, |
2476 | .ids = SONY_PIC_HID, | 2493 | .ids = sony_pic_device_ids, |
2477 | .owner = THIS_MODULE, | 2494 | .owner = THIS_MODULE, |
2478 | .ops = { | 2495 | .ops = { |
2479 | .add = sony_pic_add, | 2496 | .add = sony_pic_add, |
diff --git a/drivers/misc/thinkpad_acpi.c b/drivers/misc/thinkpad_acpi.c index f15a58f7403f..fa80f355e522 100644 --- a/drivers/misc/thinkpad_acpi.c +++ b/drivers/misc/thinkpad_acpi.c | |||
@@ -411,12 +411,13 @@ static int __init register_tpacpi_subdriver(struct ibm_struct *ibm) | |||
411 | 411 | ||
412 | sprintf(ibm->acpi->driver->name, "%s_%s", IBM_NAME, ibm->name); | 412 | sprintf(ibm->acpi->driver->name, "%s_%s", IBM_NAME, ibm->name); |
413 | ibm->acpi->driver->ids = ibm->acpi->hid; | 413 | ibm->acpi->driver->ids = ibm->acpi->hid; |
414 | |||
414 | ibm->acpi->driver->ops.add = &tpacpi_device_add; | 415 | ibm->acpi->driver->ops.add = &tpacpi_device_add; |
415 | 416 | ||
416 | rc = acpi_bus_register_driver(ibm->acpi->driver); | 417 | rc = acpi_bus_register_driver(ibm->acpi->driver); |
417 | if (rc < 0) { | 418 | if (rc < 0) { |
418 | printk(IBM_ERR "acpi_bus_register_driver(%s) failed: %d\n", | 419 | printk(IBM_ERR "acpi_bus_register_driver(%s) failed: %d\n", |
419 | ibm->acpi->hid, rc); | 420 | ibm->name, rc); |
420 | kfree(ibm->acpi->driver); | 421 | kfree(ibm->acpi->driver); |
421 | ibm->acpi->driver = NULL; | 422 | ibm->acpi->driver = NULL; |
422 | } else if (!rc) | 423 | } else if (!rc) |
@@ -1316,8 +1317,13 @@ errexit: | |||
1316 | return res; | 1317 | return res; |
1317 | } | 1318 | } |
1318 | 1319 | ||
1320 | static const struct acpi_device_id ibm_htk_device_ids[] = { | ||
1321 | {IBM_HKEY_HID, 0}, | ||
1322 | {"", 0}, | ||
1323 | }; | ||
1324 | |||
1319 | static struct tp_acpi_drv_struct ibm_hotkey_acpidriver = { | 1325 | static struct tp_acpi_drv_struct ibm_hotkey_acpidriver = { |
1320 | .hid = IBM_HKEY_HID, | 1326 | .hid = ibm_htk_device_ids, |
1321 | .notify = hotkey_notify, | 1327 | .notify = hotkey_notify, |
1322 | .handle = &hkey_handle, | 1328 | .handle = &hkey_handle, |
1323 | .type = ACPI_DEVICE_NOTIFY, | 1329 | .type = ACPI_DEVICE_NOTIFY, |
@@ -2080,6 +2086,11 @@ IBM_HANDLE(dock, root, "\\_SB.GDCK", /* X30, X31, X40 */ | |||
2080 | /* don't list other alternatives as we install a notify handler on the 570 */ | 2086 | /* don't list other alternatives as we install a notify handler on the 570 */ |
2081 | IBM_HANDLE(pci, root, "\\_SB.PCI"); /* 570 */ | 2087 | IBM_HANDLE(pci, root, "\\_SB.PCI"); /* 570 */ |
2082 | 2088 | ||
2089 | static const struct acpi_device_id ibm_pci_device_ids[] = { | ||
2090 | {PCI_ROOT_HID_STRING, 0}, | ||
2091 | {"", 0}, | ||
2092 | }; | ||
2093 | |||
2083 | static struct tp_acpi_drv_struct ibm_dock_acpidriver[2] = { | 2094 | static struct tp_acpi_drv_struct ibm_dock_acpidriver[2] = { |
2084 | { | 2095 | { |
2085 | .notify = dock_notify, | 2096 | .notify = dock_notify, |
@@ -2090,7 +2101,7 @@ static struct tp_acpi_drv_struct ibm_dock_acpidriver[2] = { | |||
2090 | /* THIS ONE MUST NEVER BE USED FOR DRIVER AUTOLOADING. | 2101 | /* THIS ONE MUST NEVER BE USED FOR DRIVER AUTOLOADING. |
2091 | * We just use it to get notifications of dock hotplug | 2102 | * We just use it to get notifications of dock hotplug |
2092 | * in very old thinkpads */ | 2103 | * in very old thinkpads */ |
2093 | .hid = PCI_ROOT_HID_STRING, | 2104 | .hid = ibm_pci_device_ids, |
2094 | .notify = dock_notify, | 2105 | .notify = dock_notify, |
2095 | .handle = &pci_handle, | 2106 | .handle = &pci_handle, |
2096 | .type = ACPI_SYSTEM_NOTIFY, | 2107 | .type = ACPI_SYSTEM_NOTIFY, |
@@ -2149,7 +2160,8 @@ static int __init dock_init2(struct ibm_init_struct *iibm) | |||
2149 | static void dock_notify(struct ibm_struct *ibm, u32 event) | 2160 | static void dock_notify(struct ibm_struct *ibm, u32 event) |
2150 | { | 2161 | { |
2151 | int docked = dock_docked(); | 2162 | int docked = dock_docked(); |
2152 | int pci = ibm->acpi->hid && strstr(ibm->acpi->hid, PCI_ROOT_HID_STRING); | 2163 | int pci = ibm->acpi->hid && ibm->acpi->device && |
2164 | acpi_match_device_ids(ibm->acpi->device, ibm_pci_device_ids); | ||
2153 | 2165 | ||
2154 | if (event == 1 && !pci) /* 570 */ | 2166 | if (event == 1 && !pci) /* 570 */ |
2155 | acpi_bus_generate_event(ibm->acpi->device, event, 1); /* button */ | 2167 | acpi_bus_generate_event(ibm->acpi->device, event, 1); /* button */ |
diff --git a/drivers/misc/thinkpad_acpi.h b/drivers/misc/thinkpad_acpi.h index b7a4a888cc8b..88af089d6494 100644 --- a/drivers/misc/thinkpad_acpi.h +++ b/drivers/misc/thinkpad_acpi.h | |||
@@ -193,7 +193,7 @@ static void thinkpad_acpi_module_exit(void); | |||
193 | struct ibm_struct; | 193 | struct ibm_struct; |
194 | 194 | ||
195 | struct tp_acpi_drv_struct { | 195 | struct tp_acpi_drv_struct { |
196 | char *hid; | 196 | const struct acpi_device_id *hid; |
197 | struct acpi_driver *driver; | 197 | struct acpi_driver *driver; |
198 | 198 | ||
199 | void (*notify) (struct ibm_struct *, u32); | 199 | void (*notify) (struct ibm_struct *, u32); |
diff --git a/drivers/mmc/card/queue.c b/drivers/mmc/card/queue.c index 9f2b20fd9ab2..c9a289c6c139 100644 --- a/drivers/mmc/card/queue.c +++ b/drivers/mmc/card/queue.c | |||
@@ -83,7 +83,7 @@ static int mmc_queue_thread(void *d) | |||
83 | * on any queue on this host, and attempt to issue it. This may | 83 | * on any queue on this host, and attempt to issue it. This may |
84 | * not be the queue we were asked to process. | 84 | * not be the queue we were asked to process. |
85 | */ | 85 | */ |
86 | static void mmc_request(request_queue_t *q) | 86 | static void mmc_request(struct request_queue *q) |
87 | { | 87 | { |
88 | struct mmc_queue *mq = q->queuedata; | 88 | struct mmc_queue *mq = q->queuedata; |
89 | struct request *req; | 89 | struct request *req; |
@@ -211,7 +211,7 @@ int mmc_init_queue(struct mmc_queue *mq, struct mmc_card *card, spinlock_t *lock | |||
211 | 211 | ||
212 | void mmc_cleanup_queue(struct mmc_queue *mq) | 212 | void mmc_cleanup_queue(struct mmc_queue *mq) |
213 | { | 213 | { |
214 | request_queue_t *q = mq->queue; | 214 | struct request_queue *q = mq->queue; |
215 | unsigned long flags; | 215 | unsigned long flags; |
216 | 216 | ||
217 | /* Mark that we should start throwing out stragglers */ | 217 | /* Mark that we should start throwing out stragglers */ |
@@ -252,7 +252,7 @@ EXPORT_SYMBOL(mmc_cleanup_queue); | |||
252 | */ | 252 | */ |
253 | void mmc_queue_suspend(struct mmc_queue *mq) | 253 | void mmc_queue_suspend(struct mmc_queue *mq) |
254 | { | 254 | { |
255 | request_queue_t *q = mq->queue; | 255 | struct request_queue *q = mq->queue; |
256 | unsigned long flags; | 256 | unsigned long flags; |
257 | 257 | ||
258 | if (!(mq->flags & MMC_QUEUE_SUSPENDED)) { | 258 | if (!(mq->flags & MMC_QUEUE_SUSPENDED)) { |
@@ -272,7 +272,7 @@ void mmc_queue_suspend(struct mmc_queue *mq) | |||
272 | */ | 272 | */ |
273 | void mmc_queue_resume(struct mmc_queue *mq) | 273 | void mmc_queue_resume(struct mmc_queue *mq) |
274 | { | 274 | { |
275 | request_queue_t *q = mq->queue; | 275 | struct request_queue *q = mq->queue; |
276 | unsigned long flags; | 276 | unsigned long flags; |
277 | 277 | ||
278 | if (mq->flags & MMC_QUEUE_SUSPENDED) { | 278 | if (mq->flags & MMC_QUEUE_SUSPENDED) { |
diff --git a/drivers/mtd/maps/Kconfig b/drivers/mtd/maps/Kconfig index f88ebc5b685e..cc6c73442435 100644 --- a/drivers/mtd/maps/Kconfig +++ b/drivers/mtd/maps/Kconfig | |||
@@ -103,7 +103,7 @@ config MTD_PMC_MSP_RAMROOT | |||
103 | 103 | ||
104 | config MTD_SUN_UFLASH | 104 | config MTD_SUN_UFLASH |
105 | tristate "Sun Microsystems userflash support" | 105 | tristate "Sun Microsystems userflash support" |
106 | depends on SPARC && MTD_CFI | 106 | depends on SPARC && MTD_CFI && PCI |
107 | help | 107 | help |
108 | This provides a 'mapping' driver which supports the way in | 108 | This provides a 'mapping' driver which supports the way in |
109 | which user-programmable flash chips are connected on various | 109 | which user-programmable flash chips are connected on various |
diff --git a/drivers/net/82596.c b/drivers/net/82596.c index 3ff1155459a3..d915837193cc 100644 --- a/drivers/net/82596.c +++ b/drivers/net/82596.c | |||
@@ -57,6 +57,7 @@ | |||
57 | #include <asm/io.h> | 57 | #include <asm/io.h> |
58 | #include <asm/dma.h> | 58 | #include <asm/dma.h> |
59 | #include <asm/pgtable.h> | 59 | #include <asm/pgtable.h> |
60 | #include <asm/cacheflush.h> | ||
60 | 61 | ||
61 | static char version[] __initdata = | 62 | static char version[] __initdata = |
62 | "82596.c $Revision: 1.5 $\n"; | 63 | "82596.c $Revision: 1.5 $\n"; |
diff --git a/drivers/net/Makefile b/drivers/net/Makefile index 336af0635df8..94b78cc5fe86 100644 --- a/drivers/net/Makefile +++ b/drivers/net/Makefile | |||
@@ -18,7 +18,7 @@ gianfar_driver-objs := gianfar.o \ | |||
18 | gianfar_sysfs.o | 18 | gianfar_sysfs.o |
19 | 19 | ||
20 | obj-$(CONFIG_UCC_GETH) += ucc_geth_driver.o | 20 | obj-$(CONFIG_UCC_GETH) += ucc_geth_driver.o |
21 | ucc_geth_driver-objs := ucc_geth.o ucc_geth_mii.o | 21 | ucc_geth_driver-objs := ucc_geth.o ucc_geth_mii.o ucc_geth_ethtool.o |
22 | 22 | ||
23 | # | 23 | # |
24 | # link order important here | 24 | # link order important here |
diff --git a/drivers/net/acenic.c b/drivers/net/acenic.c index b78a4e5ceeb2..62e660a79387 100644 --- a/drivers/net/acenic.c +++ b/drivers/net/acenic.c | |||
@@ -3128,12 +3128,6 @@ static int __devinit read_eeprom_byte(struct net_device *dev, | |||
3128 | int result = 0; | 3128 | int result = 0; |
3129 | short i; | 3129 | short i; |
3130 | 3130 | ||
3131 | if (!dev) { | ||
3132 | printk(KERN_ERR "No device!\n"); | ||
3133 | result = -ENODEV; | ||
3134 | goto out; | ||
3135 | } | ||
3136 | |||
3137 | /* | 3131 | /* |
3138 | * Don't take interrupts on this CPU will bit banging | 3132 | * Don't take interrupts on this CPU will bit banging |
3139 | * the %#%#@$ I2C device | 3133 | * the %#%#@$ I2C device |
diff --git a/drivers/net/atl1/atl1_hw.h b/drivers/net/atl1/atl1_hw.h index 100c09c66e64..939aa0f53f6e 100644 --- a/drivers/net/atl1/atl1_hw.h +++ b/drivers/net/atl1/atl1_hw.h | |||
@@ -680,11 +680,6 @@ void atl1_check_options(struct atl1_adapter *adapter); | |||
680 | #define AUTONEG_ADVERTISE_10_100_ALL 0x000F /* All 10/100 speeds */ | 680 | #define AUTONEG_ADVERTISE_10_100_ALL 0x000F /* All 10/100 speeds */ |
681 | #define AUTONEG_ADVERTISE_10_ALL 0x0003 /* 10Mbps Full & Half speeds */ | 681 | #define AUTONEG_ADVERTISE_10_ALL 0x0003 /* 10Mbps Full & Half speeds */ |
682 | 682 | ||
683 | /* The size (in bytes) of a ethernet packet */ | ||
684 | #define ENET_HEADER_SIZE 14 | ||
685 | #define MAXIMUM_ETHERNET_FRAME_SIZE 1518 /* with FCS */ | ||
686 | #define MINIMUM_ETHERNET_FRAME_SIZE 64 /* with FCS */ | ||
687 | #define ETHERNET_FCS_SIZE 4 | ||
688 | #define MAX_JUMBO_FRAME_SIZE 0x2800 | 683 | #define MAX_JUMBO_FRAME_SIZE 0x2800 |
689 | 684 | ||
690 | #define PHY_AUTO_NEG_TIME 45 /* 4.5 Seconds */ | 685 | #define PHY_AUTO_NEG_TIME 45 /* 4.5 Seconds */ |
@@ -929,8 +924,8 @@ enum atl1_dma_req_block { | |||
929 | atl1_dma_req_128 = 0, | 924 | atl1_dma_req_128 = 0, |
930 | atl1_dma_req_256 = 1, | 925 | atl1_dma_req_256 = 1, |
931 | atl1_dma_req_512 = 2, | 926 | atl1_dma_req_512 = 2, |
932 | atl1_dam_req_1024 = 3, | 927 | atl1_dma_req_1024 = 3, |
933 | atl1_dam_req_2048 = 4, | 928 | atl1_dma_req_2048 = 4, |
934 | atl1_dma_req_4096 = 5 | 929 | atl1_dma_req_4096 = 5 |
935 | }; | 930 | }; |
936 | 931 | ||
diff --git a/drivers/net/atl1/atl1_main.c b/drivers/net/atl1/atl1_main.c index fd1e156f1747..56f6389a300e 100644 --- a/drivers/net/atl1/atl1_main.c +++ b/drivers/net/atl1/atl1_main.c | |||
@@ -59,6 +59,7 @@ | |||
59 | #include <linux/skbuff.h> | 59 | #include <linux/skbuff.h> |
60 | #include <linux/etherdevice.h> | 60 | #include <linux/etherdevice.h> |
61 | #include <linux/if_vlan.h> | 61 | #include <linux/if_vlan.h> |
62 | #include <linux/if_ether.h> | ||
62 | #include <linux/irqreturn.h> | 63 | #include <linux/irqreturn.h> |
63 | #include <linux/workqueue.h> | 64 | #include <linux/workqueue.h> |
64 | #include <linux/timer.h> | 65 | #include <linux/timer.h> |
@@ -120,8 +121,8 @@ static int __devinit atl1_sw_init(struct atl1_adapter *adapter) | |||
120 | struct atl1_hw *hw = &adapter->hw; | 121 | struct atl1_hw *hw = &adapter->hw; |
121 | struct net_device *netdev = adapter->netdev; | 122 | struct net_device *netdev = adapter->netdev; |
122 | 123 | ||
123 | hw->max_frame_size = netdev->mtu + ENET_HEADER_SIZE + ETHERNET_FCS_SIZE; | 124 | hw->max_frame_size = netdev->mtu + ETH_HLEN + ETH_FCS_LEN; |
124 | hw->min_frame_size = MINIMUM_ETHERNET_FRAME_SIZE; | 125 | hw->min_frame_size = ETH_ZLEN + ETH_FCS_LEN; |
125 | 126 | ||
126 | adapter->wol = 0; | 127 | adapter->wol = 0; |
127 | adapter->rx_buffer_len = (hw->max_frame_size + 7) & ~7; | 128 | adapter->rx_buffer_len = (hw->max_frame_size + 7) & ~7; |
@@ -314,7 +315,7 @@ err_nomem: | |||
314 | return -ENOMEM; | 315 | return -ENOMEM; |
315 | } | 316 | } |
316 | 317 | ||
317 | void atl1_init_ring_ptrs(struct atl1_adapter *adapter) | 318 | static void atl1_init_ring_ptrs(struct atl1_adapter *adapter) |
318 | { | 319 | { |
319 | struct atl1_tpd_ring *tpd_ring = &adapter->tpd_ring; | 320 | struct atl1_tpd_ring *tpd_ring = &adapter->tpd_ring; |
320 | struct atl1_rfd_ring *rfd_ring = &adapter->rfd_ring; | 321 | struct atl1_rfd_ring *rfd_ring = &adapter->rfd_ring; |
@@ -688,9 +689,9 @@ static int atl1_change_mtu(struct net_device *netdev, int new_mtu) | |||
688 | { | 689 | { |
689 | struct atl1_adapter *adapter = netdev_priv(netdev); | 690 | struct atl1_adapter *adapter = netdev_priv(netdev); |
690 | int old_mtu = netdev->mtu; | 691 | int old_mtu = netdev->mtu; |
691 | int max_frame = new_mtu + ENET_HEADER_SIZE + ETHERNET_FCS_SIZE; | 692 | int max_frame = new_mtu + ETH_HLEN + ETH_FCS_LEN; |
692 | 693 | ||
693 | if ((max_frame < MINIMUM_ETHERNET_FRAME_SIZE) || | 694 | if ((max_frame < ETH_ZLEN + ETH_FCS_LEN) || |
694 | (max_frame > MAX_JUMBO_FRAME_SIZE)) { | 695 | (max_frame > MAX_JUMBO_FRAME_SIZE)) { |
695 | dev_warn(&adapter->pdev->dev, "invalid MTU setting\n"); | 696 | dev_warn(&adapter->pdev->dev, "invalid MTU setting\n"); |
696 | return -EINVAL; | 697 | return -EINVAL; |
@@ -908,8 +909,8 @@ static u32 atl1_configure(struct atl1_adapter *adapter) | |||
908 | /* config DMA Engine */ | 909 | /* config DMA Engine */ |
909 | value = ((((u32) hw->dmar_block) & DMA_CTRL_DMAR_BURST_LEN_MASK) | 910 | value = ((((u32) hw->dmar_block) & DMA_CTRL_DMAR_BURST_LEN_MASK) |
910 | << DMA_CTRL_DMAR_BURST_LEN_SHIFT) | | 911 | << DMA_CTRL_DMAR_BURST_LEN_SHIFT) | |
911 | ((((u32) hw->dmaw_block) & DMA_CTRL_DMAR_BURST_LEN_MASK) | 912 | ((((u32) hw->dmaw_block) & DMA_CTRL_DMAW_BURST_LEN_MASK) |
912 | << DMA_CTRL_DMAR_BURST_LEN_SHIFT) | DMA_CTRL_DMAR_EN | | 913 | << DMA_CTRL_DMAW_BURST_LEN_SHIFT) | DMA_CTRL_DMAR_EN | |
913 | DMA_CTRL_DMAW_EN; | 914 | DMA_CTRL_DMAW_EN; |
914 | value |= (u32) hw->dma_ord; | 915 | value |= (u32) hw->dma_ord; |
915 | if (atl1_rcb_128 == hw->rcb_value) | 916 | if (atl1_rcb_128 == hw->rcb_value) |
@@ -917,7 +918,10 @@ static u32 atl1_configure(struct atl1_adapter *adapter) | |||
917 | iowrite32(value, hw->hw_addr + REG_DMA_CTRL); | 918 | iowrite32(value, hw->hw_addr + REG_DMA_CTRL); |
918 | 919 | ||
919 | /* config CMB / SMB */ | 920 | /* config CMB / SMB */ |
920 | value = hw->cmb_rrd | ((u32) hw->cmb_tpd << 16); | 921 | value = (hw->cmb_tpd > adapter->tpd_ring.count) ? |
922 | hw->cmb_tpd : adapter->tpd_ring.count; | ||
923 | value <<= 16; | ||
924 | value |= hw->cmb_rrd; | ||
921 | iowrite32(value, hw->hw_addr + REG_CMB_WRITE_TH); | 925 | iowrite32(value, hw->hw_addr + REG_CMB_WRITE_TH); |
922 | value = hw->cmb_rx_timer | ((u32) hw->cmb_tx_timer << 16); | 926 | value = hw->cmb_rx_timer | ((u32) hw->cmb_tx_timer << 16); |
923 | iowrite32(value, hw->hw_addr + REG_CMB_WRITE_TIMER); | 927 | iowrite32(value, hw->hw_addr + REG_CMB_WRITE_TIMER); |
@@ -1334,7 +1338,7 @@ rrd_ok: | |||
1334 | skb = buffer_info->skb; | 1338 | skb = buffer_info->skb; |
1335 | length = le16_to_cpu(rrd->xsz.xsum_sz.pkt_size); | 1339 | length = le16_to_cpu(rrd->xsz.xsum_sz.pkt_size); |
1336 | 1340 | ||
1337 | skb_put(skb, length - ETHERNET_FCS_SIZE); | 1341 | skb_put(skb, length - ETH_FCS_LEN); |
1338 | 1342 | ||
1339 | /* Receive Checksum Offload */ | 1343 | /* Receive Checksum Offload */ |
1340 | atl1_rx_checksum(adapter, rrd, skb); | 1344 | atl1_rx_checksum(adapter, rrd, skb); |
@@ -1422,7 +1426,7 @@ static void atl1_intr_tx(struct atl1_adapter *adapter) | |||
1422 | netif_wake_queue(adapter->netdev); | 1426 | netif_wake_queue(adapter->netdev); |
1423 | } | 1427 | } |
1424 | 1428 | ||
1425 | static u16 tpd_avail(struct atl1_tpd_ring *tpd_ring) | 1429 | static u16 atl1_tpd_avail(struct atl1_tpd_ring *tpd_ring) |
1426 | { | 1430 | { |
1427 | u16 next_to_clean = atomic_read(&tpd_ring->next_to_clean); | 1431 | u16 next_to_clean = atomic_read(&tpd_ring->next_to_clean); |
1428 | u16 next_to_use = atomic_read(&tpd_ring->next_to_use); | 1432 | u16 next_to_use = atomic_read(&tpd_ring->next_to_use); |
@@ -1453,7 +1457,7 @@ static int atl1_tso(struct atl1_adapter *adapter, struct sk_buff *skb, | |||
1453 | tcp_hdr(skb)->check = ~csum_tcpudp_magic(iph->saddr, | 1457 | tcp_hdr(skb)->check = ~csum_tcpudp_magic(iph->saddr, |
1454 | iph->daddr, 0, IPPROTO_TCP, 0); | 1458 | iph->daddr, 0, IPPROTO_TCP, 0); |
1455 | ipofst = skb_network_offset(skb); | 1459 | ipofst = skb_network_offset(skb); |
1456 | if (ipofst != ENET_HEADER_SIZE) /* 802.3 frame */ | 1460 | if (ipofst != ETH_HLEN) /* 802.3 frame */ |
1457 | tso->tsopl |= 1 << TSO_PARAM_ETHTYPE_SHIFT; | 1461 | tso->tsopl |= 1 << TSO_PARAM_ETHTYPE_SHIFT; |
1458 | 1462 | ||
1459 | tso->tsopl |= (iph->ihl & | 1463 | tso->tsopl |= (iph->ihl & |
@@ -1708,7 +1712,7 @@ static int atl1_xmit_frame(struct sk_buff *skb, struct net_device *netdev) | |||
1708 | return NETDEV_TX_LOCKED; | 1712 | return NETDEV_TX_LOCKED; |
1709 | } | 1713 | } |
1710 | 1714 | ||
1711 | if (tpd_avail(&adapter->tpd_ring) < count) { | 1715 | if (atl1_tpd_avail(&adapter->tpd_ring) < count) { |
1712 | /* not enough descriptors */ | 1716 | /* not enough descriptors */ |
1713 | netif_stop_queue(netdev); | 1717 | netif_stop_queue(netdev); |
1714 | spin_unlock_irqrestore(&adapter->lock, flags); | 1718 | spin_unlock_irqrestore(&adapter->lock, flags); |
diff --git a/drivers/net/ax88796.c b/drivers/net/ax88796.c index 1d882360b34d..e43e8047b90e 100644 --- a/drivers/net/ax88796.c +++ b/drivers/net/ax88796.c | |||
@@ -819,7 +819,7 @@ static int ax_probe(struct platform_device *pdev) | |||
819 | } | 819 | } |
820 | 820 | ||
821 | ei_status.mem = ioremap(res->start, size); | 821 | ei_status.mem = ioremap(res->start, size); |
822 | dev->base_addr = (long)ei_status.mem; | 822 | dev->base_addr = (unsigned long)ei_status.mem; |
823 | 823 | ||
824 | if (ei_status.mem == NULL) { | 824 | if (ei_status.mem == NULL) { |
825 | dev_err(&pdev->dev, "Cannot ioremap area (%08zx,%08zx)\n", | 825 | dev_err(&pdev->dev, "Cannot ioremap area (%08zx,%08zx)\n", |
diff --git a/drivers/net/cxgb3/cxgb3_offload.c b/drivers/net/cxgb3/cxgb3_offload.c index ebcf35e4cf5b..e620ed4c3ff0 100644 --- a/drivers/net/cxgb3/cxgb3_offload.c +++ b/drivers/net/cxgb3/cxgb3_offload.c | |||
@@ -699,7 +699,7 @@ static int do_cr(struct t3cdev *dev, struct sk_buff *skb) | |||
699 | * the buffer. | 699 | * the buffer. |
700 | */ | 700 | */ |
701 | static struct sk_buff *cxgb3_get_cpl_reply_skb(struct sk_buff *skb, size_t len, | 701 | static struct sk_buff *cxgb3_get_cpl_reply_skb(struct sk_buff *skb, size_t len, |
702 | int gfp) | 702 | gfp_t gfp) |
703 | { | 703 | { |
704 | if (likely(!skb_cloned(skb))) { | 704 | if (likely(!skb_cloned(skb))) { |
705 | BUG_ON(skb->len < len); | 705 | BUG_ON(skb->len < len); |
diff --git a/drivers/net/defxx.c b/drivers/net/defxx.c index 7df23dc28190..9c8e3f9f5e58 100644 --- a/drivers/net/defxx.c +++ b/drivers/net/defxx.c | |||
@@ -200,6 +200,7 @@ | |||
200 | 200 | ||
201 | /* Include files */ | 201 | /* Include files */ |
202 | #include <linux/bitops.h> | 202 | #include <linux/bitops.h> |
203 | #include <linux/compiler.h> | ||
203 | #include <linux/delay.h> | 204 | #include <linux/delay.h> |
204 | #include <linux/dma-mapping.h> | 205 | #include <linux/dma-mapping.h> |
205 | #include <linux/eisa.h> | 206 | #include <linux/eisa.h> |
@@ -240,8 +241,6 @@ static char version[] __devinitdata = | |||
240 | */ | 241 | */ |
241 | #define NEW_SKB_SIZE (PI_RCV_DATA_K_SIZE_MAX+128) | 242 | #define NEW_SKB_SIZE (PI_RCV_DATA_K_SIZE_MAX+128) |
242 | 243 | ||
243 | #define __unused __attribute__ ((unused)) | ||
244 | |||
245 | #ifdef CONFIG_PCI | 244 | #ifdef CONFIG_PCI |
246 | #define DFX_BUS_PCI(dev) (dev->bus == &pci_bus_type) | 245 | #define DFX_BUS_PCI(dev) (dev->bus == &pci_bus_type) |
247 | #else | 246 | #else |
@@ -375,7 +374,7 @@ static inline void dfx_outl(DFX_board_t *bp, int offset, u32 data) | |||
375 | 374 | ||
376 | static void dfx_port_write_long(DFX_board_t *bp, int offset, u32 data) | 375 | static void dfx_port_write_long(DFX_board_t *bp, int offset, u32 data) |
377 | { | 376 | { |
378 | struct device __unused *bdev = bp->bus_dev; | 377 | struct device __maybe_unused *bdev = bp->bus_dev; |
379 | int dfx_bus_tc = DFX_BUS_TC(bdev); | 378 | int dfx_bus_tc = DFX_BUS_TC(bdev); |
380 | int dfx_use_mmio = DFX_MMIO || dfx_bus_tc; | 379 | int dfx_use_mmio = DFX_MMIO || dfx_bus_tc; |
381 | 380 | ||
@@ -399,7 +398,7 @@ static inline void dfx_inl(DFX_board_t *bp, int offset, u32 *data) | |||
399 | 398 | ||
400 | static void dfx_port_read_long(DFX_board_t *bp, int offset, u32 *data) | 399 | static void dfx_port_read_long(DFX_board_t *bp, int offset, u32 *data) |
401 | { | 400 | { |
402 | struct device __unused *bdev = bp->bus_dev; | 401 | struct device __maybe_unused *bdev = bp->bus_dev; |
403 | int dfx_bus_tc = DFX_BUS_TC(bdev); | 402 | int dfx_bus_tc = DFX_BUS_TC(bdev); |
404 | int dfx_use_mmio = DFX_MMIO || dfx_bus_tc; | 403 | int dfx_use_mmio = DFX_MMIO || dfx_bus_tc; |
405 | 404 | ||
@@ -866,7 +865,7 @@ static void __devinit dfx_bus_uninit(struct net_device *dev) | |||
866 | 865 | ||
867 | static void __devinit dfx_bus_config_check(DFX_board_t *bp) | 866 | static void __devinit dfx_bus_config_check(DFX_board_t *bp) |
868 | { | 867 | { |
869 | struct device __unused *bdev = bp->bus_dev; | 868 | struct device __maybe_unused *bdev = bp->bus_dev; |
870 | int dfx_bus_eisa = DFX_BUS_EISA(bdev); | 869 | int dfx_bus_eisa = DFX_BUS_EISA(bdev); |
871 | int status; /* return code from adapter port control call */ | 870 | int status; /* return code from adapter port control call */ |
872 | u32 host_data; /* LW data returned from port control call */ | 871 | u32 host_data; /* LW data returned from port control call */ |
@@ -3624,8 +3623,8 @@ static void __devexit dfx_unregister(struct device *bdev) | |||
3624 | } | 3623 | } |
3625 | 3624 | ||
3626 | 3625 | ||
3627 | static int __devinit __unused dfx_dev_register(struct device *); | 3626 | static int __devinit __maybe_unused dfx_dev_register(struct device *); |
3628 | static int __devexit __unused dfx_dev_unregister(struct device *); | 3627 | static int __devexit __maybe_unused dfx_dev_unregister(struct device *); |
3629 | 3628 | ||
3630 | #ifdef CONFIG_PCI | 3629 | #ifdef CONFIG_PCI |
3631 | static int __devinit dfx_pci_register(struct pci_dev *, | 3630 | static int __devinit dfx_pci_register(struct pci_dev *, |
@@ -3699,7 +3698,7 @@ static struct tc_driver dfx_tc_driver = { | |||
3699 | }; | 3698 | }; |
3700 | #endif /* CONFIG_TC */ | 3699 | #endif /* CONFIG_TC */ |
3701 | 3700 | ||
3702 | static int __devinit __unused dfx_dev_register(struct device *dev) | 3701 | static int __devinit __maybe_unused dfx_dev_register(struct device *dev) |
3703 | { | 3702 | { |
3704 | int status; | 3703 | int status; |
3705 | 3704 | ||
@@ -3709,7 +3708,7 @@ static int __devinit __unused dfx_dev_register(struct device *dev) | |||
3709 | return status; | 3708 | return status; |
3710 | } | 3709 | } |
3711 | 3710 | ||
3712 | static int __devexit __unused dfx_dev_unregister(struct device *dev) | 3711 | static int __devexit __maybe_unused dfx_dev_unregister(struct device *dev) |
3713 | { | 3712 | { |
3714 | put_device(dev); | 3713 | put_device(dev); |
3715 | dfx_unregister(dev); | 3714 | dfx_unregister(dev); |
diff --git a/drivers/net/ehea/ehea.h b/drivers/net/ehea/ehea.h index 489c8b260dd8..8ee2c2c86b42 100644 --- a/drivers/net/ehea/ehea.h +++ b/drivers/net/ehea/ehea.h | |||
@@ -39,7 +39,7 @@ | |||
39 | #include <asm/io.h> | 39 | #include <asm/io.h> |
40 | 40 | ||
41 | #define DRV_NAME "ehea" | 41 | #define DRV_NAME "ehea" |
42 | #define DRV_VERSION "EHEA_0071" | 42 | #define DRV_VERSION "EHEA_0072" |
43 | 43 | ||
44 | /* eHEA capability flags */ | 44 | /* eHEA capability flags */ |
45 | #define DLPAR_PORT_ADD_REM 1 | 45 | #define DLPAR_PORT_ADD_REM 1 |
diff --git a/drivers/net/ehea/ehea_main.c b/drivers/net/ehea/ehea_main.c index 4c70a9301c1b..58702f54c3fb 100644 --- a/drivers/net/ehea/ehea_main.c +++ b/drivers/net/ehea/ehea_main.c | |||
@@ -589,6 +589,23 @@ static int ehea_poll(struct net_device *dev, int *budget) | |||
589 | return 1; | 589 | return 1; |
590 | } | 590 | } |
591 | 591 | ||
592 | #ifdef CONFIG_NET_POLL_CONTROLLER | ||
593 | static void ehea_netpoll(struct net_device *dev) | ||
594 | { | ||
595 | struct ehea_port *port = netdev_priv(dev); | ||
596 | |||
597 | netif_rx_schedule(port->port_res[0].d_netdev); | ||
598 | } | ||
599 | #endif | ||
600 | |||
601 | static int ehea_poll_firstqueue(struct net_device *dev, int *budget) | ||
602 | { | ||
603 | struct ehea_port *port = netdev_priv(dev); | ||
604 | struct net_device *d_dev = port->port_res[0].d_netdev; | ||
605 | |||
606 | return ehea_poll(d_dev, budget); | ||
607 | } | ||
608 | |||
592 | static irqreturn_t ehea_recv_irq_handler(int irq, void *param) | 609 | static irqreturn_t ehea_recv_irq_handler(int irq, void *param) |
593 | { | 610 | { |
594 | struct ehea_port_res *pr = param; | 611 | struct ehea_port_res *pr = param; |
@@ -2626,7 +2643,10 @@ struct ehea_port *ehea_setup_single_port(struct ehea_adapter *adapter, | |||
2626 | memcpy(dev->dev_addr, &port->mac_addr, ETH_ALEN); | 2643 | memcpy(dev->dev_addr, &port->mac_addr, ETH_ALEN); |
2627 | 2644 | ||
2628 | dev->open = ehea_open; | 2645 | dev->open = ehea_open; |
2629 | dev->poll = ehea_poll; | 2646 | dev->poll = ehea_poll_firstqueue; |
2647 | #ifdef CONFIG_NET_POLL_CONTROLLER | ||
2648 | dev->poll_controller = ehea_netpoll; | ||
2649 | #endif | ||
2630 | dev->weight = 64; | 2650 | dev->weight = 64; |
2631 | dev->stop = ehea_stop; | 2651 | dev->stop = ehea_stop; |
2632 | dev->hard_start_xmit = ehea_start_xmit; | 2652 | dev->hard_start_xmit = ehea_start_xmit; |
diff --git a/drivers/net/forcedeth.c b/drivers/net/forcedeth.c index 6d1d50a19783..661c747389e4 100644 --- a/drivers/net/forcedeth.c +++ b/drivers/net/forcedeth.c | |||
@@ -5546,6 +5546,22 @@ static struct pci_device_id pci_tbl[] = { | |||
5546 | PCI_DEVICE(PCI_VENDOR_ID_NVIDIA, PCI_DEVICE_ID_NVIDIA_NVENET_27), | 5546 | PCI_DEVICE(PCI_VENDOR_ID_NVIDIA, PCI_DEVICE_ID_NVIDIA_NVENET_27), |
5547 | .driver_data = DEV_NEED_TIMERIRQ|DEV_NEED_LINKTIMER|DEV_HAS_HIGH_DMA|DEV_HAS_POWER_CNTRL|DEV_HAS_MSI|DEV_HAS_PAUSEFRAME_TX|DEV_HAS_STATISTICS_V2|DEV_HAS_TEST_EXTENDED|DEV_HAS_MGMT_UNIT, | 5547 | .driver_data = DEV_NEED_TIMERIRQ|DEV_NEED_LINKTIMER|DEV_HAS_HIGH_DMA|DEV_HAS_POWER_CNTRL|DEV_HAS_MSI|DEV_HAS_PAUSEFRAME_TX|DEV_HAS_STATISTICS_V2|DEV_HAS_TEST_EXTENDED|DEV_HAS_MGMT_UNIT, |
5548 | }, | 5548 | }, |
5549 | { /* MCP73 Ethernet Controller */ | ||
5550 | PCI_DEVICE(PCI_VENDOR_ID_NVIDIA, PCI_DEVICE_ID_NVIDIA_NVENET_28), | ||
5551 | .driver_data = DEV_NEED_TIMERIRQ|DEV_NEED_LINKTIMER|DEV_HAS_HIGH_DMA|DEV_HAS_POWER_CNTRL|DEV_HAS_MSI|DEV_HAS_PAUSEFRAME_TX|DEV_HAS_STATISTICS_V2|DEV_HAS_TEST_EXTENDED|DEV_HAS_MGMT_UNIT, | ||
5552 | }, | ||
5553 | { /* MCP73 Ethernet Controller */ | ||
5554 | PCI_DEVICE(PCI_VENDOR_ID_NVIDIA, PCI_DEVICE_ID_NVIDIA_NVENET_29), | ||
5555 | .driver_data = DEV_NEED_TIMERIRQ|DEV_NEED_LINKTIMER|DEV_HAS_HIGH_DMA|DEV_HAS_POWER_CNTRL|DEV_HAS_MSI|DEV_HAS_PAUSEFRAME_TX|DEV_HAS_STATISTICS_V2|DEV_HAS_TEST_EXTENDED|DEV_HAS_MGMT_UNIT, | ||
5556 | }, | ||
5557 | { /* MCP73 Ethernet Controller */ | ||
5558 | PCI_DEVICE(PCI_VENDOR_ID_NVIDIA, PCI_DEVICE_ID_NVIDIA_NVENET_30), | ||
5559 | .driver_data = DEV_NEED_TIMERIRQ|DEV_NEED_LINKTIMER|DEV_HAS_HIGH_DMA|DEV_HAS_POWER_CNTRL|DEV_HAS_MSI|DEV_HAS_PAUSEFRAME_TX|DEV_HAS_STATISTICS_V2|DEV_HAS_TEST_EXTENDED|DEV_HAS_MGMT_UNIT, | ||
5560 | }, | ||
5561 | { /* MCP73 Ethernet Controller */ | ||
5562 | PCI_DEVICE(PCI_VENDOR_ID_NVIDIA, PCI_DEVICE_ID_NVIDIA_NVENET_31), | ||
5563 | .driver_data = DEV_NEED_TIMERIRQ|DEV_NEED_LINKTIMER|DEV_HAS_HIGH_DMA|DEV_HAS_POWER_CNTRL|DEV_HAS_MSI|DEV_HAS_PAUSEFRAME_TX|DEV_HAS_STATISTICS_V2|DEV_HAS_TEST_EXTENDED|DEV_HAS_MGMT_UNIT, | ||
5564 | }, | ||
5549 | {0,}, | 5565 | {0,}, |
5550 | }; | 5566 | }; |
5551 | 5567 | ||
diff --git a/drivers/net/lguest_net.c b/drivers/net/lguest_net.c index 112778652f7d..cab57911a80e 100644 --- a/drivers/net/lguest_net.c +++ b/drivers/net/lguest_net.c | |||
@@ -1,6 +1,13 @@ | |||
1 | /* A simple network driver for lguest. | 1 | /*D:500 |
2 | * The Guest network driver. | ||
2 | * | 3 | * |
3 | * Copyright 2006 Rusty Russell <rusty@rustcorp.com.au> IBM Corporation | 4 | * This is very simple a virtual network driver, and our last Guest driver. |
5 | * The only trick is that it can talk directly to multiple other recipients | ||
6 | * (ie. other Guests on the same network). It can also be used with only the | ||
7 | * Host on the network. | ||
8 | :*/ | ||
9 | |||
10 | /* Copyright 2006 Rusty Russell <rusty@rustcorp.com.au> IBM Corporation | ||
4 | * | 11 | * |
5 | * This program is free software; you can redistribute it and/or modify | 12 | * This program is free software; you can redistribute it and/or modify |
6 | * it under the terms of the GNU General Public License as published by | 13 | * it under the terms of the GNU General Public License as published by |
@@ -28,23 +35,47 @@ | |||
28 | #define MAX_LANS 4 | 35 | #define MAX_LANS 4 |
29 | #define NUM_SKBS 8 | 36 | #define NUM_SKBS 8 |
30 | 37 | ||
38 | /*M:011 Network code master Jeff Garzik points out numerous shortcomings in | ||
39 | * this driver if it aspires to greatness. | ||
40 | * | ||
41 | * Firstly, it doesn't use "NAPI": the networking's New API, and is poorer for | ||
42 | * it. As he says "NAPI means system-wide load leveling, across multiple | ||
43 | * network interfaces. Lack of NAPI can mean competition at higher loads." | ||
44 | * | ||
45 | * He also points out that we don't implement set_mac_address, so users cannot | ||
46 | * change the devices hardware address. When I asked why one would want to: | ||
47 | * "Bonding, and situations where you /do/ want the MAC address to "leak" out | ||
48 | * of the host onto the wider net." | ||
49 | * | ||
50 | * Finally, he would like module unloading: "It is not unrealistic to think of | ||
51 | * [un|re|]loading the net support module in an lguest guest. And, adding | ||
52 | * module support makes the programmer more responsible, because they now have | ||
53 | * to learn to clean up after themselves. Any driver that cannot clean up | ||
54 | * after itself is an incomplete driver in my book." | ||
55 | :*/ | ||
56 | |||
57 | /*D:530 The "struct lguestnet_info" contains all the information we need to | ||
58 | * know about the network device. */ | ||
31 | struct lguestnet_info | 59 | struct lguestnet_info |
32 | { | 60 | { |
33 | /* The shared page(s). */ | 61 | /* The mapped device page(s) (an array of "struct lguest_net"). */ |
34 | struct lguest_net *peer; | 62 | struct lguest_net *peer; |
63 | /* The physical address of the device page(s) */ | ||
35 | unsigned long peer_phys; | 64 | unsigned long peer_phys; |
65 | /* The size of the device page(s). */ | ||
36 | unsigned long mapsize; | 66 | unsigned long mapsize; |
37 | 67 | ||
38 | /* The lguest_device I come from */ | 68 | /* The lguest_device I come from */ |
39 | struct lguest_device *lgdev; | 69 | struct lguest_device *lgdev; |
40 | 70 | ||
41 | /* My peerid. */ | 71 | /* My peerid (ie. my slot in the array). */ |
42 | unsigned int me; | 72 | unsigned int me; |
43 | 73 | ||
44 | /* Receive queue. */ | 74 | /* Receive queue: the network packets waiting to be filled. */ |
45 | struct sk_buff *skb[NUM_SKBS]; | 75 | struct sk_buff *skb[NUM_SKBS]; |
46 | struct lguest_dma dma[NUM_SKBS]; | 76 | struct lguest_dma dma[NUM_SKBS]; |
47 | }; | 77 | }; |
78 | /*:*/ | ||
48 | 79 | ||
49 | /* How many bytes left in this page. */ | 80 | /* How many bytes left in this page. */ |
50 | static unsigned int rest_of_page(void *data) | 81 | static unsigned int rest_of_page(void *data) |
@@ -52,39 +83,82 @@ static unsigned int rest_of_page(void *data) | |||
52 | return PAGE_SIZE - ((unsigned long)data % PAGE_SIZE); | 83 | return PAGE_SIZE - ((unsigned long)data % PAGE_SIZE); |
53 | } | 84 | } |
54 | 85 | ||
55 | /* Simple convention: offset 4 * peernum. */ | 86 | /*D:570 Each peer (ie. Guest or Host) on the network binds their receive |
87 | * buffers to a different key: we simply use the physical address of the | ||
88 | * device's memory page plus the peer number. The Host insists that all keys | ||
89 | * be a multiple of 4, so we multiply the peer number by 4. */ | ||
56 | static unsigned long peer_key(struct lguestnet_info *info, unsigned peernum) | 90 | static unsigned long peer_key(struct lguestnet_info *info, unsigned peernum) |
57 | { | 91 | { |
58 | return info->peer_phys + 4 * peernum; | 92 | return info->peer_phys + 4 * peernum; |
59 | } | 93 | } |
60 | 94 | ||
95 | /* This is the routine which sets up a "struct lguest_dma" to point to a | ||
96 | * network packet, similar to req_to_dma() in lguest_blk.c. The structure of a | ||
97 | * "struct sk_buff" has grown complex over the years: it consists of a "head" | ||
98 | * linear section pointed to by "skb->data", and possibly an array of | ||
99 | * "fragments" in the case of a non-linear packet. | ||
100 | * | ||
101 | * Our receive buffers don't use fragments at all but outgoing skbs might, so | ||
102 | * we handle it. */ | ||
61 | static void skb_to_dma(const struct sk_buff *skb, unsigned int headlen, | 103 | static void skb_to_dma(const struct sk_buff *skb, unsigned int headlen, |
62 | struct lguest_dma *dma) | 104 | struct lguest_dma *dma) |
63 | { | 105 | { |
64 | unsigned int i, seg; | 106 | unsigned int i, seg; |
65 | 107 | ||
108 | /* First, we put the linear region into the "struct lguest_dma". Each | ||
109 | * entry can't go over a page boundary, so even though all our packets | ||
110 | * are 1514 bytes or less, we might need to use two entries here: */ | ||
66 | for (i = seg = 0; i < headlen; seg++, i += rest_of_page(skb->data+i)) { | 111 | for (i = seg = 0; i < headlen; seg++, i += rest_of_page(skb->data+i)) { |
67 | dma->addr[seg] = virt_to_phys(skb->data + i); | 112 | dma->addr[seg] = virt_to_phys(skb->data + i); |
68 | dma->len[seg] = min((unsigned)(headlen - i), | 113 | dma->len[seg] = min((unsigned)(headlen - i), |
69 | rest_of_page(skb->data + i)); | 114 | rest_of_page(skb->data + i)); |
70 | } | 115 | } |
116 | |||
117 | /* Now we handle the fragments: at least they're guaranteed not to go | ||
118 | * over a page. skb_shinfo(skb) returns a pointer to the structure | ||
119 | * which tells us about the number of fragments and the fragment | ||
120 | * array. */ | ||
71 | for (i = 0; i < skb_shinfo(skb)->nr_frags; i++, seg++) { | 121 | for (i = 0; i < skb_shinfo(skb)->nr_frags; i++, seg++) { |
72 | const skb_frag_t *f = &skb_shinfo(skb)->frags[i]; | 122 | const skb_frag_t *f = &skb_shinfo(skb)->frags[i]; |
73 | /* Should not happen with MTU less than 64k - 2 * PAGE_SIZE. */ | 123 | /* Should not happen with MTU less than 64k - 2 * PAGE_SIZE. */ |
74 | if (seg == LGUEST_MAX_DMA_SECTIONS) { | 124 | if (seg == LGUEST_MAX_DMA_SECTIONS) { |
125 | /* We will end up sending a truncated packet should | ||
126 | * this ever happen. Plus, a cool log message! */ | ||
75 | printk("Woah dude! Megapacket!\n"); | 127 | printk("Woah dude! Megapacket!\n"); |
76 | break; | 128 | break; |
77 | } | 129 | } |
78 | dma->addr[seg] = page_to_phys(f->page) + f->page_offset; | 130 | dma->addr[seg] = page_to_phys(f->page) + f->page_offset; |
79 | dma->len[seg] = f->size; | 131 | dma->len[seg] = f->size; |
80 | } | 132 | } |
133 | |||
134 | /* If after all that we didn't use the entire "struct lguest_dma" | ||
135 | * array, we terminate it with a 0 length. */ | ||
81 | if (seg < LGUEST_MAX_DMA_SECTIONS) | 136 | if (seg < LGUEST_MAX_DMA_SECTIONS) |
82 | dma->len[seg] = 0; | 137 | dma->len[seg] = 0; |
83 | } | 138 | } |
84 | 139 | ||
85 | /* We overload multicast bit to show promiscuous mode. */ | 140 | /* |
141 | * Packet transmission. | ||
142 | * | ||
143 | * Our packet transmission is a little unusual. A real network card would just | ||
144 | * send out the packet and leave the receivers to decide if they're interested. | ||
145 | * Instead, we look through the network device memory page and see if any of | ||
146 | * the ethernet addresses match the packet destination, and if so we send it to | ||
147 | * that Guest. | ||
148 | * | ||
149 | * This is made a little more complicated in two cases. The first case is | ||
150 | * broadcast packets: for that we send the packet to all Guests on the network, | ||
151 | * one at a time. The second case is "promiscuous" mode, where a Guest wants | ||
152 | * to see all the packets on the network. We need a way for the Guest to tell | ||
153 | * us it wants to see all packets, so it sets the "multicast" bit on its | ||
154 | * published MAC address, which is never valid in a real ethernet address. | ||
155 | */ | ||
86 | #define PROMISC_BIT 0x01 | 156 | #define PROMISC_BIT 0x01 |
87 | 157 | ||
158 | /* This is the callback which is summoned whenever the network device's | ||
159 | * multicast or promiscuous state changes. If the card is in promiscuous mode, | ||
160 | * we advertise that in our ethernet address in the device's memory. We do the | ||
161 | * same if Linux wants any or all multicast traffic. */ | ||
88 | static void lguestnet_set_multicast(struct net_device *dev) | 162 | static void lguestnet_set_multicast(struct net_device *dev) |
89 | { | 163 | { |
90 | struct lguestnet_info *info = netdev_priv(dev); | 164 | struct lguestnet_info *info = netdev_priv(dev); |
@@ -95,11 +169,14 @@ static void lguestnet_set_multicast(struct net_device *dev) | |||
95 | info->peer[info->me].mac[0] &= ~PROMISC_BIT; | 169 | info->peer[info->me].mac[0] &= ~PROMISC_BIT; |
96 | } | 170 | } |
97 | 171 | ||
172 | /* A simple test function to see if a peer wants to see all packets.*/ | ||
98 | static int promisc(struct lguestnet_info *info, unsigned int peer) | 173 | static int promisc(struct lguestnet_info *info, unsigned int peer) |
99 | { | 174 | { |
100 | return info->peer[peer].mac[0] & PROMISC_BIT; | 175 | return info->peer[peer].mac[0] & PROMISC_BIT; |
101 | } | 176 | } |
102 | 177 | ||
178 | /* Another simple function to see if a peer's advertised ethernet address | ||
179 | * matches a packet's destination ethernet address. */ | ||
103 | static int mac_eq(const unsigned char mac[ETH_ALEN], | 180 | static int mac_eq(const unsigned char mac[ETH_ALEN], |
104 | struct lguestnet_info *info, unsigned int peer) | 181 | struct lguestnet_info *info, unsigned int peer) |
105 | { | 182 | { |
@@ -109,6 +186,8 @@ static int mac_eq(const unsigned char mac[ETH_ALEN], | |||
109 | return memcmp(mac+1, info->peer[peer].mac+1, ETH_ALEN-1) == 0; | 186 | return memcmp(mac+1, info->peer[peer].mac+1, ETH_ALEN-1) == 0; |
110 | } | 187 | } |
111 | 188 | ||
189 | /* This is the function which actually sends a packet once we've decided a | ||
190 | * peer wants it: */ | ||
112 | static void transfer_packet(struct net_device *dev, | 191 | static void transfer_packet(struct net_device *dev, |
113 | struct sk_buff *skb, | 192 | struct sk_buff *skb, |
114 | unsigned int peernum) | 193 | unsigned int peernum) |
@@ -116,76 +195,134 @@ static void transfer_packet(struct net_device *dev, | |||
116 | struct lguestnet_info *info = netdev_priv(dev); | 195 | struct lguestnet_info *info = netdev_priv(dev); |
117 | struct lguest_dma dma; | 196 | struct lguest_dma dma; |
118 | 197 | ||
198 | /* We use our handy "struct lguest_dma" packing function to prepare | ||
199 | * the skb for sending. */ | ||
119 | skb_to_dma(skb, skb_headlen(skb), &dma); | 200 | skb_to_dma(skb, skb_headlen(skb), &dma); |
120 | pr_debug("xfer length %04x (%u)\n", htons(skb->len), skb->len); | 201 | pr_debug("xfer length %04x (%u)\n", htons(skb->len), skb->len); |
121 | 202 | ||
203 | /* This is the actual send call which copies the packet. */ | ||
122 | lguest_send_dma(peer_key(info, peernum), &dma); | 204 | lguest_send_dma(peer_key(info, peernum), &dma); |
205 | |||
206 | /* Check that the entire packet was transmitted. If not, it could mean | ||
207 | * that the other Guest registered a short receive buffer, but this | ||
208 | * driver should never do that. More likely, the peer is dead. */ | ||
123 | if (dma.used_len != skb->len) { | 209 | if (dma.used_len != skb->len) { |
124 | dev->stats.tx_carrier_errors++; | 210 | dev->stats.tx_carrier_errors++; |
125 | pr_debug("Bad xfer to peer %i: %i of %i (dma %p/%i)\n", | 211 | pr_debug("Bad xfer to peer %i: %i of %i (dma %p/%i)\n", |
126 | peernum, dma.used_len, skb->len, | 212 | peernum, dma.used_len, skb->len, |
127 | (void *)dma.addr[0], dma.len[0]); | 213 | (void *)dma.addr[0], dma.len[0]); |
128 | } else { | 214 | } else { |
215 | /* On success we update the stats. */ | ||
129 | dev->stats.tx_bytes += skb->len; | 216 | dev->stats.tx_bytes += skb->len; |
130 | dev->stats.tx_packets++; | 217 | dev->stats.tx_packets++; |
131 | } | 218 | } |
132 | } | 219 | } |
133 | 220 | ||
221 | /* Another helper function to tell is if a slot in the device memory is unused. | ||
222 | * Since we always set the Local Assignment bit in the ethernet address, the | ||
223 | * first byte can never be 0. */ | ||
134 | static int unused_peer(const struct lguest_net peer[], unsigned int num) | 224 | static int unused_peer(const struct lguest_net peer[], unsigned int num) |
135 | { | 225 | { |
136 | return peer[num].mac[0] == 0; | 226 | return peer[num].mac[0] == 0; |
137 | } | 227 | } |
138 | 228 | ||
229 | /* Finally, here is the routine which handles an outgoing packet. It's called | ||
230 | * "start_xmit" for traditional reasons. */ | ||
139 | static int lguestnet_start_xmit(struct sk_buff *skb, struct net_device *dev) | 231 | static int lguestnet_start_xmit(struct sk_buff *skb, struct net_device *dev) |
140 | { | 232 | { |
141 | unsigned int i; | 233 | unsigned int i; |
142 | int broadcast; | 234 | int broadcast; |
143 | struct lguestnet_info *info = netdev_priv(dev); | 235 | struct lguestnet_info *info = netdev_priv(dev); |
236 | /* Extract the destination ethernet address from the packet. */ | ||
144 | const unsigned char *dest = ((struct ethhdr *)skb->data)->h_dest; | 237 | const unsigned char *dest = ((struct ethhdr *)skb->data)->h_dest; |
145 | 238 | ||
146 | pr_debug("%s: xmit %02x:%02x:%02x:%02x:%02x:%02x\n", | 239 | pr_debug("%s: xmit %02x:%02x:%02x:%02x:%02x:%02x\n", |
147 | dev->name, dest[0],dest[1],dest[2],dest[3],dest[4],dest[5]); | 240 | dev->name, dest[0],dest[1],dest[2],dest[3],dest[4],dest[5]); |
148 | 241 | ||
242 | /* If it's a multicast packet, we broadcast to everyone. That's not | ||
243 | * very efficient, but there are very few applications which actually | ||
244 | * use multicast, which is a shame really. | ||
245 | * | ||
246 | * As etherdevice.h points out: "By definition the broadcast address is | ||
247 | * also a multicast address." So we don't have to test for broadcast | ||
248 | * packets separately. */ | ||
149 | broadcast = is_multicast_ether_addr(dest); | 249 | broadcast = is_multicast_ether_addr(dest); |
250 | |||
251 | /* Look through all the published ethernet addresses to see if we | ||
252 | * should send this packet. */ | ||
150 | for (i = 0; i < info->mapsize/sizeof(struct lguest_net); i++) { | 253 | for (i = 0; i < info->mapsize/sizeof(struct lguest_net); i++) { |
254 | /* We don't send to ourselves (we actually can't SEND_DMA to | ||
255 | * ourselves anyway), and don't send to unused slots.*/ | ||
151 | if (i == info->me || unused_peer(info->peer, i)) | 256 | if (i == info->me || unused_peer(info->peer, i)) |
152 | continue; | 257 | continue; |
153 | 258 | ||
259 | /* If it's broadcast we send it. If they want every packet we | ||
260 | * send it. If the destination matches their address we send | ||
261 | * it. Otherwise we go to the next peer. */ | ||
154 | if (!broadcast && !promisc(info, i) && !mac_eq(dest, info, i)) | 262 | if (!broadcast && !promisc(info, i) && !mac_eq(dest, info, i)) |
155 | continue; | 263 | continue; |
156 | 264 | ||
157 | pr_debug("lguestnet %s: sending from %i to %i\n", | 265 | pr_debug("lguestnet %s: sending from %i to %i\n", |
158 | dev->name, info->me, i); | 266 | dev->name, info->me, i); |
267 | /* Our routine which actually does the transfer. */ | ||
159 | transfer_packet(dev, skb, i); | 268 | transfer_packet(dev, skb, i); |
160 | } | 269 | } |
270 | |||
271 | /* An xmit routine is expected to dispose of the packet, so we do. */ | ||
161 | dev_kfree_skb(skb); | 272 | dev_kfree_skb(skb); |
273 | |||
274 | /* As per kernel convention, 0 means success. This is why I love | ||
275 | * networking: even if we never sent to anyone, that's still | ||
276 | * success! */ | ||
162 | return 0; | 277 | return 0; |
163 | } | 278 | } |
164 | 279 | ||
165 | /* Find a new skb to put in this slot in shared mem. */ | 280 | /*D:560 |
281 | * Packet receiving. | ||
282 | * | ||
283 | * First, here's a helper routine which fills one of our array of receive | ||
284 | * buffers: */ | ||
166 | static int fill_slot(struct net_device *dev, unsigned int slot) | 285 | static int fill_slot(struct net_device *dev, unsigned int slot) |
167 | { | 286 | { |
168 | struct lguestnet_info *info = netdev_priv(dev); | 287 | struct lguestnet_info *info = netdev_priv(dev); |
169 | /* Try to create and register a new one. */ | 288 | |
289 | /* We can receive ETH_DATA_LEN (1500) byte packets, plus a standard | ||
290 | * ethernet header of ETH_HLEN (14) bytes. */ | ||
170 | info->skb[slot] = netdev_alloc_skb(dev, ETH_HLEN + ETH_DATA_LEN); | 291 | info->skb[slot] = netdev_alloc_skb(dev, ETH_HLEN + ETH_DATA_LEN); |
171 | if (!info->skb[slot]) { | 292 | if (!info->skb[slot]) { |
172 | printk("%s: could not fill slot %i\n", dev->name, slot); | 293 | printk("%s: could not fill slot %i\n", dev->name, slot); |
173 | return -ENOMEM; | 294 | return -ENOMEM; |
174 | } | 295 | } |
175 | 296 | ||
297 | /* skb_to_dma() is a helper which sets up the "struct lguest_dma" to | ||
298 | * point to the data in the skb: we also use it for sending out a | ||
299 | * packet. */ | ||
176 | skb_to_dma(info->skb[slot], ETH_HLEN + ETH_DATA_LEN, &info->dma[slot]); | 300 | skb_to_dma(info->skb[slot], ETH_HLEN + ETH_DATA_LEN, &info->dma[slot]); |
301 | |||
302 | /* This is a Write Memory Barrier: it ensures that the entry in the | ||
303 | * receive buffer array is written *before* we set the "used_len" entry | ||
304 | * to 0. If the Host were looking at the receive buffer array from a | ||
305 | * different CPU, it could potentially see "used_len = 0" and not see | ||
306 | * the updated receive buffer information. This would be a horribly | ||
307 | * nasty bug, so make sure the compiler and CPU know this has to happen | ||
308 | * first. */ | ||
177 | wmb(); | 309 | wmb(); |
178 | /* Now we tell hypervisor it can use the slot. */ | 310 | /* Writing 0 to "used_len" tells the Host it can use this receive |
311 | * buffer now. */ | ||
179 | info->dma[slot].used_len = 0; | 312 | info->dma[slot].used_len = 0; |
180 | return 0; | 313 | return 0; |
181 | } | 314 | } |
182 | 315 | ||
316 | /* This is the actual receive routine. When we receive an interrupt from the | ||
317 | * Host to tell us a packet has been delivered, we arrive here: */ | ||
183 | static irqreturn_t lguestnet_rcv(int irq, void *dev_id) | 318 | static irqreturn_t lguestnet_rcv(int irq, void *dev_id) |
184 | { | 319 | { |
185 | struct net_device *dev = dev_id; | 320 | struct net_device *dev = dev_id; |
186 | struct lguestnet_info *info = netdev_priv(dev); | 321 | struct lguestnet_info *info = netdev_priv(dev); |
187 | unsigned int i, done = 0; | 322 | unsigned int i, done = 0; |
188 | 323 | ||
324 | /* Look through our entire receive array for an entry which has data | ||
325 | * in it. */ | ||
189 | for (i = 0; i < ARRAY_SIZE(info->dma); i++) { | 326 | for (i = 0; i < ARRAY_SIZE(info->dma); i++) { |
190 | unsigned int length; | 327 | unsigned int length; |
191 | struct sk_buff *skb; | 328 | struct sk_buff *skb; |
@@ -194,10 +331,16 @@ static irqreturn_t lguestnet_rcv(int irq, void *dev_id) | |||
194 | if (length == 0) | 331 | if (length == 0) |
195 | continue; | 332 | continue; |
196 | 333 | ||
334 | /* We've found one! Remember the skb (we grabbed the length | ||
335 | * above), and immediately refill the slot we've taken it | ||
336 | * from. */ | ||
197 | done++; | 337 | done++; |
198 | skb = info->skb[i]; | 338 | skb = info->skb[i]; |
199 | fill_slot(dev, i); | 339 | fill_slot(dev, i); |
200 | 340 | ||
341 | /* This shouldn't happen: micropackets could be sent by a | ||
342 | * badly-behaved Guest on the network, but the Host will never | ||
343 | * stuff more data in the buffer than the buffer length. */ | ||
201 | if (length < ETH_HLEN || length > ETH_HLEN + ETH_DATA_LEN) { | 344 | if (length < ETH_HLEN || length > ETH_HLEN + ETH_DATA_LEN) { |
202 | pr_debug(KERN_WARNING "%s: unbelievable skb len: %i\n", | 345 | pr_debug(KERN_WARNING "%s: unbelievable skb len: %i\n", |
203 | dev->name, length); | 346 | dev->name, length); |
@@ -205,36 +348,72 @@ static irqreturn_t lguestnet_rcv(int irq, void *dev_id) | |||
205 | continue; | 348 | continue; |
206 | } | 349 | } |
207 | 350 | ||
351 | /* skb_put(), what a great function! I've ranted about this | ||
352 | * function before (http://lkml.org/lkml/1999/9/26/24). You | ||
353 | * call it after you've added data to the end of an skb (in | ||
354 | * this case, it was the Host which wrote the data). */ | ||
208 | skb_put(skb, length); | 355 | skb_put(skb, length); |
356 | |||
357 | /* The ethernet header contains a protocol field: we use the | ||
358 | * standard helper to extract it, and place the result in | ||
359 | * skb->protocol. The helper also sets up skb->pkt_type and | ||
360 | * eats up the ethernet header from the front of the packet. */ | ||
209 | skb->protocol = eth_type_trans(skb, dev); | 361 | skb->protocol = eth_type_trans(skb, dev); |
210 | /* This is a reliable transport. */ | 362 | |
363 | /* If this device doesn't need checksums for sending, we also | ||
364 | * don't need to check the packets when they come in. */ | ||
211 | if (dev->features & NETIF_F_NO_CSUM) | 365 | if (dev->features & NETIF_F_NO_CSUM) |
212 | skb->ip_summed = CHECKSUM_UNNECESSARY; | 366 | skb->ip_summed = CHECKSUM_UNNECESSARY; |
367 | |||
368 | /* As a last resort for debugging the driver or the lguest I/O | ||
369 | * subsystem, you can uncomment the "#define DEBUG" at the top | ||
370 | * of this file, which turns all the pr_debug() into printk() | ||
371 | * and floods the logs. */ | ||
213 | pr_debug("Receiving skb proto 0x%04x len %i type %i\n", | 372 | pr_debug("Receiving skb proto 0x%04x len %i type %i\n", |
214 | ntohs(skb->protocol), skb->len, skb->pkt_type); | 373 | ntohs(skb->protocol), skb->len, skb->pkt_type); |
215 | 374 | ||
375 | /* Update the packet and byte counts (visible from ifconfig, | ||
376 | * and good for debugging). */ | ||
216 | dev->stats.rx_bytes += skb->len; | 377 | dev->stats.rx_bytes += skb->len; |
217 | dev->stats.rx_packets++; | 378 | dev->stats.rx_packets++; |
379 | |||
380 | /* Hand our fresh network packet into the stack's "network | ||
381 | * interface receive" routine. That will free the packet | ||
382 | * itself when it's finished. */ | ||
218 | netif_rx(skb); | 383 | netif_rx(skb); |
219 | } | 384 | } |
385 | |||
386 | /* If we found any packets, we assume the interrupt was for us. */ | ||
220 | return done ? IRQ_HANDLED : IRQ_NONE; | 387 | return done ? IRQ_HANDLED : IRQ_NONE; |
221 | } | 388 | } |
222 | 389 | ||
390 | /*D:550 This is where we start: when the device is brought up by dhcpd or | ||
391 | * ifconfig. At this point we advertise our MAC address to the rest of the | ||
392 | * network, and register receive buffers ready for incoming packets. */ | ||
223 | static int lguestnet_open(struct net_device *dev) | 393 | static int lguestnet_open(struct net_device *dev) |
224 | { | 394 | { |
225 | int i; | 395 | int i; |
226 | struct lguestnet_info *info = netdev_priv(dev); | 396 | struct lguestnet_info *info = netdev_priv(dev); |
227 | 397 | ||
228 | /* Set up our MAC address */ | 398 | /* Copy our MAC address into the device page, so others on the network |
399 | * can find us. */ | ||
229 | memcpy(info->peer[info->me].mac, dev->dev_addr, ETH_ALEN); | 400 | memcpy(info->peer[info->me].mac, dev->dev_addr, ETH_ALEN); |
230 | 401 | ||
231 | /* Turn on promisc mode if needed */ | 402 | /* We might already be in promisc mode (dev->flags & IFF_PROMISC). Our |
403 | * set_multicast callback handles this already, so we call it now. */ | ||
232 | lguestnet_set_multicast(dev); | 404 | lguestnet_set_multicast(dev); |
233 | 405 | ||
406 | /* Allocate packets and put them into our "struct lguest_dma" array. | ||
407 | * If we fail to allocate all the packets we could still limp along, | ||
408 | * but it's a sign of real stress so we should probably give up now. */ | ||
234 | for (i = 0; i < ARRAY_SIZE(info->dma); i++) { | 409 | for (i = 0; i < ARRAY_SIZE(info->dma); i++) { |
235 | if (fill_slot(dev, i) != 0) | 410 | if (fill_slot(dev, i) != 0) |
236 | goto cleanup; | 411 | goto cleanup; |
237 | } | 412 | } |
413 | |||
414 | /* Finally we tell the Host where our array of "struct lguest_dma" | ||
415 | * receive buffers is, binding it to the key corresponding to the | ||
416 | * device's physical memory plus our peerid. */ | ||
238 | if (lguest_bind_dma(peer_key(info,info->me), info->dma, | 417 | if (lguest_bind_dma(peer_key(info,info->me), info->dma, |
239 | NUM_SKBS, lgdev_irq(info->lgdev)) != 0) | 418 | NUM_SKBS, lgdev_irq(info->lgdev)) != 0) |
240 | goto cleanup; | 419 | goto cleanup; |
@@ -245,22 +424,29 @@ cleanup: | |||
245 | dev_kfree_skb(info->skb[i]); | 424 | dev_kfree_skb(info->skb[i]); |
246 | return -ENOMEM; | 425 | return -ENOMEM; |
247 | } | 426 | } |
427 | /*:*/ | ||
248 | 428 | ||
429 | /* The close routine is called when the device is no longer in use: we clean up | ||
430 | * elegantly. */ | ||
249 | static int lguestnet_close(struct net_device *dev) | 431 | static int lguestnet_close(struct net_device *dev) |
250 | { | 432 | { |
251 | unsigned int i; | 433 | unsigned int i; |
252 | struct lguestnet_info *info = netdev_priv(dev); | 434 | struct lguestnet_info *info = netdev_priv(dev); |
253 | 435 | ||
254 | /* Clear all trace: others might deliver packets, we'll ignore it. */ | 436 | /* Clear all trace of our existence out of the device memory by setting |
437 | * the slot which held our MAC address to 0 (unused). */ | ||
255 | memset(&info->peer[info->me], 0, sizeof(info->peer[info->me])); | 438 | memset(&info->peer[info->me], 0, sizeof(info->peer[info->me])); |
256 | 439 | ||
257 | /* Deregister sg lists. */ | 440 | /* Unregister our array of receive buffers */ |
258 | lguest_unbind_dma(peer_key(info, info->me), info->dma); | 441 | lguest_unbind_dma(peer_key(info, info->me), info->dma); |
259 | for (i = 0; i < ARRAY_SIZE(info->dma); i++) | 442 | for (i = 0; i < ARRAY_SIZE(info->dma); i++) |
260 | dev_kfree_skb(info->skb[i]); | 443 | dev_kfree_skb(info->skb[i]); |
261 | return 0; | 444 | return 0; |
262 | } | 445 | } |
263 | 446 | ||
447 | /*D:510 The network device probe function is basically a standard ethernet | ||
448 | * device setup. It reads the "struct lguest_device_desc" and sets the "struct | ||
449 | * net_device". Oh, the line-by-line excitement! Let's skip over it. :*/ | ||
264 | static int lguestnet_probe(struct lguest_device *lgdev) | 450 | static int lguestnet_probe(struct lguest_device *lgdev) |
265 | { | 451 | { |
266 | int err, irqf = IRQF_SHARED; | 452 | int err, irqf = IRQF_SHARED; |
@@ -290,10 +476,16 @@ static int lguestnet_probe(struct lguest_device *lgdev) | |||
290 | dev->stop = lguestnet_close; | 476 | dev->stop = lguestnet_close; |
291 | dev->hard_start_xmit = lguestnet_start_xmit; | 477 | dev->hard_start_xmit = lguestnet_start_xmit; |
292 | 478 | ||
293 | /* Turning on/off promisc will call dev->set_multicast_list. | 479 | /* We don't actually support multicast yet, but turning on/off |
294 | * We don't actually support multicast yet */ | 480 | * promisc also calls dev->set_multicast_list. */ |
295 | dev->set_multicast_list = lguestnet_set_multicast; | 481 | dev->set_multicast_list = lguestnet_set_multicast; |
296 | SET_NETDEV_DEV(dev, &lgdev->dev); | 482 | SET_NETDEV_DEV(dev, &lgdev->dev); |
483 | |||
484 | /* The network code complains if you have "scatter-gather" capability | ||
485 | * if you don't also handle checksums (it seem that would be | ||
486 | * "illogical"). So we use a lie of omission and don't tell it that we | ||
487 | * can handle scattered packets unless we also don't want checksums, | ||
488 | * even though to us they're completely independent. */ | ||
297 | if (desc->features & LGUEST_NET_F_NOCSUM) | 489 | if (desc->features & LGUEST_NET_F_NOCSUM) |
298 | dev->features = NETIF_F_SG|NETIF_F_NO_CSUM; | 490 | dev->features = NETIF_F_SG|NETIF_F_NO_CSUM; |
299 | 491 | ||
@@ -325,6 +517,9 @@ static int lguestnet_probe(struct lguest_device *lgdev) | |||
325 | } | 517 | } |
326 | 518 | ||
327 | pr_debug("lguestnet: registered device %s\n", dev->name); | 519 | pr_debug("lguestnet: registered device %s\n", dev->name); |
520 | /* Finally, we put the "struct net_device" in the generic "struct | ||
521 | * lguest_device"s private pointer. Again, it's not necessary, but | ||
522 | * makes sure the cool kernel kids don't tease us. */ | ||
328 | lgdev->private = dev; | 523 | lgdev->private = dev; |
329 | return 0; | 524 | return 0; |
330 | 525 | ||
@@ -352,3 +547,11 @@ module_init(lguestnet_init); | |||
352 | 547 | ||
353 | MODULE_DESCRIPTION("Lguest network driver"); | 548 | MODULE_DESCRIPTION("Lguest network driver"); |
354 | MODULE_LICENSE("GPL"); | 549 | MODULE_LICENSE("GPL"); |
550 | |||
551 | /*D:580 | ||
552 | * This is the last of the Drivers, and with this we have covered the many and | ||
553 | * wonderous and fine (and boring) details of the Guest. | ||
554 | * | ||
555 | * "make Launcher" beckons, where we answer questions like "Where do Guests | ||
556 | * come from?", and "What do you do when someone asks for optimization?" | ||
557 | */ | ||
diff --git a/drivers/net/lib8390.c b/drivers/net/lib8390.c index 5c86e737f954..721ee38d2241 100644 --- a/drivers/net/lib8390.c +++ b/drivers/net/lib8390.c | |||
@@ -219,15 +219,6 @@ static void ei_tx_timeout(struct net_device *dev) | |||
219 | int txsr, isr, tickssofar = jiffies - dev->trans_start; | 219 | int txsr, isr, tickssofar = jiffies - dev->trans_start; |
220 | unsigned long flags; | 220 | unsigned long flags; |
221 | 221 | ||
222 | #if defined(CONFIG_M32R) && defined(CONFIG_SMP) | ||
223 | unsigned long icucr; | ||
224 | |||
225 | local_irq_save(flags); | ||
226 | icucr = inl(M32R_ICU_CR1_PORTL); | ||
227 | icucr |= M32R_ICUCR_ISMOD11; | ||
228 | outl(icucr, M32R_ICU_CR1_PORTL); | ||
229 | local_irq_restore(flags); | ||
230 | #endif | ||
231 | ei_local->stat.tx_errors++; | 222 | ei_local->stat.tx_errors++; |
232 | 223 | ||
233 | spin_lock_irqsave(&ei_local->page_lock, flags); | 224 | spin_lock_irqsave(&ei_local->page_lock, flags); |
diff --git a/drivers/net/netxen/netxen_nic.h b/drivers/net/netxen/netxen_nic.h index 325269d8ae38..d4c92cc879d4 100644 --- a/drivers/net/netxen/netxen_nic.h +++ b/drivers/net/netxen/netxen_nic.h | |||
@@ -1179,8 +1179,7 @@ dma_watchdog_shutdown_poll_result(struct netxen_adapter *adapter) | |||
1179 | NETXEN_CAM_RAM(NETXEN_CAM_RAM_DMA_WATCHDOG_CTRL), &ctrl, 4)) | 1179 | NETXEN_CAM_RAM(NETXEN_CAM_RAM_DMA_WATCHDOG_CTRL), &ctrl, 4)) |
1180 | printk(KERN_ERR "failed to read dma watchdog status\n"); | 1180 | printk(KERN_ERR "failed to read dma watchdog status\n"); |
1181 | 1181 | ||
1182 | return ((netxen_get_dma_watchdog_enabled(ctrl) == 0) && | 1182 | return (netxen_get_dma_watchdog_enabled(ctrl) == 0); |
1183 | (netxen_get_dma_watchdog_disabled(ctrl) == 0)); | ||
1184 | } | 1183 | } |
1185 | 1184 | ||
1186 | static inline int | 1185 | static inline int |
diff --git a/drivers/net/netxen/netxen_nic_main.c b/drivers/net/netxen/netxen_nic_main.c index b703ccfe040b..19e2fa940ac0 100644 --- a/drivers/net/netxen/netxen_nic_main.c +++ b/drivers/net/netxen/netxen_nic_main.c | |||
@@ -46,7 +46,7 @@ MODULE_DESCRIPTION("NetXen Multi port (1/10) Gigabit Network Driver"); | |||
46 | MODULE_LICENSE("GPL"); | 46 | MODULE_LICENSE("GPL"); |
47 | MODULE_VERSION(NETXEN_NIC_LINUX_VERSIONID); | 47 | MODULE_VERSION(NETXEN_NIC_LINUX_VERSIONID); |
48 | 48 | ||
49 | char netxen_nic_driver_name[] = "netxen-nic"; | 49 | char netxen_nic_driver_name[] = "netxen_nic"; |
50 | static char netxen_nic_driver_string[] = "NetXen Network Driver version " | 50 | static char netxen_nic_driver_string[] = "NetXen Network Driver version " |
51 | NETXEN_NIC_LINUX_VERSIONID; | 51 | NETXEN_NIC_LINUX_VERSIONID; |
52 | 52 | ||
@@ -640,6 +640,10 @@ netxen_nic_probe(struct pci_dev *pdev, const struct pci_device_id *ent) | |||
640 | NETXEN_CRB_NORMALIZE(adapter, | 640 | NETXEN_CRB_NORMALIZE(adapter, |
641 | NETXEN_ROMUSB_GLB_PEGTUNE_DONE)); | 641 | NETXEN_ROMUSB_GLB_PEGTUNE_DONE)); |
642 | /* Handshake with the card before we register the devices. */ | 642 | /* Handshake with the card before we register the devices. */ |
643 | writel(0, NETXEN_CRB_NORMALIZE(adapter, CRB_CMDPEG_STATE)); | ||
644 | netxen_pinit_from_rom(adapter, 0); | ||
645 | msleep(1); | ||
646 | netxen_load_firmware(adapter); | ||
643 | netxen_phantom_init(adapter, NETXEN_NIC_PEG_TUNE); | 647 | netxen_phantom_init(adapter, NETXEN_NIC_PEG_TUNE); |
644 | } | 648 | } |
645 | 649 | ||
@@ -782,19 +786,18 @@ static void __devexit netxen_nic_remove(struct pci_dev *pdev) | |||
782 | 786 | ||
783 | if (adapter->portnum == 0) { | 787 | if (adapter->portnum == 0) { |
784 | if (init_firmware_done) { | 788 | if (init_firmware_done) { |
785 | dma_watchdog_shutdown_request(adapter); | ||
786 | msleep(100); | ||
787 | i = 100; | 789 | i = 100; |
788 | while ((dma_watchdog_shutdown_poll_result(adapter) != 1) && i) { | 790 | do { |
789 | printk(KERN_INFO "dma_watchdog_shutdown_poll still in progress\n"); | 791 | if (dma_watchdog_shutdown_request(adapter) == 1) |
792 | break; | ||
790 | msleep(100); | 793 | msleep(100); |
791 | i--; | 794 | if (dma_watchdog_shutdown_poll_result(adapter) == 1) |
792 | } | 795 | break; |
796 | } while (--i); | ||
793 | 797 | ||
794 | if (i == 0) { | 798 | if (i == 0) |
795 | printk(KERN_ERR "dma_watchdog_shutdown_request failed\n"); | 799 | printk(KERN_ERR "%s: dma_watchdog_shutdown failed\n", |
796 | return; | 800 | netdev->name); |
797 | } | ||
798 | 801 | ||
799 | /* clear the register for future unloads/loads */ | 802 | /* clear the register for future unloads/loads */ |
800 | writel(0, NETXEN_CRB_NORMALIZE(adapter, NETXEN_CAM_RAM(0x1fc))); | 803 | writel(0, NETXEN_CRB_NORMALIZE(adapter, NETXEN_CAM_RAM(0x1fc))); |
@@ -803,11 +806,9 @@ static void __devexit netxen_nic_remove(struct pci_dev *pdev) | |||
803 | 806 | ||
804 | /* leave the hw in the same state as reboot */ | 807 | /* leave the hw in the same state as reboot */ |
805 | writel(0, NETXEN_CRB_NORMALIZE(adapter, CRB_CMDPEG_STATE)); | 808 | writel(0, NETXEN_CRB_NORMALIZE(adapter, CRB_CMDPEG_STATE)); |
806 | if (netxen_pinit_from_rom(adapter, 0)) | 809 | netxen_pinit_from_rom(adapter, 0); |
807 | return; | ||
808 | msleep(1); | 810 | msleep(1); |
809 | if (netxen_load_firmware(adapter)) | 811 | netxen_load_firmware(adapter); |
810 | return; | ||
811 | netxen_phantom_init(adapter, NETXEN_NIC_PEG_TUNE); | 812 | netxen_phantom_init(adapter, NETXEN_NIC_PEG_TUNE); |
812 | } | 813 | } |
813 | 814 | ||
@@ -816,22 +817,21 @@ static void __devexit netxen_nic_remove(struct pci_dev *pdev) | |||
816 | printk(KERN_INFO "State: 0x%0x\n", | 817 | printk(KERN_INFO "State: 0x%0x\n", |
817 | readl(NETXEN_CRB_NORMALIZE(adapter, CRB_CMDPEG_STATE))); | 818 | readl(NETXEN_CRB_NORMALIZE(adapter, CRB_CMDPEG_STATE))); |
818 | 819 | ||
819 | dma_watchdog_shutdown_request(adapter); | ||
820 | msleep(100); | ||
821 | i = 100; | 820 | i = 100; |
822 | while ((dma_watchdog_shutdown_poll_result(adapter) != 1) && i) { | 821 | do { |
823 | printk(KERN_INFO "dma_watchdog_shutdown_poll still in progress\n"); | 822 | if (dma_watchdog_shutdown_request(adapter) == 1) |
823 | break; | ||
824 | msleep(100); | 824 | msleep(100); |
825 | i--; | 825 | if (dma_watchdog_shutdown_poll_result(adapter) == 1) |
826 | } | 826 | break; |
827 | } while (--i); | ||
827 | 828 | ||
828 | if (i) { | 829 | if (i) { |
829 | netxen_free_adapter_offload(adapter); | 830 | netxen_free_adapter_offload(adapter); |
830 | } else { | 831 | } else { |
831 | printk(KERN_ERR "failed to dma shutdown\n"); | 832 | printk(KERN_ERR "%s: dma_watchdog_shutdown failed\n", |
832 | return; | 833 | netdev->name); |
833 | } | 834 | } |
834 | |||
835 | } | 835 | } |
836 | 836 | ||
837 | iounmap(adapter->ahw.db_base); | 837 | iounmap(adapter->ahw.db_base); |
diff --git a/drivers/net/phy/vitesse.c b/drivers/net/phy/vitesse.c index 6a5385647911..8874497b6bbf 100644 --- a/drivers/net/phy/vitesse.c +++ b/drivers/net/phy/vitesse.c | |||
@@ -109,7 +109,7 @@ static int vsc824x_config_intr(struct phy_device *phydev) | |||
109 | */ | 109 | */ |
110 | err = phy_read(phydev, MII_VSC8244_ISTAT); | 110 | err = phy_read(phydev, MII_VSC8244_ISTAT); |
111 | 111 | ||
112 | if (err) | 112 | if (err < 0) |
113 | return err; | 113 | return err; |
114 | 114 | ||
115 | err = phy_write(phydev, MII_VSC8244_IMASK, 0); | 115 | err = phy_write(phydev, MII_VSC8244_IMASK, 0); |
diff --git a/drivers/net/pppol2tp.c b/drivers/net/pppol2tp.c index f87176055d0e..266e8b38fe10 100644 --- a/drivers/net/pppol2tp.c +++ b/drivers/net/pppol2tp.c | |||
@@ -2054,7 +2054,7 @@ end: | |||
2054 | */ | 2054 | */ |
2055 | static int pppol2tp_tunnel_getsockopt(struct sock *sk, | 2055 | static int pppol2tp_tunnel_getsockopt(struct sock *sk, |
2056 | struct pppol2tp_tunnel *tunnel, | 2056 | struct pppol2tp_tunnel *tunnel, |
2057 | int optname, int __user *val) | 2057 | int optname, int *val) |
2058 | { | 2058 | { |
2059 | int err = 0; | 2059 | int err = 0; |
2060 | 2060 | ||
@@ -2077,7 +2077,7 @@ static int pppol2tp_tunnel_getsockopt(struct sock *sk, | |||
2077 | */ | 2077 | */ |
2078 | static int pppol2tp_session_getsockopt(struct sock *sk, | 2078 | static int pppol2tp_session_getsockopt(struct sock *sk, |
2079 | struct pppol2tp_session *session, | 2079 | struct pppol2tp_session *session, |
2080 | int optname, int __user *val) | 2080 | int optname, int *val) |
2081 | { | 2081 | { |
2082 | int err = 0; | 2082 | int err = 0; |
2083 | 2083 | ||
diff --git a/drivers/net/ps3_gelic_net.c b/drivers/net/ps3_gelic_net.c index 08d25066f051..13d1c0a2a25f 100644 --- a/drivers/net/ps3_gelic_net.c +++ b/drivers/net/ps3_gelic_net.c | |||
@@ -290,7 +290,8 @@ static void gelic_net_release_rx_chain(struct gelic_net_card *card) | |||
290 | descr->buf_addr = 0; | 290 | descr->buf_addr = 0; |
291 | dev_kfree_skb_any(descr->skb); | 291 | dev_kfree_skb_any(descr->skb); |
292 | descr->skb = NULL; | 292 | descr->skb = NULL; |
293 | descr->dmac_cmd_status = GELIC_NET_DESCR_NOT_IN_USE; | 293 | gelic_net_set_descr_status(descr, |
294 | GELIC_NET_DESCR_NOT_IN_USE); | ||
294 | } | 295 | } |
295 | descr = descr->next; | 296 | descr = descr->next; |
296 | } while (descr != card->rx_chain.head); | 297 | } while (descr != card->rx_chain.head); |
@@ -374,7 +375,7 @@ static void gelic_net_release_tx_descr(struct gelic_net_card *card, | |||
374 | descr->skb = NULL; | 375 | descr->skb = NULL; |
375 | 376 | ||
376 | /* set descr status */ | 377 | /* set descr status */ |
377 | descr->dmac_cmd_status = GELIC_NET_DMAC_CMDSTAT_NOT_IN_USE; | 378 | gelic_net_set_descr_status(descr, GELIC_NET_DESCR_NOT_IN_USE); |
378 | } | 379 | } |
379 | 380 | ||
380 | /** | 381 | /** |
@@ -403,26 +404,29 @@ static void gelic_net_release_tx_chain(struct gelic_net_card *card, int stop) | |||
403 | "%s: forcing end of tx descriptor " \ | 404 | "%s: forcing end of tx descriptor " \ |
404 | "with status %x\n", | 405 | "with status %x\n", |
405 | __func__, status); | 406 | __func__, status); |
406 | card->netdev_stats.tx_dropped++; | 407 | card->netdev->stats.tx_dropped++; |
407 | break; | 408 | break; |
408 | 409 | ||
409 | case GELIC_NET_DESCR_COMPLETE: | 410 | case GELIC_NET_DESCR_COMPLETE: |
410 | card->netdev_stats.tx_packets++; | 411 | if (tx_chain->tail->skb) { |
411 | card->netdev_stats.tx_bytes += | 412 | card->netdev->stats.tx_packets++; |
412 | tx_chain->tail->skb->len; | 413 | card->netdev->stats.tx_bytes += |
414 | tx_chain->tail->skb->len; | ||
415 | } | ||
413 | break; | 416 | break; |
414 | 417 | ||
415 | case GELIC_NET_DESCR_CARDOWNED: | 418 | case GELIC_NET_DESCR_CARDOWNED: |
416 | /* pending tx request */ | 419 | /* pending tx request */ |
417 | default: | 420 | default: |
418 | /* any other value (== GELIC_NET_DESCR_NOT_IN_USE) */ | 421 | /* any other value (== GELIC_NET_DESCR_NOT_IN_USE) */ |
419 | goto out; | 422 | if (!stop) |
423 | goto out; | ||
420 | } | 424 | } |
421 | gelic_net_release_tx_descr(card, tx_chain->tail); | 425 | gelic_net_release_tx_descr(card, tx_chain->tail); |
422 | release = 1; | 426 | release ++; |
423 | } | 427 | } |
424 | out: | 428 | out: |
425 | if (!stop && release) | 429 | if (!stop && (2 < release)) |
426 | netif_wake_queue(card->netdev); | 430 | netif_wake_queue(card->netdev); |
427 | } | 431 | } |
428 | 432 | ||
@@ -659,19 +663,21 @@ static int gelic_net_prepare_tx_descr_v(struct gelic_net_card *card, | |||
659 | { | 663 | { |
660 | dma_addr_t buf[2]; | 664 | dma_addr_t buf[2]; |
661 | unsigned int vlan_len; | 665 | unsigned int vlan_len; |
666 | struct gelic_net_descr *sec_descr = descr->next; | ||
662 | 667 | ||
663 | if (skb->len < GELIC_NET_VLAN_POS) | 668 | if (skb->len < GELIC_NET_VLAN_POS) |
664 | return -EINVAL; | 669 | return -EINVAL; |
665 | 670 | ||
666 | memcpy(&descr->vlan, skb->data, GELIC_NET_VLAN_POS); | 671 | vlan_len = GELIC_NET_VLAN_POS; |
672 | memcpy(&descr->vlan, skb->data, vlan_len); | ||
667 | if (card->vlan_index != -1) { | 673 | if (card->vlan_index != -1) { |
674 | /* internal vlan tag used */ | ||
668 | descr->vlan.h_vlan_proto = htons(ETH_P_8021Q); /* vlan 0x8100*/ | 675 | descr->vlan.h_vlan_proto = htons(ETH_P_8021Q); /* vlan 0x8100*/ |
669 | descr->vlan.h_vlan_TCI = htons(card->vlan_id[card->vlan_index]); | 676 | descr->vlan.h_vlan_TCI = htons(card->vlan_id[card->vlan_index]); |
670 | vlan_len = GELIC_NET_VLAN_POS + VLAN_HLEN; /* VLAN_HLEN=4 */ | 677 | vlan_len += VLAN_HLEN; /* added for above two lines */ |
671 | } else | 678 | } |
672 | vlan_len = GELIC_NET_VLAN_POS; /* no vlan tag */ | ||
673 | 679 | ||
674 | /* first descr */ | 680 | /* map data area */ |
675 | buf[0] = dma_map_single(ctodev(card), &descr->vlan, | 681 | buf[0] = dma_map_single(ctodev(card), &descr->vlan, |
676 | vlan_len, DMA_TO_DEVICE); | 682 | vlan_len, DMA_TO_DEVICE); |
677 | 683 | ||
@@ -682,20 +688,6 @@ static int gelic_net_prepare_tx_descr_v(struct gelic_net_card *card, | |||
682 | return -ENOMEM; | 688 | return -ENOMEM; |
683 | } | 689 | } |
684 | 690 | ||
685 | descr->buf_addr = buf[0]; | ||
686 | descr->buf_size = vlan_len; | ||
687 | descr->skb = skb; /* not used */ | ||
688 | descr->data_status = 0; | ||
689 | gelic_net_set_txdescr_cmdstat(descr, skb, 1); /* not the frame end */ | ||
690 | |||
691 | /* second descr */ | ||
692 | card->tx_chain.head = card->tx_chain.head->next; | ||
693 | descr->next_descr_addr = descr->next->bus_addr; | ||
694 | descr = descr->next; | ||
695 | if (gelic_net_get_descr_status(descr) != GELIC_NET_DESCR_NOT_IN_USE) | ||
696 | /* XXX will be removed */ | ||
697 | dev_err(ctodev(card), "descr is not free!\n"); | ||
698 | |||
699 | buf[1] = dma_map_single(ctodev(card), skb->data + GELIC_NET_VLAN_POS, | 691 | buf[1] = dma_map_single(ctodev(card), skb->data + GELIC_NET_VLAN_POS, |
700 | skb->len - GELIC_NET_VLAN_POS, | 692 | skb->len - GELIC_NET_VLAN_POS, |
701 | DMA_TO_DEVICE); | 693 | DMA_TO_DEVICE); |
@@ -710,13 +702,24 @@ static int gelic_net_prepare_tx_descr_v(struct gelic_net_card *card, | |||
710 | return -ENOMEM; | 702 | return -ENOMEM; |
711 | } | 703 | } |
712 | 704 | ||
713 | descr->buf_addr = buf[1]; | 705 | /* first descr */ |
714 | descr->buf_size = skb->len - GELIC_NET_VLAN_POS; | 706 | descr->buf_addr = buf[0]; |
715 | descr->skb = skb; | 707 | descr->buf_size = vlan_len; |
708 | descr->skb = NULL; /* not used */ | ||
716 | descr->data_status = 0; | 709 | descr->data_status = 0; |
717 | descr->next_descr_addr = 0; /* terminate hw descr */ | 710 | descr->next_descr_addr = descr->next->bus_addr; |
718 | gelic_net_set_txdescr_cmdstat(descr, skb, 0); | 711 | gelic_net_set_txdescr_cmdstat(descr, skb, 1); /* not the frame end */ |
719 | 712 | ||
713 | /* second descr */ | ||
714 | sec_descr->buf_addr = buf[1]; | ||
715 | sec_descr->buf_size = skb->len - GELIC_NET_VLAN_POS; | ||
716 | sec_descr->skb = skb; | ||
717 | sec_descr->data_status = 0; | ||
718 | sec_descr->next_descr_addr = 0; /* terminate hw descr */ | ||
719 | gelic_net_set_txdescr_cmdstat(sec_descr, skb, 0); | ||
720 | |||
721 | /* bump free descriptor pointer */ | ||
722 | card->tx_chain.head = sec_descr->next; | ||
720 | return 0; | 723 | return 0; |
721 | } | 724 | } |
722 | 725 | ||
@@ -729,7 +732,7 @@ static int gelic_net_prepare_tx_descr_v(struct gelic_net_card *card, | |||
729 | static int gelic_net_kick_txdma(struct gelic_net_card *card, | 732 | static int gelic_net_kick_txdma(struct gelic_net_card *card, |
730 | struct gelic_net_descr *descr) | 733 | struct gelic_net_descr *descr) |
731 | { | 734 | { |
732 | int status = -ENXIO; | 735 | int status = 0; |
733 | int count = 10; | 736 | int count = 10; |
734 | 737 | ||
735 | if (card->tx_dma_progress) | 738 | if (card->tx_dma_progress) |
@@ -763,47 +766,62 @@ static int gelic_net_kick_txdma(struct gelic_net_card *card, | |||
763 | static int gelic_net_xmit(struct sk_buff *skb, struct net_device *netdev) | 766 | static int gelic_net_xmit(struct sk_buff *skb, struct net_device *netdev) |
764 | { | 767 | { |
765 | struct gelic_net_card *card = netdev_priv(netdev); | 768 | struct gelic_net_card *card = netdev_priv(netdev); |
766 | struct gelic_net_descr *descr = NULL; | 769 | struct gelic_net_descr *descr; |
767 | int result; | 770 | int result; |
768 | unsigned long flags; | 771 | unsigned long flags; |
769 | 772 | ||
770 | spin_lock_irqsave(&card->tx_dma_lock, flags); | 773 | spin_lock_irqsave(&card->tx_dma_lock, flags); |
771 | 774 | ||
772 | gelic_net_release_tx_chain(card, 0); | 775 | gelic_net_release_tx_chain(card, 0); |
773 | if (!skb) | 776 | |
774 | goto kick; | ||
775 | descr = gelic_net_get_next_tx_descr(card); | 777 | descr = gelic_net_get_next_tx_descr(card); |
776 | if (!descr) { | 778 | if (!descr) { |
779 | /* | ||
780 | * no more descriptors free | ||
781 | */ | ||
777 | netif_stop_queue(netdev); | 782 | netif_stop_queue(netdev); |
778 | spin_unlock_irqrestore(&card->tx_dma_lock, flags); | 783 | spin_unlock_irqrestore(&card->tx_dma_lock, flags); |
779 | return NETDEV_TX_BUSY; | 784 | return NETDEV_TX_BUSY; |
780 | } | 785 | } |
781 | result = gelic_net_prepare_tx_descr_v(card, descr, skb); | ||
782 | |||
783 | if (result) | ||
784 | goto error; | ||
785 | 786 | ||
786 | card->tx_chain.head = card->tx_chain.head->next; | 787 | result = gelic_net_prepare_tx_descr_v(card, descr, skb); |
787 | 788 | if (result) { | |
788 | if (descr->prev) | 789 | /* |
789 | descr->prev->next_descr_addr = descr->bus_addr; | 790 | * DMA map failed. As chanses are that failure |
790 | kick: | 791 | * would continue, just release skb and return |
792 | */ | ||
793 | card->netdev->stats.tx_dropped++; | ||
794 | dev_kfree_skb_any(skb); | ||
795 | spin_unlock_irqrestore(&card->tx_dma_lock, flags); | ||
796 | return NETDEV_TX_OK; | ||
797 | } | ||
798 | /* | ||
799 | * link this prepared descriptor to previous one | ||
800 | * to achieve high performance | ||
801 | */ | ||
802 | descr->prev->next_descr_addr = descr->bus_addr; | ||
791 | /* | 803 | /* |
792 | * as hardware descriptor is modified in the above lines, | 804 | * as hardware descriptor is modified in the above lines, |
793 | * ensure that the hardware sees it | 805 | * ensure that the hardware sees it |
794 | */ | 806 | */ |
795 | wmb(); | 807 | wmb(); |
796 | if (gelic_net_kick_txdma(card, card->tx_chain.tail)) | 808 | if (gelic_net_kick_txdma(card, descr)) { |
797 | goto error; | 809 | /* |
810 | * kick failed. | ||
811 | * release descriptors which were just prepared | ||
812 | */ | ||
813 | card->netdev->stats.tx_dropped++; | ||
814 | gelic_net_release_tx_descr(card, descr); | ||
815 | gelic_net_release_tx_descr(card, descr->next); | ||
816 | card->tx_chain.tail = descr->next->next; | ||
817 | dev_info(ctodev(card), "%s: kick failure\n", __func__); | ||
818 | } else { | ||
819 | /* OK, DMA started/reserved */ | ||
820 | netdev->trans_start = jiffies; | ||
821 | } | ||
798 | 822 | ||
799 | netdev->trans_start = jiffies; | ||
800 | spin_unlock_irqrestore(&card->tx_dma_lock, flags); | 823 | spin_unlock_irqrestore(&card->tx_dma_lock, flags); |
801 | return NETDEV_TX_OK; | 824 | return NETDEV_TX_OK; |
802 | |||
803 | error: | ||
804 | card->netdev_stats.tx_dropped++; | ||
805 | spin_unlock_irqrestore(&card->tx_dma_lock, flags); | ||
806 | return NETDEV_TX_LOCKED; | ||
807 | } | 825 | } |
808 | 826 | ||
809 | /** | 827 | /** |
@@ -854,8 +872,8 @@ static void gelic_net_pass_skb_up(struct gelic_net_descr *descr, | |||
854 | skb->ip_summed = CHECKSUM_NONE; | 872 | skb->ip_summed = CHECKSUM_NONE; |
855 | 873 | ||
856 | /* update netdevice statistics */ | 874 | /* update netdevice statistics */ |
857 | card->netdev_stats.rx_packets++; | 875 | card->netdev->stats.rx_packets++; |
858 | card->netdev_stats.rx_bytes += skb->len; | 876 | card->netdev->stats.rx_bytes += skb->len; |
859 | 877 | ||
860 | /* pass skb up to stack */ | 878 | /* pass skb up to stack */ |
861 | netif_receive_skb(skb); | 879 | netif_receive_skb(skb); |
@@ -895,38 +913,67 @@ static int gelic_net_decode_one_descr(struct gelic_net_card *card) | |||
895 | (status == GELIC_NET_DESCR_FORCE_END)) { | 913 | (status == GELIC_NET_DESCR_FORCE_END)) { |
896 | dev_info(ctodev(card), "dropping RX descriptor with state %x\n", | 914 | dev_info(ctodev(card), "dropping RX descriptor with state %x\n", |
897 | status); | 915 | status); |
898 | card->netdev_stats.rx_dropped++; | 916 | card->netdev->stats.rx_dropped++; |
899 | goto refill; | 917 | goto refill; |
900 | } | 918 | } |
901 | 919 | ||
902 | if ((status != GELIC_NET_DESCR_COMPLETE) && | 920 | if (status == GELIC_NET_DESCR_BUFFER_FULL) { |
903 | (status != GELIC_NET_DESCR_FRAME_END)) { | 921 | /* |
922 | * Buffer full would occur if and only if | ||
923 | * the frame length was longer than the size of this | ||
924 | * descriptor's buffer. If the frame length was equal | ||
925 | * to or shorter than buffer'size, FRAME_END condition | ||
926 | * would occur. | ||
927 | * Anyway this frame was longer than the MTU, | ||
928 | * just drop it. | ||
929 | */ | ||
930 | dev_info(ctodev(card), "overlength frame\n"); | ||
931 | goto refill; | ||
932 | } | ||
933 | /* | ||
934 | * descriptoers any other than FRAME_END here should | ||
935 | * be treated as error. | ||
936 | */ | ||
937 | if (status != GELIC_NET_DESCR_FRAME_END) { | ||
904 | dev_dbg(ctodev(card), "RX descriptor with state %x\n", | 938 | dev_dbg(ctodev(card), "RX descriptor with state %x\n", |
905 | status); | 939 | status); |
906 | goto refill; | 940 | goto refill; |
907 | } | 941 | } |
908 | 942 | ||
909 | /* ok, we've got a packet in descr */ | 943 | /* ok, we've got a packet in descr */ |
910 | gelic_net_pass_skb_up(descr, card); /* 1: skb_up sccess */ | 944 | gelic_net_pass_skb_up(descr, card); |
911 | |||
912 | refill: | 945 | refill: |
913 | descr->next_descr_addr = 0; /* unlink the descr */ | 946 | /* |
947 | * So that always DMAC can see the end | ||
948 | * of the descriptor chain to avoid | ||
949 | * from unwanted DMAC overrun. | ||
950 | */ | ||
951 | descr->next_descr_addr = 0; | ||
914 | 952 | ||
915 | /* change the descriptor state: */ | 953 | /* change the descriptor state: */ |
916 | gelic_net_set_descr_status(descr, GELIC_NET_DESCR_NOT_IN_USE); | 954 | gelic_net_set_descr_status(descr, GELIC_NET_DESCR_NOT_IN_USE); |
917 | 955 | ||
918 | /* refill one desc | 956 | /* |
919 | * FIXME: this can fail, but for now, just leave this | 957 | * this call can fail, but for now, just leave this |
920 | * descriptor without skb | 958 | * decriptor without skb |
921 | */ | 959 | */ |
922 | gelic_net_prepare_rx_descr(card, descr); | 960 | gelic_net_prepare_rx_descr(card, descr); |
961 | |||
923 | chain->head = descr; | 962 | chain->head = descr; |
924 | chain->tail = descr->next; | 963 | chain->tail = descr->next; |
964 | |||
965 | /* | ||
966 | * Set this descriptor the end of the chain. | ||
967 | */ | ||
925 | descr->prev->next_descr_addr = descr->bus_addr; | 968 | descr->prev->next_descr_addr = descr->bus_addr; |
926 | 969 | ||
970 | /* | ||
971 | * If dmac chain was met, DMAC stopped. | ||
972 | * thus re-enable it | ||
973 | */ | ||
927 | if (dmac_chain_ended) { | 974 | if (dmac_chain_ended) { |
928 | gelic_net_enable_rxdmac(card); | 975 | card->rx_dma_restart_required = 1; |
929 | dev_dbg(ctodev(card), "reenable rx dma\n"); | 976 | dev_dbg(ctodev(card), "reenable rx dma scheduled\n"); |
930 | } | 977 | } |
931 | 978 | ||
932 | return 1; | 979 | return 1; |
@@ -968,20 +1015,6 @@ static int gelic_net_poll(struct net_device *netdev, int *budget) | |||
968 | } else | 1015 | } else |
969 | return 1; | 1016 | return 1; |
970 | } | 1017 | } |
971 | |||
972 | /** | ||
973 | * gelic_net_get_stats - get interface statistics | ||
974 | * @netdev: interface device structure | ||
975 | * | ||
976 | * returns the interface statistics residing in the gelic_net_card struct | ||
977 | */ | ||
978 | static struct net_device_stats *gelic_net_get_stats(struct net_device *netdev) | ||
979 | { | ||
980 | struct gelic_net_card *card = netdev_priv(netdev); | ||
981 | |||
982 | return &card->netdev_stats; | ||
983 | } | ||
984 | |||
985 | /** | 1018 | /** |
986 | * gelic_net_change_mtu - changes the MTU of an interface | 1019 | * gelic_net_change_mtu - changes the MTU of an interface |
987 | * @netdev: interface device structure | 1020 | * @netdev: interface device structure |
@@ -1016,6 +1049,11 @@ static irqreturn_t gelic_net_interrupt(int irq, void *ptr) | |||
1016 | if (!status) | 1049 | if (!status) |
1017 | return IRQ_NONE; | 1050 | return IRQ_NONE; |
1018 | 1051 | ||
1052 | if (card->rx_dma_restart_required) { | ||
1053 | card->rx_dma_restart_required = 0; | ||
1054 | gelic_net_enable_rxdmac(card); | ||
1055 | } | ||
1056 | |||
1019 | if (status & GELIC_NET_RXINT) { | 1057 | if (status & GELIC_NET_RXINT) { |
1020 | gelic_net_rx_irq_off(card); | 1058 | gelic_net_rx_irq_off(card); |
1021 | netif_rx_schedule(netdev); | 1059 | netif_rx_schedule(netdev); |
@@ -1024,9 +1062,10 @@ static irqreturn_t gelic_net_interrupt(int irq, void *ptr) | |||
1024 | if (status & GELIC_NET_TXINT) { | 1062 | if (status & GELIC_NET_TXINT) { |
1025 | spin_lock_irqsave(&card->tx_dma_lock, flags); | 1063 | spin_lock_irqsave(&card->tx_dma_lock, flags); |
1026 | card->tx_dma_progress = 0; | 1064 | card->tx_dma_progress = 0; |
1065 | gelic_net_release_tx_chain(card, 0); | ||
1066 | /* kick outstanding tx descriptor if any */ | ||
1067 | gelic_net_kick_txdma(card, card->tx_chain.tail); | ||
1027 | spin_unlock_irqrestore(&card->tx_dma_lock, flags); | 1068 | spin_unlock_irqrestore(&card->tx_dma_lock, flags); |
1028 | /* start pending DMA */ | ||
1029 | gelic_net_xmit(NULL, netdev); | ||
1030 | } | 1069 | } |
1031 | return IRQ_HANDLED; | 1070 | return IRQ_HANDLED; |
1032 | } | 1071 | } |
@@ -1068,7 +1107,7 @@ static int gelic_net_open_device(struct gelic_net_card *card) | |||
1068 | } | 1107 | } |
1069 | 1108 | ||
1070 | result = request_irq(card->netdev->irq, gelic_net_interrupt, | 1109 | result = request_irq(card->netdev->irq, gelic_net_interrupt, |
1071 | IRQF_DISABLED, "gelic network", card->netdev); | 1110 | IRQF_DISABLED, card->netdev->name, card->netdev); |
1072 | 1111 | ||
1073 | if (result) { | 1112 | if (result) { |
1074 | dev_info(ctodev(card), "%s:%d: request_irq failed (%d)\n", | 1113 | dev_info(ctodev(card), "%s:%d: request_irq failed (%d)\n", |
@@ -1107,7 +1146,7 @@ static int gelic_net_open(struct net_device *netdev) | |||
1107 | card->descr, GELIC_NET_TX_DESCRIPTORS)) | 1146 | card->descr, GELIC_NET_TX_DESCRIPTORS)) |
1108 | goto alloc_tx_failed; | 1147 | goto alloc_tx_failed; |
1109 | if (gelic_net_init_chain(card, &card->rx_chain, | 1148 | if (gelic_net_init_chain(card, &card->rx_chain, |
1110 | card->descr + GELIC_NET_RX_DESCRIPTORS, | 1149 | card->descr + GELIC_NET_TX_DESCRIPTORS, |
1111 | GELIC_NET_RX_DESCRIPTORS)) | 1150 | GELIC_NET_RX_DESCRIPTORS)) |
1112 | goto alloc_rx_failed; | 1151 | goto alloc_rx_failed; |
1113 | 1152 | ||
@@ -1129,7 +1168,6 @@ static int gelic_net_open(struct net_device *netdev) | |||
1129 | 1168 | ||
1130 | netif_start_queue(netdev); | 1169 | netif_start_queue(netdev); |
1131 | netif_carrier_on(netdev); | 1170 | netif_carrier_on(netdev); |
1132 | netif_poll_enable(netdev); | ||
1133 | 1171 | ||
1134 | return 0; | 1172 | return 0; |
1135 | 1173 | ||
@@ -1141,7 +1179,6 @@ alloc_tx_failed: | |||
1141 | return -ENOMEM; | 1179 | return -ENOMEM; |
1142 | } | 1180 | } |
1143 | 1181 | ||
1144 | #ifdef GELIC_NET_ETHTOOL | ||
1145 | static void gelic_net_get_drvinfo (struct net_device *netdev, | 1182 | static void gelic_net_get_drvinfo (struct net_device *netdev, |
1146 | struct ethtool_drvinfo *info) | 1183 | struct ethtool_drvinfo *info) |
1147 | { | 1184 | { |
@@ -1261,7 +1298,6 @@ static struct ethtool_ops gelic_net_ethtool_ops = { | |||
1261 | .get_rx_csum = gelic_net_get_rx_csum, | 1298 | .get_rx_csum = gelic_net_get_rx_csum, |
1262 | .set_rx_csum = gelic_net_set_rx_csum, | 1299 | .set_rx_csum = gelic_net_set_rx_csum, |
1263 | }; | 1300 | }; |
1264 | #endif | ||
1265 | 1301 | ||
1266 | /** | 1302 | /** |
1267 | * gelic_net_tx_timeout_task - task scheduled by the watchdog timeout | 1303 | * gelic_net_tx_timeout_task - task scheduled by the watchdog timeout |
@@ -1320,7 +1356,6 @@ static void gelic_net_setup_netdev_ops(struct net_device *netdev) | |||
1320 | netdev->open = &gelic_net_open; | 1356 | netdev->open = &gelic_net_open; |
1321 | netdev->stop = &gelic_net_stop; | 1357 | netdev->stop = &gelic_net_stop; |
1322 | netdev->hard_start_xmit = &gelic_net_xmit; | 1358 | netdev->hard_start_xmit = &gelic_net_xmit; |
1323 | netdev->get_stats = &gelic_net_get_stats; | ||
1324 | netdev->set_multicast_list = &gelic_net_set_multi; | 1359 | netdev->set_multicast_list = &gelic_net_set_multi; |
1325 | netdev->change_mtu = &gelic_net_change_mtu; | 1360 | netdev->change_mtu = &gelic_net_change_mtu; |
1326 | /* tx watchdog */ | 1361 | /* tx watchdog */ |
@@ -1329,9 +1364,7 @@ static void gelic_net_setup_netdev_ops(struct net_device *netdev) | |||
1329 | /* NAPI */ | 1364 | /* NAPI */ |
1330 | netdev->poll = &gelic_net_poll; | 1365 | netdev->poll = &gelic_net_poll; |
1331 | netdev->weight = GELIC_NET_NAPI_WEIGHT; | 1366 | netdev->weight = GELIC_NET_NAPI_WEIGHT; |
1332 | #ifdef GELIC_NET_ETHTOOL | ||
1333 | netdev->ethtool_ops = &gelic_net_ethtool_ops; | 1367 | netdev->ethtool_ops = &gelic_net_ethtool_ops; |
1334 | #endif | ||
1335 | } | 1368 | } |
1336 | 1369 | ||
1337 | /** | 1370 | /** |
diff --git a/drivers/net/ps3_gelic_net.h b/drivers/net/ps3_gelic_net.h index 5e1c28654e16..a9c4c4fc2547 100644 --- a/drivers/net/ps3_gelic_net.h +++ b/drivers/net/ps3_gelic_net.h | |||
@@ -28,21 +28,12 @@ | |||
28 | #ifndef _GELIC_NET_H | 28 | #ifndef _GELIC_NET_H |
29 | #define _GELIC_NET_H | 29 | #define _GELIC_NET_H |
30 | 30 | ||
31 | #define GELIC_NET_DRV_NAME "Gelic Network Driver" | ||
32 | #define GELIC_NET_DRV_VERSION "1.0" | ||
33 | |||
34 | #define GELIC_NET_ETHTOOL /* use ethtool */ | ||
35 | |||
36 | /* ioctl */ | ||
37 | #define GELIC_NET_GET_MODE (SIOCDEVPRIVATE + 0) | ||
38 | #define GELIC_NET_SET_MODE (SIOCDEVPRIVATE + 1) | ||
39 | |||
40 | /* descriptors */ | 31 | /* descriptors */ |
41 | #define GELIC_NET_RX_DESCRIPTORS 128 /* num of descriptors */ | 32 | #define GELIC_NET_RX_DESCRIPTORS 128 /* num of descriptors */ |
42 | #define GELIC_NET_TX_DESCRIPTORS 128 /* num of descriptors */ | 33 | #define GELIC_NET_TX_DESCRIPTORS 128 /* num of descriptors */ |
43 | 34 | ||
44 | #define GELIC_NET_MAX_MTU 2308 | 35 | #define GELIC_NET_MAX_MTU VLAN_ETH_FRAME_LEN |
45 | #define GELIC_NET_MIN_MTU 64 | 36 | #define GELIC_NET_MIN_MTU VLAN_ETH_ZLEN |
46 | #define GELIC_NET_RXBUF_ALIGN 128 | 37 | #define GELIC_NET_RXBUF_ALIGN 128 |
47 | #define GELIC_NET_RX_CSUM_DEFAULT 1 /* hw chksum */ | 38 | #define GELIC_NET_RX_CSUM_DEFAULT 1 /* hw chksum */ |
48 | #define GELIC_NET_WATCHDOG_TIMEOUT 5*HZ | 39 | #define GELIC_NET_WATCHDOG_TIMEOUT 5*HZ |
@@ -90,7 +81,8 @@ enum gelic_net_int1_status { | |||
90 | */ | 81 | */ |
91 | #define GELIC_NET_RXVLNPKT 0x00200000 /* VLAN packet */ | 82 | #define GELIC_NET_RXVLNPKT 0x00200000 /* VLAN packet */ |
92 | /* bit 20..16 reserved */ | 83 | /* bit 20..16 reserved */ |
93 | #define GELIC_NET_RXRECNUM 0x0000ff00 /* reception receipt number */ | 84 | #define GELIC_NET_RXRRECNUM 0x0000ff00 /* reception receipt number */ |
85 | #define GELIC_NET_RXRRECNUM_SHIFT 8 | ||
94 | /* bit 7..0 reserved */ | 86 | /* bit 7..0 reserved */ |
95 | 87 | ||
96 | #define GELIC_NET_TXDESC_TAIL 0 | 88 | #define GELIC_NET_TXDESC_TAIL 0 |
@@ -133,19 +125,19 @@ enum gelic_net_int1_status { | |||
133 | * interrupt status */ | 125 | * interrupt status */ |
134 | 126 | ||
135 | #define GELIC_NET_DMAC_CMDSTAT_CHAIN_END 0x00000002 /* RXDCEIS:DMA stopped */ | 127 | #define GELIC_NET_DMAC_CMDSTAT_CHAIN_END 0x00000002 /* RXDCEIS:DMA stopped */ |
136 | #define GELIC_NET_DMAC_CMDSTAT_NOT_IN_USE 0xb0000000 | ||
137 | #define GELIC_NET_DESCR_IND_PROC_SHIFT 28 | 128 | #define GELIC_NET_DESCR_IND_PROC_SHIFT 28 |
138 | #define GELIC_NET_DESCR_IND_PROC_MASKO 0x0fffffff | 129 | #define GELIC_NET_DESCR_IND_PROC_MASKO 0x0fffffff |
139 | 130 | ||
140 | 131 | ||
141 | enum gelic_net_descr_status { | 132 | enum gelic_net_descr_status { |
142 | GELIC_NET_DESCR_COMPLETE = 0x00, /* used in rx and tx */ | 133 | GELIC_NET_DESCR_COMPLETE = 0x00, /* used in tx */ |
134 | GELIC_NET_DESCR_BUFFER_FULL = 0x00, /* used in rx */ | ||
143 | GELIC_NET_DESCR_RESPONSE_ERROR = 0x01, /* used in rx and tx */ | 135 | GELIC_NET_DESCR_RESPONSE_ERROR = 0x01, /* used in rx and tx */ |
144 | GELIC_NET_DESCR_PROTECTION_ERROR = 0x02, /* used in rx and tx */ | 136 | GELIC_NET_DESCR_PROTECTION_ERROR = 0x02, /* used in rx and tx */ |
145 | GELIC_NET_DESCR_FRAME_END = 0x04, /* used in rx */ | 137 | GELIC_NET_DESCR_FRAME_END = 0x04, /* used in rx */ |
146 | GELIC_NET_DESCR_FORCE_END = 0x05, /* used in rx and tx */ | 138 | GELIC_NET_DESCR_FORCE_END = 0x05, /* used in rx and tx */ |
147 | GELIC_NET_DESCR_CARDOWNED = 0x0a, /* used in rx and tx */ | 139 | GELIC_NET_DESCR_CARDOWNED = 0x0a, /* used in rx and tx */ |
148 | GELIC_NET_DESCR_NOT_IN_USE /* any other value */ | 140 | GELIC_NET_DESCR_NOT_IN_USE = 0x0b /* any other value */ |
149 | }; | 141 | }; |
150 | /* for lv1_net_control */ | 142 | /* for lv1_net_control */ |
151 | #define GELIC_NET_GET_MAC_ADDRESS 0x0000000000000001 | 143 | #define GELIC_NET_GET_MAC_ADDRESS 0x0000000000000001 |
@@ -216,10 +208,10 @@ struct gelic_net_card { | |||
216 | 208 | ||
217 | struct gelic_net_descr_chain tx_chain; | 209 | struct gelic_net_descr_chain tx_chain; |
218 | struct gelic_net_descr_chain rx_chain; | 210 | struct gelic_net_descr_chain rx_chain; |
211 | int rx_dma_restart_required; | ||
219 | /* gurad dmac descriptor chain*/ | 212 | /* gurad dmac descriptor chain*/ |
220 | spinlock_t chain_lock; | 213 | spinlock_t chain_lock; |
221 | 214 | ||
222 | struct net_device_stats netdev_stats; | ||
223 | int rx_csum; | 215 | int rx_csum; |
224 | /* guard tx_dma_progress */ | 216 | /* guard tx_dma_progress */ |
225 | spinlock_t tx_dma_lock; | 217 | spinlock_t tx_dma_lock; |
diff --git a/drivers/net/ucc_geth.c b/drivers/net/ucc_geth.c index e4736a3b1b7a..12e01b24105a 100644 --- a/drivers/net/ucc_geth.c +++ b/drivers/net/ucc_geth.c | |||
@@ -43,10 +43,6 @@ | |||
43 | 43 | ||
44 | #undef DEBUG | 44 | #undef DEBUG |
45 | 45 | ||
46 | #define DRV_DESC "QE UCC Gigabit Ethernet Controller" | ||
47 | #define DRV_NAME "ucc_geth" | ||
48 | #define DRV_VERSION "1.1" | ||
49 | |||
50 | #define ugeth_printk(level, format, arg...) \ | 46 | #define ugeth_printk(level, format, arg...) \ |
51 | printk(level format "\n", ## arg) | 47 | printk(level format "\n", ## arg) |
52 | 48 | ||
@@ -64,9 +60,19 @@ | |||
64 | #else | 60 | #else |
65 | #define ugeth_vdbg(fmt, args...) do { } while (0) | 61 | #define ugeth_vdbg(fmt, args...) do { } while (0) |
66 | #endif /* UGETH_VERBOSE_DEBUG */ | 62 | #endif /* UGETH_VERBOSE_DEBUG */ |
63 | #define UGETH_MSG_DEFAULT (NETIF_MSG_IFUP << 1 ) - 1 | ||
67 | 64 | ||
65 | void uec_set_ethtool_ops(struct net_device *netdev); | ||
66 | |||
68 | static DEFINE_SPINLOCK(ugeth_lock); | 67 | static DEFINE_SPINLOCK(ugeth_lock); |
69 | 68 | ||
69 | static struct { | ||
70 | u32 msg_enable; | ||
71 | } debug = { -1 }; | ||
72 | |||
73 | module_param_named(debug, debug.msg_enable, int, 0); | ||
74 | MODULE_PARM_DESC(debug, "Debug verbosity level (0=none, ..., 0xffff=all)"); | ||
75 | |||
70 | static struct ucc_geth_info ugeth_primary_info = { | 76 | static struct ucc_geth_info ugeth_primary_info = { |
71 | .uf_info = { | 77 | .uf_info = { |
72 | .bd_mem_part = MEM_PART_SYSTEM, | 78 | .bd_mem_part = MEM_PART_SYSTEM, |
@@ -104,6 +110,7 @@ static struct ucc_geth_info ugeth_primary_info = { | |||
104 | .maxRetransmission = 0xf, | 110 | .maxRetransmission = 0xf, |
105 | .collisionWindow = 0x37, | 111 | .collisionWindow = 0x37, |
106 | .receiveFlowControl = 1, | 112 | .receiveFlowControl = 1, |
113 | .transmitFlowControl = 1, | ||
107 | .maxGroupAddrInHash = 4, | 114 | .maxGroupAddrInHash = 4, |
108 | .maxIndAddrInHash = 4, | 115 | .maxIndAddrInHash = 4, |
109 | .prel = 7, | 116 | .prel = 7, |
@@ -139,7 +146,9 @@ static struct ucc_geth_info ugeth_primary_info = { | |||
139 | .numStationAddresses = UCC_GETH_NUM_OF_STATION_ADDRESSES_1, | 146 | .numStationAddresses = UCC_GETH_NUM_OF_STATION_ADDRESSES_1, |
140 | .largestexternallookupkeysize = | 147 | .largestexternallookupkeysize = |
141 | QE_FLTR_LARGEST_EXTERNAL_TABLE_LOOKUP_KEY_SIZE_NONE, | 148 | QE_FLTR_LARGEST_EXTERNAL_TABLE_LOOKUP_KEY_SIZE_NONE, |
142 | .statisticsMode = UCC_GETH_STATISTICS_GATHERING_MODE_NONE, | 149 | .statisticsMode = UCC_GETH_STATISTICS_GATHERING_MODE_HARDWARE | |
150 | UCC_GETH_STATISTICS_GATHERING_MODE_FIRMWARE_TX | | ||
151 | UCC_GETH_STATISTICS_GATHERING_MODE_FIRMWARE_RX, | ||
143 | .vlanOperationTagged = UCC_GETH_VLAN_OPERATION_TAGGED_NOP, | 152 | .vlanOperationTagged = UCC_GETH_VLAN_OPERATION_TAGGED_NOP, |
144 | .vlanOperationNonTagged = UCC_GETH_VLAN_OPERATION_NON_TAGGED_NOP, | 153 | .vlanOperationNonTagged = UCC_GETH_VLAN_OPERATION_NON_TAGGED_NOP, |
145 | .rxQoSMode = UCC_GETH_QOS_MODE_DEFAULT, | 154 | .rxQoSMode = UCC_GETH_QOS_MODE_DEFAULT, |
@@ -281,7 +290,8 @@ static int fill_init_enet_entries(struct ucc_geth_private *ugeth, | |||
281 | 290 | ||
282 | for (i = 0; i < num_entries; i++) { | 291 | for (i = 0; i < num_entries; i++) { |
283 | if ((snum = qe_get_snum()) < 0) { | 292 | if ((snum = qe_get_snum()) < 0) { |
284 | ugeth_err("fill_init_enet_entries: Can not get SNUM."); | 293 | if (netif_msg_ifup(ugeth)) |
294 | ugeth_err("fill_init_enet_entries: Can not get SNUM."); | ||
285 | return snum; | 295 | return snum; |
286 | } | 296 | } |
287 | if ((i == 0) && skip_page_for_first_entry) | 297 | if ((i == 0) && skip_page_for_first_entry) |
@@ -291,8 +301,8 @@ static int fill_init_enet_entries(struct ucc_geth_private *ugeth, | |||
291 | init_enet_offset = | 301 | init_enet_offset = |
292 | qe_muram_alloc(thread_size, thread_alignment); | 302 | qe_muram_alloc(thread_size, thread_alignment); |
293 | if (IS_ERR_VALUE(init_enet_offset)) { | 303 | if (IS_ERR_VALUE(init_enet_offset)) { |
294 | ugeth_err | 304 | if (netif_msg_ifup(ugeth)) |
295 | ("fill_init_enet_entries: Can not allocate DPRAM memory."); | 305 | ugeth_err("fill_init_enet_entries: Can not allocate DPRAM memory."); |
296 | qe_put_snum((u8) snum); | 306 | qe_put_snum((u8) snum); |
297 | return -ENOMEM; | 307 | return -ENOMEM; |
298 | } | 308 | } |
@@ -1200,7 +1210,7 @@ static int init_inter_frame_gap_params(u8 non_btb_cs_ipg, | |||
1200 | return 0; | 1210 | return 0; |
1201 | } | 1211 | } |
1202 | 1212 | ||
1203 | static int init_flow_control_params(u32 automatic_flow_control_mode, | 1213 | int init_flow_control_params(u32 automatic_flow_control_mode, |
1204 | int rx_flow_control_enable, | 1214 | int rx_flow_control_enable, |
1205 | int tx_flow_control_enable, | 1215 | int tx_flow_control_enable, |
1206 | u16 pause_period, | 1216 | u16 pause_period, |
@@ -1486,9 +1496,9 @@ static int adjust_enet_interface(struct ucc_geth_private *ugeth) | |||
1486 | 1496 | ||
1487 | ret_val = init_preamble_length(ug_info->prel, &ug_regs->maccfg2); | 1497 | ret_val = init_preamble_length(ug_info->prel, &ug_regs->maccfg2); |
1488 | if (ret_val != 0) { | 1498 | if (ret_val != 0) { |
1489 | ugeth_err | 1499 | if (netif_msg_probe(ugeth)) |
1490 | ("%s: Preamble length must be between 3 and 7 inclusive.", | 1500 | ugeth_err("%s: Preamble length must be between 3 and 7 inclusive.", |
1491 | __FUNCTION__); | 1501 | __FUNCTION__); |
1492 | return ret_val; | 1502 | return ret_val; |
1493 | } | 1503 | } |
1494 | 1504 | ||
@@ -1726,7 +1736,8 @@ static int ugeth_enable(struct ucc_geth_private *ugeth, enum comm_dir mode) | |||
1726 | 1736 | ||
1727 | /* check if the UCC number is in range. */ | 1737 | /* check if the UCC number is in range. */ |
1728 | if (ugeth->ug_info->uf_info.ucc_num >= UCC_MAX_NUM) { | 1738 | if (ugeth->ug_info->uf_info.ucc_num >= UCC_MAX_NUM) { |
1729 | ugeth_err("%s: ucc_num out of range.", __FUNCTION__); | 1739 | if (netif_msg_probe(ugeth)) |
1740 | ugeth_err("%s: ucc_num out of range.", __FUNCTION__); | ||
1730 | return -EINVAL; | 1741 | return -EINVAL; |
1731 | } | 1742 | } |
1732 | 1743 | ||
@@ -1754,7 +1765,8 @@ static int ugeth_disable(struct ucc_geth_private * ugeth, enum comm_dir mode) | |||
1754 | 1765 | ||
1755 | /* check if the UCC number is in range. */ | 1766 | /* check if the UCC number is in range. */ |
1756 | if (ugeth->ug_info->uf_info.ucc_num >= UCC_MAX_NUM) { | 1767 | if (ugeth->ug_info->uf_info.ucc_num >= UCC_MAX_NUM) { |
1757 | ugeth_err("%s: ucc_num out of range.", __FUNCTION__); | 1768 | if (netif_msg_probe(ugeth)) |
1769 | ugeth_err("%s: ucc_num out of range.", __FUNCTION__); | ||
1758 | return -EINVAL; | 1770 | return -EINVAL; |
1759 | } | 1771 | } |
1760 | 1772 | ||
@@ -2306,7 +2318,9 @@ static int ucc_struct_init(struct ucc_geth_private *ugeth) | |||
2306 | 2318 | ||
2307 | if (!((uf_info->bd_mem_part == MEM_PART_SYSTEM) || | 2319 | if (!((uf_info->bd_mem_part == MEM_PART_SYSTEM) || |
2308 | (uf_info->bd_mem_part == MEM_PART_MURAM))) { | 2320 | (uf_info->bd_mem_part == MEM_PART_MURAM))) { |
2309 | ugeth_err("%s: Bad memory partition value.", __FUNCTION__); | 2321 | if (netif_msg_probe(ugeth)) |
2322 | ugeth_err("%s: Bad memory partition value.", | ||
2323 | __FUNCTION__); | ||
2310 | return -EINVAL; | 2324 | return -EINVAL; |
2311 | } | 2325 | } |
2312 | 2326 | ||
@@ -2315,9 +2329,10 @@ static int ucc_struct_init(struct ucc_geth_private *ugeth) | |||
2315 | if ((ug_info->bdRingLenRx[i] < UCC_GETH_RX_BD_RING_SIZE_MIN) || | 2329 | if ((ug_info->bdRingLenRx[i] < UCC_GETH_RX_BD_RING_SIZE_MIN) || |
2316 | (ug_info->bdRingLenRx[i] % | 2330 | (ug_info->bdRingLenRx[i] % |
2317 | UCC_GETH_RX_BD_RING_SIZE_ALIGNMENT)) { | 2331 | UCC_GETH_RX_BD_RING_SIZE_ALIGNMENT)) { |
2318 | ugeth_err | 2332 | if (netif_msg_probe(ugeth)) |
2319 | ("%s: Rx BD ring length must be multiple of 4," | 2333 | ugeth_err |
2320 | " no smaller than 8.", __FUNCTION__); | 2334 | ("%s: Rx BD ring length must be multiple of 4, no smaller than 8.", |
2335 | __FUNCTION__); | ||
2321 | return -EINVAL; | 2336 | return -EINVAL; |
2322 | } | 2337 | } |
2323 | } | 2338 | } |
@@ -2325,9 +2340,10 @@ static int ucc_struct_init(struct ucc_geth_private *ugeth) | |||
2325 | /* Tx BD lengths */ | 2340 | /* Tx BD lengths */ |
2326 | for (i = 0; i < ug_info->numQueuesTx; i++) { | 2341 | for (i = 0; i < ug_info->numQueuesTx; i++) { |
2327 | if (ug_info->bdRingLenTx[i] < UCC_GETH_TX_BD_RING_SIZE_MIN) { | 2342 | if (ug_info->bdRingLenTx[i] < UCC_GETH_TX_BD_RING_SIZE_MIN) { |
2328 | ugeth_err | 2343 | if (netif_msg_probe(ugeth)) |
2329 | ("%s: Tx BD ring length must be no smaller than 2.", | 2344 | ugeth_err |
2330 | __FUNCTION__); | 2345 | ("%s: Tx BD ring length must be no smaller than 2.", |
2346 | __FUNCTION__); | ||
2331 | return -EINVAL; | 2347 | return -EINVAL; |
2332 | } | 2348 | } |
2333 | } | 2349 | } |
@@ -2335,31 +2351,35 @@ static int ucc_struct_init(struct ucc_geth_private *ugeth) | |||
2335 | /* mrblr */ | 2351 | /* mrblr */ |
2336 | if ((uf_info->max_rx_buf_length == 0) || | 2352 | if ((uf_info->max_rx_buf_length == 0) || |
2337 | (uf_info->max_rx_buf_length % UCC_GETH_MRBLR_ALIGNMENT)) { | 2353 | (uf_info->max_rx_buf_length % UCC_GETH_MRBLR_ALIGNMENT)) { |
2338 | ugeth_err | 2354 | if (netif_msg_probe(ugeth)) |
2339 | ("%s: max_rx_buf_length must be non-zero multiple of 128.", | 2355 | ugeth_err |
2340 | __FUNCTION__); | 2356 | ("%s: max_rx_buf_length must be non-zero multiple of 128.", |
2357 | __FUNCTION__); | ||
2341 | return -EINVAL; | 2358 | return -EINVAL; |
2342 | } | 2359 | } |
2343 | 2360 | ||
2344 | /* num Tx queues */ | 2361 | /* num Tx queues */ |
2345 | if (ug_info->numQueuesTx > NUM_TX_QUEUES) { | 2362 | if (ug_info->numQueuesTx > NUM_TX_QUEUES) { |
2346 | ugeth_err("%s: number of tx queues too large.", __FUNCTION__); | 2363 | if (netif_msg_probe(ugeth)) |
2364 | ugeth_err("%s: number of tx queues too large.", __FUNCTION__); | ||
2347 | return -EINVAL; | 2365 | return -EINVAL; |
2348 | } | 2366 | } |
2349 | 2367 | ||
2350 | /* num Rx queues */ | 2368 | /* num Rx queues */ |
2351 | if (ug_info->numQueuesRx > NUM_RX_QUEUES) { | 2369 | if (ug_info->numQueuesRx > NUM_RX_QUEUES) { |
2352 | ugeth_err("%s: number of rx queues too large.", __FUNCTION__); | 2370 | if (netif_msg_probe(ugeth)) |
2371 | ugeth_err("%s: number of rx queues too large.", __FUNCTION__); | ||
2353 | return -EINVAL; | 2372 | return -EINVAL; |
2354 | } | 2373 | } |
2355 | 2374 | ||
2356 | /* l2qt */ | 2375 | /* l2qt */ |
2357 | for (i = 0; i < UCC_GETH_VLAN_PRIORITY_MAX; i++) { | 2376 | for (i = 0; i < UCC_GETH_VLAN_PRIORITY_MAX; i++) { |
2358 | if (ug_info->l2qt[i] >= ug_info->numQueuesRx) { | 2377 | if (ug_info->l2qt[i] >= ug_info->numQueuesRx) { |
2359 | ugeth_err | 2378 | if (netif_msg_probe(ugeth)) |
2360 | ("%s: VLAN priority table entry must not be" | 2379 | ugeth_err |
2361 | " larger than number of Rx queues.", | 2380 | ("%s: VLAN priority table entry must not be" |
2362 | __FUNCTION__); | 2381 | " larger than number of Rx queues.", |
2382 | __FUNCTION__); | ||
2363 | return -EINVAL; | 2383 | return -EINVAL; |
2364 | } | 2384 | } |
2365 | } | 2385 | } |
@@ -2367,26 +2387,29 @@ static int ucc_struct_init(struct ucc_geth_private *ugeth) | |||
2367 | /* l3qt */ | 2387 | /* l3qt */ |
2368 | for (i = 0; i < UCC_GETH_IP_PRIORITY_MAX; i++) { | 2388 | for (i = 0; i < UCC_GETH_IP_PRIORITY_MAX; i++) { |
2369 | if (ug_info->l3qt[i] >= ug_info->numQueuesRx) { | 2389 | if (ug_info->l3qt[i] >= ug_info->numQueuesRx) { |
2370 | ugeth_err | 2390 | if (netif_msg_probe(ugeth)) |
2371 | ("%s: IP priority table entry must not be" | 2391 | ugeth_err |
2372 | " larger than number of Rx queues.", | 2392 | ("%s: IP priority table entry must not be" |
2373 | __FUNCTION__); | 2393 | " larger than number of Rx queues.", |
2394 | __FUNCTION__); | ||
2374 | return -EINVAL; | 2395 | return -EINVAL; |
2375 | } | 2396 | } |
2376 | } | 2397 | } |
2377 | 2398 | ||
2378 | if (ug_info->cam && !ug_info->ecamptr) { | 2399 | if (ug_info->cam && !ug_info->ecamptr) { |
2379 | ugeth_err("%s: If cam mode is chosen, must supply cam ptr.", | 2400 | if (netif_msg_probe(ugeth)) |
2380 | __FUNCTION__); | 2401 | ugeth_err("%s: If cam mode is chosen, must supply cam ptr.", |
2402 | __FUNCTION__); | ||
2381 | return -EINVAL; | 2403 | return -EINVAL; |
2382 | } | 2404 | } |
2383 | 2405 | ||
2384 | if ((ug_info->numStationAddresses != | 2406 | if ((ug_info->numStationAddresses != |
2385 | UCC_GETH_NUM_OF_STATION_ADDRESSES_1) | 2407 | UCC_GETH_NUM_OF_STATION_ADDRESSES_1) |
2386 | && ug_info->rxExtendedFiltering) { | 2408 | && ug_info->rxExtendedFiltering) { |
2387 | ugeth_err("%s: Number of station addresses greater than 1 " | 2409 | if (netif_msg_probe(ugeth)) |
2388 | "not allowed in extended parsing mode.", | 2410 | ugeth_err("%s: Number of station addresses greater than 1 " |
2389 | __FUNCTION__); | 2411 | "not allowed in extended parsing mode.", |
2412 | __FUNCTION__); | ||
2390 | return -EINVAL; | 2413 | return -EINVAL; |
2391 | } | 2414 | } |
2392 | 2415 | ||
@@ -2399,7 +2422,8 @@ static int ucc_struct_init(struct ucc_geth_private *ugeth) | |||
2399 | uf_info->uccm_mask |= (UCCE_TXBF_SINGLE_MASK << i); | 2422 | uf_info->uccm_mask |= (UCCE_TXBF_SINGLE_MASK << i); |
2400 | /* Initialize the general fast UCC block. */ | 2423 | /* Initialize the general fast UCC block. */ |
2401 | if (ucc_fast_init(uf_info, &ugeth->uccf)) { | 2424 | if (ucc_fast_init(uf_info, &ugeth->uccf)) { |
2402 | ugeth_err("%s: Failed to init uccf.", __FUNCTION__); | 2425 | if (netif_msg_probe(ugeth)) |
2426 | ugeth_err("%s: Failed to init uccf.", __FUNCTION__); | ||
2403 | ucc_geth_memclean(ugeth); | 2427 | ucc_geth_memclean(ugeth); |
2404 | return -ENOMEM; | 2428 | return -ENOMEM; |
2405 | } | 2429 | } |
@@ -2452,7 +2476,9 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
2452 | numThreadsRxNumerical = 8; | 2476 | numThreadsRxNumerical = 8; |
2453 | break; | 2477 | break; |
2454 | default: | 2478 | default: |
2455 | ugeth_err("%s: Bad number of Rx threads value.", __FUNCTION__); | 2479 | if (netif_msg_ifup(ugeth)) |
2480 | ugeth_err("%s: Bad number of Rx threads value.", | ||
2481 | __FUNCTION__); | ||
2456 | ucc_geth_memclean(ugeth); | 2482 | ucc_geth_memclean(ugeth); |
2457 | return -EINVAL; | 2483 | return -EINVAL; |
2458 | break; | 2484 | break; |
@@ -2475,7 +2501,9 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
2475 | numThreadsTxNumerical = 8; | 2501 | numThreadsTxNumerical = 8; |
2476 | break; | 2502 | break; |
2477 | default: | 2503 | default: |
2478 | ugeth_err("%s: Bad number of Tx threads value.", __FUNCTION__); | 2504 | if (netif_msg_ifup(ugeth)) |
2505 | ugeth_err("%s: Bad number of Tx threads value.", | ||
2506 | __FUNCTION__); | ||
2479 | ucc_geth_memclean(ugeth); | 2507 | ucc_geth_memclean(ugeth); |
2480 | return -EINVAL; | 2508 | return -EINVAL; |
2481 | break; | 2509 | break; |
@@ -2507,7 +2535,7 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
2507 | /* For more details see the hardware spec. */ | 2535 | /* For more details see the hardware spec. */ |
2508 | init_flow_control_params(ug_info->aufc, | 2536 | init_flow_control_params(ug_info->aufc, |
2509 | ug_info->receiveFlowControl, | 2537 | ug_info->receiveFlowControl, |
2510 | 1, | 2538 | ug_info->transmitFlowControl, |
2511 | ug_info->pausePeriod, | 2539 | ug_info->pausePeriod, |
2512 | ug_info->extensionField, | 2540 | ug_info->extensionField, |
2513 | &uf_regs->upsmr, | 2541 | &uf_regs->upsmr, |
@@ -2527,8 +2555,9 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
2527 | ug_info->backToBackInterFrameGap, | 2555 | ug_info->backToBackInterFrameGap, |
2528 | &ug_regs->ipgifg); | 2556 | &ug_regs->ipgifg); |
2529 | if (ret_val != 0) { | 2557 | if (ret_val != 0) { |
2530 | ugeth_err("%s: IPGIFG initialization parameter too large.", | 2558 | if (netif_msg_ifup(ugeth)) |
2531 | __FUNCTION__); | 2559 | ugeth_err("%s: IPGIFG initialization parameter too large.", |
2560 | __FUNCTION__); | ||
2532 | ucc_geth_memclean(ugeth); | 2561 | ucc_geth_memclean(ugeth); |
2533 | return ret_val; | 2562 | return ret_val; |
2534 | } | 2563 | } |
@@ -2544,7 +2573,8 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
2544 | ug_info->collisionWindow, | 2573 | ug_info->collisionWindow, |
2545 | &ug_regs->hafdup); | 2574 | &ug_regs->hafdup); |
2546 | if (ret_val != 0) { | 2575 | if (ret_val != 0) { |
2547 | ugeth_err("%s: Half Duplex initialization parameter too large.", | 2576 | if (netif_msg_ifup(ugeth)) |
2577 | ugeth_err("%s: Half Duplex initialization parameter too large.", | ||
2548 | __FUNCTION__); | 2578 | __FUNCTION__); |
2549 | ucc_geth_memclean(ugeth); | 2579 | ucc_geth_memclean(ugeth); |
2550 | return ret_val; | 2580 | return ret_val; |
@@ -2597,9 +2627,10 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
2597 | tx_bd_ring_offset[j]); | 2627 | tx_bd_ring_offset[j]); |
2598 | } | 2628 | } |
2599 | if (!ugeth->p_tx_bd_ring[j]) { | 2629 | if (!ugeth->p_tx_bd_ring[j]) { |
2600 | ugeth_err | 2630 | if (netif_msg_ifup(ugeth)) |
2601 | ("%s: Can not allocate memory for Tx bd rings.", | 2631 | ugeth_err |
2602 | __FUNCTION__); | 2632 | ("%s: Can not allocate memory for Tx bd rings.", |
2633 | __FUNCTION__); | ||
2603 | ucc_geth_memclean(ugeth); | 2634 | ucc_geth_memclean(ugeth); |
2604 | return -ENOMEM; | 2635 | return -ENOMEM; |
2605 | } | 2636 | } |
@@ -2632,9 +2663,10 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
2632 | rx_bd_ring_offset[j]); | 2663 | rx_bd_ring_offset[j]); |
2633 | } | 2664 | } |
2634 | if (!ugeth->p_rx_bd_ring[j]) { | 2665 | if (!ugeth->p_rx_bd_ring[j]) { |
2635 | ugeth_err | 2666 | if (netif_msg_ifup(ugeth)) |
2636 | ("%s: Can not allocate memory for Rx bd rings.", | 2667 | ugeth_err |
2637 | __FUNCTION__); | 2668 | ("%s: Can not allocate memory for Rx bd rings.", |
2669 | __FUNCTION__); | ||
2638 | ucc_geth_memclean(ugeth); | 2670 | ucc_geth_memclean(ugeth); |
2639 | return -ENOMEM; | 2671 | return -ENOMEM; |
2640 | } | 2672 | } |
@@ -2648,8 +2680,9 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
2648 | GFP_KERNEL); | 2680 | GFP_KERNEL); |
2649 | 2681 | ||
2650 | if (ugeth->tx_skbuff[j] == NULL) { | 2682 | if (ugeth->tx_skbuff[j] == NULL) { |
2651 | ugeth_err("%s: Could not allocate tx_skbuff", | 2683 | if (netif_msg_ifup(ugeth)) |
2652 | __FUNCTION__); | 2684 | ugeth_err("%s: Could not allocate tx_skbuff", |
2685 | __FUNCTION__); | ||
2653 | ucc_geth_memclean(ugeth); | 2686 | ucc_geth_memclean(ugeth); |
2654 | return -ENOMEM; | 2687 | return -ENOMEM; |
2655 | } | 2688 | } |
@@ -2679,8 +2712,9 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
2679 | GFP_KERNEL); | 2712 | GFP_KERNEL); |
2680 | 2713 | ||
2681 | if (ugeth->rx_skbuff[j] == NULL) { | 2714 | if (ugeth->rx_skbuff[j] == NULL) { |
2682 | ugeth_err("%s: Could not allocate rx_skbuff", | 2715 | if (netif_msg_ifup(ugeth)) |
2683 | __FUNCTION__); | 2716 | ugeth_err("%s: Could not allocate rx_skbuff", |
2717 | __FUNCTION__); | ||
2684 | ucc_geth_memclean(ugeth); | 2718 | ucc_geth_memclean(ugeth); |
2685 | return -ENOMEM; | 2719 | return -ENOMEM; |
2686 | } | 2720 | } |
@@ -2711,9 +2745,10 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
2711 | qe_muram_alloc(sizeof(struct ucc_geth_tx_global_pram), | 2745 | qe_muram_alloc(sizeof(struct ucc_geth_tx_global_pram), |
2712 | UCC_GETH_TX_GLOBAL_PRAM_ALIGNMENT); | 2746 | UCC_GETH_TX_GLOBAL_PRAM_ALIGNMENT); |
2713 | if (IS_ERR_VALUE(ugeth->tx_glbl_pram_offset)) { | 2747 | if (IS_ERR_VALUE(ugeth->tx_glbl_pram_offset)) { |
2714 | ugeth_err | 2748 | if (netif_msg_ifup(ugeth)) |
2715 | ("%s: Can not allocate DPRAM memory for p_tx_glbl_pram.", | 2749 | ugeth_err |
2716 | __FUNCTION__); | 2750 | ("%s: Can not allocate DPRAM memory for p_tx_glbl_pram.", |
2751 | __FUNCTION__); | ||
2717 | ucc_geth_memclean(ugeth); | 2752 | ucc_geth_memclean(ugeth); |
2718 | return -ENOMEM; | 2753 | return -ENOMEM; |
2719 | } | 2754 | } |
@@ -2733,9 +2768,10 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
2733 | 32 * (numThreadsTxNumerical == 1), | 2768 | 32 * (numThreadsTxNumerical == 1), |
2734 | UCC_GETH_THREAD_DATA_ALIGNMENT); | 2769 | UCC_GETH_THREAD_DATA_ALIGNMENT); |
2735 | if (IS_ERR_VALUE(ugeth->thread_dat_tx_offset)) { | 2770 | if (IS_ERR_VALUE(ugeth->thread_dat_tx_offset)) { |
2736 | ugeth_err | 2771 | if (netif_msg_ifup(ugeth)) |
2737 | ("%s: Can not allocate DPRAM memory for p_thread_data_tx.", | 2772 | ugeth_err |
2738 | __FUNCTION__); | 2773 | ("%s: Can not allocate DPRAM memory for p_thread_data_tx.", |
2774 | __FUNCTION__); | ||
2739 | ucc_geth_memclean(ugeth); | 2775 | ucc_geth_memclean(ugeth); |
2740 | return -ENOMEM; | 2776 | return -ENOMEM; |
2741 | } | 2777 | } |
@@ -2761,9 +2797,10 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
2761 | sizeof(struct ucc_geth_send_queue_qd), | 2797 | sizeof(struct ucc_geth_send_queue_qd), |
2762 | UCC_GETH_SEND_QUEUE_QUEUE_DESCRIPTOR_ALIGNMENT); | 2798 | UCC_GETH_SEND_QUEUE_QUEUE_DESCRIPTOR_ALIGNMENT); |
2763 | if (IS_ERR_VALUE(ugeth->send_q_mem_reg_offset)) { | 2799 | if (IS_ERR_VALUE(ugeth->send_q_mem_reg_offset)) { |
2764 | ugeth_err | 2800 | if (netif_msg_ifup(ugeth)) |
2765 | ("%s: Can not allocate DPRAM memory for p_send_q_mem_reg.", | 2801 | ugeth_err |
2766 | __FUNCTION__); | 2802 | ("%s: Can not allocate DPRAM memory for p_send_q_mem_reg.", |
2803 | __FUNCTION__); | ||
2767 | ucc_geth_memclean(ugeth); | 2804 | ucc_geth_memclean(ugeth); |
2768 | return -ENOMEM; | 2805 | return -ENOMEM; |
2769 | } | 2806 | } |
@@ -2804,9 +2841,10 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
2804 | qe_muram_alloc(sizeof(struct ucc_geth_scheduler), | 2841 | qe_muram_alloc(sizeof(struct ucc_geth_scheduler), |
2805 | UCC_GETH_SCHEDULER_ALIGNMENT); | 2842 | UCC_GETH_SCHEDULER_ALIGNMENT); |
2806 | if (IS_ERR_VALUE(ugeth->scheduler_offset)) { | 2843 | if (IS_ERR_VALUE(ugeth->scheduler_offset)) { |
2807 | ugeth_err | 2844 | if (netif_msg_ifup(ugeth)) |
2808 | ("%s: Can not allocate DPRAM memory for p_scheduler.", | 2845 | ugeth_err |
2809 | __FUNCTION__); | 2846 | ("%s: Can not allocate DPRAM memory for p_scheduler.", |
2847 | __FUNCTION__); | ||
2810 | ucc_geth_memclean(ugeth); | 2848 | ucc_geth_memclean(ugeth); |
2811 | return -ENOMEM; | 2849 | return -ENOMEM; |
2812 | } | 2850 | } |
@@ -2852,9 +2890,11 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
2852 | (struct ucc_geth_tx_firmware_statistics_pram), | 2890 | (struct ucc_geth_tx_firmware_statistics_pram), |
2853 | UCC_GETH_TX_STATISTICS_ALIGNMENT); | 2891 | UCC_GETH_TX_STATISTICS_ALIGNMENT); |
2854 | if (IS_ERR_VALUE(ugeth->tx_fw_statistics_pram_offset)) { | 2892 | if (IS_ERR_VALUE(ugeth->tx_fw_statistics_pram_offset)) { |
2855 | ugeth_err | 2893 | if (netif_msg_ifup(ugeth)) |
2856 | ("%s: Can not allocate DPRAM memory for" | 2894 | ugeth_err |
2857 | " p_tx_fw_statistics_pram.", __FUNCTION__); | 2895 | ("%s: Can not allocate DPRAM memory for" |
2896 | " p_tx_fw_statistics_pram.", | ||
2897 | __FUNCTION__); | ||
2858 | ucc_geth_memclean(ugeth); | 2898 | ucc_geth_memclean(ugeth); |
2859 | return -ENOMEM; | 2899 | return -ENOMEM; |
2860 | } | 2900 | } |
@@ -2891,9 +2931,10 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
2891 | qe_muram_alloc(sizeof(struct ucc_geth_rx_global_pram), | 2931 | qe_muram_alloc(sizeof(struct ucc_geth_rx_global_pram), |
2892 | UCC_GETH_RX_GLOBAL_PRAM_ALIGNMENT); | 2932 | UCC_GETH_RX_GLOBAL_PRAM_ALIGNMENT); |
2893 | if (IS_ERR_VALUE(ugeth->rx_glbl_pram_offset)) { | 2933 | if (IS_ERR_VALUE(ugeth->rx_glbl_pram_offset)) { |
2894 | ugeth_err | 2934 | if (netif_msg_ifup(ugeth)) |
2895 | ("%s: Can not allocate DPRAM memory for p_rx_glbl_pram.", | 2935 | ugeth_err |
2896 | __FUNCTION__); | 2936 | ("%s: Can not allocate DPRAM memory for p_rx_glbl_pram.", |
2937 | __FUNCTION__); | ||
2897 | ucc_geth_memclean(ugeth); | 2938 | ucc_geth_memclean(ugeth); |
2898 | return -ENOMEM; | 2939 | return -ENOMEM; |
2899 | } | 2940 | } |
@@ -2912,9 +2953,10 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
2912 | sizeof(struct ucc_geth_thread_data_rx), | 2953 | sizeof(struct ucc_geth_thread_data_rx), |
2913 | UCC_GETH_THREAD_DATA_ALIGNMENT); | 2954 | UCC_GETH_THREAD_DATA_ALIGNMENT); |
2914 | if (IS_ERR_VALUE(ugeth->thread_dat_rx_offset)) { | 2955 | if (IS_ERR_VALUE(ugeth->thread_dat_rx_offset)) { |
2915 | ugeth_err | 2956 | if (netif_msg_ifup(ugeth)) |
2916 | ("%s: Can not allocate DPRAM memory for p_thread_data_rx.", | 2957 | ugeth_err |
2917 | __FUNCTION__); | 2958 | ("%s: Can not allocate DPRAM memory for p_thread_data_rx.", |
2959 | __FUNCTION__); | ||
2918 | ucc_geth_memclean(ugeth); | 2960 | ucc_geth_memclean(ugeth); |
2919 | return -ENOMEM; | 2961 | return -ENOMEM; |
2920 | } | 2962 | } |
@@ -2935,9 +2977,10 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
2935 | (struct ucc_geth_rx_firmware_statistics_pram), | 2977 | (struct ucc_geth_rx_firmware_statistics_pram), |
2936 | UCC_GETH_RX_STATISTICS_ALIGNMENT); | 2978 | UCC_GETH_RX_STATISTICS_ALIGNMENT); |
2937 | if (IS_ERR_VALUE(ugeth->rx_fw_statistics_pram_offset)) { | 2979 | if (IS_ERR_VALUE(ugeth->rx_fw_statistics_pram_offset)) { |
2938 | ugeth_err | 2980 | if (netif_msg_ifup(ugeth)) |
2939 | ("%s: Can not allocate DPRAM memory for" | 2981 | ugeth_err |
2940 | " p_rx_fw_statistics_pram.", __FUNCTION__); | 2982 | ("%s: Can not allocate DPRAM memory for" |
2983 | " p_rx_fw_statistics_pram.", __FUNCTION__); | ||
2941 | ucc_geth_memclean(ugeth); | 2984 | ucc_geth_memclean(ugeth); |
2942 | return -ENOMEM; | 2985 | return -ENOMEM; |
2943 | } | 2986 | } |
@@ -2957,9 +3000,10 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
2957 | sizeof(struct ucc_geth_rx_interrupt_coalescing_entry) | 3000 | sizeof(struct ucc_geth_rx_interrupt_coalescing_entry) |
2958 | + 4, UCC_GETH_RX_INTERRUPT_COALESCING_ALIGNMENT); | 3001 | + 4, UCC_GETH_RX_INTERRUPT_COALESCING_ALIGNMENT); |
2959 | if (IS_ERR_VALUE(ugeth->rx_irq_coalescing_tbl_offset)) { | 3002 | if (IS_ERR_VALUE(ugeth->rx_irq_coalescing_tbl_offset)) { |
2960 | ugeth_err | 3003 | if (netif_msg_ifup(ugeth)) |
2961 | ("%s: Can not allocate DPRAM memory for" | 3004 | ugeth_err |
2962 | " p_rx_irq_coalescing_tbl.", __FUNCTION__); | 3005 | ("%s: Can not allocate DPRAM memory for" |
3006 | " p_rx_irq_coalescing_tbl.", __FUNCTION__); | ||
2963 | ucc_geth_memclean(ugeth); | 3007 | ucc_geth_memclean(ugeth); |
2964 | return -ENOMEM; | 3008 | return -ENOMEM; |
2965 | } | 3009 | } |
@@ -3025,9 +3069,10 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
3025 | sizeof(struct ucc_geth_rx_prefetched_bds)), | 3069 | sizeof(struct ucc_geth_rx_prefetched_bds)), |
3026 | UCC_GETH_RX_BD_QUEUES_ALIGNMENT); | 3070 | UCC_GETH_RX_BD_QUEUES_ALIGNMENT); |
3027 | if (IS_ERR_VALUE(ugeth->rx_bd_qs_tbl_offset)) { | 3071 | if (IS_ERR_VALUE(ugeth->rx_bd_qs_tbl_offset)) { |
3028 | ugeth_err | 3072 | if (netif_msg_ifup(ugeth)) |
3029 | ("%s: Can not allocate DPRAM memory for p_rx_bd_qs_tbl.", | 3073 | ugeth_err |
3030 | __FUNCTION__); | 3074 | ("%s: Can not allocate DPRAM memory for p_rx_bd_qs_tbl.", |
3075 | __FUNCTION__); | ||
3031 | ucc_geth_memclean(ugeth); | 3076 | ucc_geth_memclean(ugeth); |
3032 | return -ENOMEM; | 3077 | return -ENOMEM; |
3033 | } | 3078 | } |
@@ -3102,8 +3147,9 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
3102 | /* initialize extended filtering */ | 3147 | /* initialize extended filtering */ |
3103 | if (ug_info->rxExtendedFiltering) { | 3148 | if (ug_info->rxExtendedFiltering) { |
3104 | if (!ug_info->extendedFilteringChainPointer) { | 3149 | if (!ug_info->extendedFilteringChainPointer) { |
3105 | ugeth_err("%s: Null Extended Filtering Chain Pointer.", | 3150 | if (netif_msg_ifup(ugeth)) |
3106 | __FUNCTION__); | 3151 | ugeth_err("%s: Null Extended Filtering Chain Pointer.", |
3152 | __FUNCTION__); | ||
3107 | ucc_geth_memclean(ugeth); | 3153 | ucc_geth_memclean(ugeth); |
3108 | return -EINVAL; | 3154 | return -EINVAL; |
3109 | } | 3155 | } |
@@ -3114,9 +3160,10 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
3114 | qe_muram_alloc(sizeof(struct ucc_geth_exf_global_pram), | 3160 | qe_muram_alloc(sizeof(struct ucc_geth_exf_global_pram), |
3115 | UCC_GETH_RX_EXTENDED_FILTERING_GLOBAL_PARAMETERS_ALIGNMENT); | 3161 | UCC_GETH_RX_EXTENDED_FILTERING_GLOBAL_PARAMETERS_ALIGNMENT); |
3116 | if (IS_ERR_VALUE(ugeth->exf_glbl_param_offset)) { | 3162 | if (IS_ERR_VALUE(ugeth->exf_glbl_param_offset)) { |
3117 | ugeth_err | 3163 | if (netif_msg_ifup(ugeth)) |
3118 | ("%s: Can not allocate DPRAM memory for" | 3164 | ugeth_err |
3119 | " p_exf_glbl_param.", __FUNCTION__); | 3165 | ("%s: Can not allocate DPRAM memory for" |
3166 | " p_exf_glbl_param.", __FUNCTION__); | ||
3120 | ucc_geth_memclean(ugeth); | 3167 | ucc_geth_memclean(ugeth); |
3121 | return -ENOMEM; | 3168 | return -ENOMEM; |
3122 | } | 3169 | } |
@@ -3161,9 +3208,10 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
3161 | */ | 3208 | */ |
3162 | if (!(ugeth->p_init_enet_param_shadow = | 3209 | if (!(ugeth->p_init_enet_param_shadow = |
3163 | kmalloc(sizeof(struct ucc_geth_init_pram), GFP_KERNEL))) { | 3210 | kmalloc(sizeof(struct ucc_geth_init_pram), GFP_KERNEL))) { |
3164 | ugeth_err | 3211 | if (netif_msg_ifup(ugeth)) |
3165 | ("%s: Can not allocate memory for" | 3212 | ugeth_err |
3166 | " p_UccInitEnetParamShadows.", __FUNCTION__); | 3213 | ("%s: Can not allocate memory for" |
3214 | " p_UccInitEnetParamShadows.", __FUNCTION__); | ||
3167 | ucc_geth_memclean(ugeth); | 3215 | ucc_geth_memclean(ugeth); |
3168 | return -ENOMEM; | 3216 | return -ENOMEM; |
3169 | } | 3217 | } |
@@ -3196,8 +3244,9 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
3196 | QE_FLTR_LARGEST_EXTERNAL_TABLE_LOOKUP_KEY_SIZE_8_BYTES) | 3244 | QE_FLTR_LARGEST_EXTERNAL_TABLE_LOOKUP_KEY_SIZE_8_BYTES) |
3197 | && (ug_info->largestexternallookupkeysize != | 3245 | && (ug_info->largestexternallookupkeysize != |
3198 | QE_FLTR_LARGEST_EXTERNAL_TABLE_LOOKUP_KEY_SIZE_16_BYTES)) { | 3246 | QE_FLTR_LARGEST_EXTERNAL_TABLE_LOOKUP_KEY_SIZE_16_BYTES)) { |
3199 | ugeth_err("%s: Invalid largest External Lookup Key Size.", | 3247 | if (netif_msg_ifup(ugeth)) |
3200 | __FUNCTION__); | 3248 | ugeth_err("%s: Invalid largest External Lookup Key Size.", |
3249 | __FUNCTION__); | ||
3201 | ucc_geth_memclean(ugeth); | 3250 | ucc_geth_memclean(ugeth); |
3202 | return -EINVAL; | 3251 | return -EINVAL; |
3203 | } | 3252 | } |
@@ -3222,8 +3271,9 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
3222 | /* Rx needs one extra for terminator */ | 3271 | /* Rx needs one extra for terminator */ |
3223 | , size, UCC_GETH_THREAD_RX_PRAM_ALIGNMENT, | 3272 | , size, UCC_GETH_THREAD_RX_PRAM_ALIGNMENT, |
3224 | ug_info->riscRx, 1)) != 0) { | 3273 | ug_info->riscRx, 1)) != 0) { |
3225 | ugeth_err("%s: Can not fill p_init_enet_param_shadow.", | 3274 | if (netif_msg_ifup(ugeth)) |
3226 | __FUNCTION__); | 3275 | ugeth_err("%s: Can not fill p_init_enet_param_shadow.", |
3276 | __FUNCTION__); | ||
3227 | ucc_geth_memclean(ugeth); | 3277 | ucc_geth_memclean(ugeth); |
3228 | return ret_val; | 3278 | return ret_val; |
3229 | } | 3279 | } |
@@ -3237,8 +3287,9 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
3237 | sizeof(struct ucc_geth_thread_tx_pram), | 3287 | sizeof(struct ucc_geth_thread_tx_pram), |
3238 | UCC_GETH_THREAD_TX_PRAM_ALIGNMENT, | 3288 | UCC_GETH_THREAD_TX_PRAM_ALIGNMENT, |
3239 | ug_info->riscTx, 0)) != 0) { | 3289 | ug_info->riscTx, 0)) != 0) { |
3240 | ugeth_err("%s: Can not fill p_init_enet_param_shadow.", | 3290 | if (netif_msg_ifup(ugeth)) |
3241 | __FUNCTION__); | 3291 | ugeth_err("%s: Can not fill p_init_enet_param_shadow.", |
3292 | __FUNCTION__); | ||
3242 | ucc_geth_memclean(ugeth); | 3293 | ucc_geth_memclean(ugeth); |
3243 | return ret_val; | 3294 | return ret_val; |
3244 | } | 3295 | } |
@@ -3246,8 +3297,9 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
3246 | /* Load Rx bds with buffers */ | 3297 | /* Load Rx bds with buffers */ |
3247 | for (i = 0; i < ug_info->numQueuesRx; i++) { | 3298 | for (i = 0; i < ug_info->numQueuesRx; i++) { |
3248 | if ((ret_val = rx_bd_buffer_set(ugeth, (u8) i)) != 0) { | 3299 | if ((ret_val = rx_bd_buffer_set(ugeth, (u8) i)) != 0) { |
3249 | ugeth_err("%s: Can not fill Rx bds with buffers.", | 3300 | if (netif_msg_ifup(ugeth)) |
3250 | __FUNCTION__); | 3301 | ugeth_err("%s: Can not fill Rx bds with buffers.", |
3302 | __FUNCTION__); | ||
3251 | ucc_geth_memclean(ugeth); | 3303 | ucc_geth_memclean(ugeth); |
3252 | return ret_val; | 3304 | return ret_val; |
3253 | } | 3305 | } |
@@ -3256,9 +3308,10 @@ static int ucc_geth_startup(struct ucc_geth_private *ugeth) | |||
3256 | /* Allocate InitEnet command parameter structure */ | 3308 | /* Allocate InitEnet command parameter structure */ |
3257 | init_enet_pram_offset = qe_muram_alloc(sizeof(struct ucc_geth_init_pram), 4); | 3309 | init_enet_pram_offset = qe_muram_alloc(sizeof(struct ucc_geth_init_pram), 4); |
3258 | if (IS_ERR_VALUE(init_enet_pram_offset)) { | 3310 | if (IS_ERR_VALUE(init_enet_pram_offset)) { |
3259 | ugeth_err | 3311 | if (netif_msg_ifup(ugeth)) |
3260 | ("%s: Can not allocate DPRAM memory for p_init_enet_pram.", | 3312 | ugeth_err |
3261 | __FUNCTION__); | 3313 | ("%s: Can not allocate DPRAM memory for p_init_enet_pram.", |
3314 | __FUNCTION__); | ||
3262 | ucc_geth_memclean(ugeth); | 3315 | ucc_geth_memclean(ugeth); |
3263 | return -ENOMEM; | 3316 | return -ENOMEM; |
3264 | } | 3317 | } |
@@ -3428,8 +3481,9 @@ static int ucc_geth_rx(struct ucc_geth_private *ugeth, u8 rxQ, int rx_work_limit | |||
3428 | if (!skb || | 3481 | if (!skb || |
3429 | (!(bd_status & (R_F | R_L))) || | 3482 | (!(bd_status & (R_F | R_L))) || |
3430 | (bd_status & R_ERRORS_FATAL)) { | 3483 | (bd_status & R_ERRORS_FATAL)) { |
3431 | ugeth_vdbg("%s, %d: ERROR!!! skb - 0x%08x", | 3484 | if (netif_msg_rx_err(ugeth)) |
3432 | __FUNCTION__, __LINE__, (u32) skb); | 3485 | ugeth_err("%s, %d: ERROR!!! skb - 0x%08x", |
3486 | __FUNCTION__, __LINE__, (u32) skb); | ||
3433 | if (skb) | 3487 | if (skb) |
3434 | dev_kfree_skb_any(skb); | 3488 | dev_kfree_skb_any(skb); |
3435 | 3489 | ||
@@ -3458,7 +3512,8 @@ static int ucc_geth_rx(struct ucc_geth_private *ugeth, u8 rxQ, int rx_work_limit | |||
3458 | 3512 | ||
3459 | skb = get_new_skb(ugeth, bd); | 3513 | skb = get_new_skb(ugeth, bd); |
3460 | if (!skb) { | 3514 | if (!skb) { |
3461 | ugeth_warn("%s: No Rx Data Buffer", __FUNCTION__); | 3515 | if (netif_msg_rx_err(ugeth)) |
3516 | ugeth_warn("%s: No Rx Data Buffer", __FUNCTION__); | ||
3462 | ugeth->stats.rx_dropped++; | 3517 | ugeth->stats.rx_dropped++; |
3463 | break; | 3518 | break; |
3464 | } | 3519 | } |
@@ -3649,28 +3704,32 @@ static int ucc_geth_open(struct net_device *dev) | |||
3649 | 3704 | ||
3650 | /* Test station address */ | 3705 | /* Test station address */ |
3651 | if (dev->dev_addr[0] & ENET_GROUP_ADDR) { | 3706 | if (dev->dev_addr[0] & ENET_GROUP_ADDR) { |
3652 | ugeth_err("%s: Multicast address used for station address" | 3707 | if (netif_msg_ifup(ugeth)) |
3653 | " - is this what you wanted?", __FUNCTION__); | 3708 | ugeth_err("%s: Multicast address used for station address" |
3709 | " - is this what you wanted?", __FUNCTION__); | ||
3654 | return -EINVAL; | 3710 | return -EINVAL; |
3655 | } | 3711 | } |
3656 | 3712 | ||
3657 | err = ucc_struct_init(ugeth); | 3713 | err = ucc_struct_init(ugeth); |
3658 | if (err) { | 3714 | if (err) { |
3659 | ugeth_err("%s: Cannot configure internal struct, aborting.", dev->name); | 3715 | if (netif_msg_ifup(ugeth)) |
3716 | ugeth_err("%s: Cannot configure internal struct, aborting.", dev->name); | ||
3660 | return err; | 3717 | return err; |
3661 | } | 3718 | } |
3662 | 3719 | ||
3663 | err = ucc_geth_startup(ugeth); | 3720 | err = ucc_geth_startup(ugeth); |
3664 | if (err) { | 3721 | if (err) { |
3665 | ugeth_err("%s: Cannot configure net device, aborting.", | 3722 | if (netif_msg_ifup(ugeth)) |
3666 | dev->name); | 3723 | ugeth_err("%s: Cannot configure net device, aborting.", |
3724 | dev->name); | ||
3667 | return err; | 3725 | return err; |
3668 | } | 3726 | } |
3669 | 3727 | ||
3670 | err = adjust_enet_interface(ugeth); | 3728 | err = adjust_enet_interface(ugeth); |
3671 | if (err) { | 3729 | if (err) { |
3672 | ugeth_err("%s: Cannot configure net device, aborting.", | 3730 | if (netif_msg_ifup(ugeth)) |
3673 | dev->name); | 3731 | ugeth_err("%s: Cannot configure net device, aborting.", |
3732 | dev->name); | ||
3674 | return err; | 3733 | return err; |
3675 | } | 3734 | } |
3676 | 3735 | ||
@@ -3687,7 +3746,8 @@ static int ucc_geth_open(struct net_device *dev) | |||
3687 | 3746 | ||
3688 | err = init_phy(dev); | 3747 | err = init_phy(dev); |
3689 | if (err) { | 3748 | if (err) { |
3690 | ugeth_err("%s: Cannot initialize PHY, aborting.", dev->name); | 3749 | if (netif_msg_ifup(ugeth)) |
3750 | ugeth_err("%s: Cannot initialize PHY, aborting.", dev->name); | ||
3691 | return err; | 3751 | return err; |
3692 | } | 3752 | } |
3693 | 3753 | ||
@@ -3697,15 +3757,17 @@ static int ucc_geth_open(struct net_device *dev) | |||
3697 | request_irq(ugeth->ug_info->uf_info.irq, ucc_geth_irq_handler, 0, | 3757 | request_irq(ugeth->ug_info->uf_info.irq, ucc_geth_irq_handler, 0, |
3698 | "UCC Geth", dev); | 3758 | "UCC Geth", dev); |
3699 | if (err) { | 3759 | if (err) { |
3700 | ugeth_err("%s: Cannot get IRQ for net device, aborting.", | 3760 | if (netif_msg_ifup(ugeth)) |
3701 | dev->name); | 3761 | ugeth_err("%s: Cannot get IRQ for net device, aborting.", |
3762 | dev->name); | ||
3702 | ucc_geth_stop(ugeth); | 3763 | ucc_geth_stop(ugeth); |
3703 | return err; | 3764 | return err; |
3704 | } | 3765 | } |
3705 | 3766 | ||
3706 | err = ugeth_enable(ugeth, COMM_DIR_RX_AND_TX); | 3767 | err = ugeth_enable(ugeth, COMM_DIR_RX_AND_TX); |
3707 | if (err) { | 3768 | if (err) { |
3708 | ugeth_err("%s: Cannot enable net device, aborting.", dev->name); | 3769 | if (netif_msg_ifup(ugeth)) |
3770 | ugeth_err("%s: Cannot enable net device, aborting.", dev->name); | ||
3709 | ucc_geth_stop(ugeth); | 3771 | ucc_geth_stop(ugeth); |
3710 | return err; | 3772 | return err; |
3711 | } | 3773 | } |
@@ -3732,8 +3794,6 @@ static int ucc_geth_close(struct net_device *dev) | |||
3732 | return 0; | 3794 | return 0; |
3733 | } | 3795 | } |
3734 | 3796 | ||
3735 | const struct ethtool_ops ucc_geth_ethtool_ops = { }; | ||
3736 | |||
3737 | static phy_interface_t to_phy_interface(const char *phy_connection_type) | 3797 | static phy_interface_t to_phy_interface(const char *phy_connection_type) |
3738 | { | 3798 | { |
3739 | if (strcasecmp(phy_connection_type, "mii") == 0) | 3799 | if (strcasecmp(phy_connection_type, "mii") == 0) |
@@ -3790,6 +3850,13 @@ static int ucc_geth_probe(struct of_device* ofdev, const struct of_device_id *ma | |||
3790 | return -ENODEV; | 3850 | return -ENODEV; |
3791 | 3851 | ||
3792 | ug_info = &ugeth_info[ucc_num]; | 3852 | ug_info = &ugeth_info[ucc_num]; |
3853 | if (ug_info == NULL) { | ||
3854 | if (netif_msg_probe(&debug)) | ||
3855 | ugeth_err("%s: [%d] Missing additional data!", | ||
3856 | __FUNCTION__, ucc_num); | ||
3857 | return -ENODEV; | ||
3858 | } | ||
3859 | |||
3793 | ug_info->uf_info.ucc_num = ucc_num; | 3860 | ug_info->uf_info.ucc_num = ucc_num; |
3794 | 3861 | ||
3795 | prop = of_get_property(np, "rx-clock", NULL); | 3862 | prop = of_get_property(np, "rx-clock", NULL); |
@@ -3868,15 +3935,10 @@ static int ucc_geth_probe(struct of_device* ofdev, const struct of_device_id *ma | |||
3868 | 3935 | ||
3869 | ug_info->mdio_bus = res.start; | 3936 | ug_info->mdio_bus = res.start; |
3870 | 3937 | ||
3871 | printk(KERN_INFO "ucc_geth: UCC%1d at 0x%8x (irq = %d) \n", | 3938 | if (netif_msg_probe(&debug)) |
3872 | ug_info->uf_info.ucc_num + 1, ug_info->uf_info.regs, | 3939 | printk(KERN_INFO "ucc_geth: UCC%1d at 0x%8x (irq = %d) \n", |
3873 | ug_info->uf_info.irq); | 3940 | ug_info->uf_info.ucc_num + 1, ug_info->uf_info.regs, |
3874 | 3941 | ug_info->uf_info.irq); | |
3875 | if (ug_info == NULL) { | ||
3876 | ugeth_err("%s: [%d] Missing additional data!", __FUNCTION__, | ||
3877 | ucc_num); | ||
3878 | return -ENODEV; | ||
3879 | } | ||
3880 | 3942 | ||
3881 | /* Create an ethernet device instance */ | 3943 | /* Create an ethernet device instance */ |
3882 | dev = alloc_etherdev(sizeof(*ugeth)); | 3944 | dev = alloc_etherdev(sizeof(*ugeth)); |
@@ -3896,6 +3958,7 @@ static int ucc_geth_probe(struct of_device* ofdev, const struct of_device_id *ma | |||
3896 | SET_NETDEV_DEV(dev, device); | 3958 | SET_NETDEV_DEV(dev, device); |
3897 | 3959 | ||
3898 | /* Fill in the dev structure */ | 3960 | /* Fill in the dev structure */ |
3961 | uec_set_ethtool_ops(dev); | ||
3899 | dev->open = ucc_geth_open; | 3962 | dev->open = ucc_geth_open; |
3900 | dev->hard_start_xmit = ucc_geth_start_xmit; | 3963 | dev->hard_start_xmit = ucc_geth_start_xmit; |
3901 | dev->tx_timeout = ucc_geth_timeout; | 3964 | dev->tx_timeout = ucc_geth_timeout; |
@@ -3909,16 +3972,16 @@ static int ucc_geth_probe(struct of_device* ofdev, const struct of_device_id *ma | |||
3909 | // dev->change_mtu = ucc_geth_change_mtu; | 3972 | // dev->change_mtu = ucc_geth_change_mtu; |
3910 | dev->mtu = 1500; | 3973 | dev->mtu = 1500; |
3911 | dev->set_multicast_list = ucc_geth_set_multi; | 3974 | dev->set_multicast_list = ucc_geth_set_multi; |
3912 | dev->ethtool_ops = &ucc_geth_ethtool_ops; | ||
3913 | 3975 | ||
3914 | ugeth->msg_enable = (NETIF_MSG_IFUP << 1 ) - 1; | 3976 | ugeth->msg_enable = netif_msg_init(debug.msg_enable, UGETH_MSG_DEFAULT); |
3915 | ugeth->phy_interface = phy_interface; | 3977 | ugeth->phy_interface = phy_interface; |
3916 | ugeth->max_speed = max_speed; | 3978 | ugeth->max_speed = max_speed; |
3917 | 3979 | ||
3918 | err = register_netdev(dev); | 3980 | err = register_netdev(dev); |
3919 | if (err) { | 3981 | if (err) { |
3920 | ugeth_err("%s: Cannot register net device, aborting.", | 3982 | if (netif_msg_probe(ugeth)) |
3921 | dev->name); | 3983 | ugeth_err("%s: Cannot register net device, aborting.", |
3984 | dev->name); | ||
3922 | free_netdev(dev); | 3985 | free_netdev(dev); |
3923 | return err; | 3986 | return err; |
3924 | } | 3987 | } |
@@ -3972,7 +4035,8 @@ static int __init ucc_geth_init(void) | |||
3972 | if (ret) | 4035 | if (ret) |
3973 | return ret; | 4036 | return ret; |
3974 | 4037 | ||
3975 | printk(KERN_INFO "ucc_geth: " DRV_DESC "\n"); | 4038 | if (netif_msg_drv(&debug)) |
4039 | printk(KERN_INFO "ucc_geth: " DRV_DESC "\n"); | ||
3976 | for (i = 0; i < 8; i++) | 4040 | for (i = 0; i < 8; i++) |
3977 | memcpy(&(ugeth_info[i]), &ugeth_primary_info, | 4041 | memcpy(&(ugeth_info[i]), &ugeth_primary_info, |
3978 | sizeof(ugeth_primary_info)); | 4042 | sizeof(ugeth_primary_info)); |
diff --git a/drivers/net/ucc_geth.h b/drivers/net/ucc_geth.h index a29e1c3ca4b7..bb4dac8c0c65 100644 --- a/drivers/net/ucc_geth.h +++ b/drivers/net/ucc_geth.h | |||
@@ -30,6 +30,10 @@ | |||
30 | 30 | ||
31 | #include "ucc_geth_mii.h" | 31 | #include "ucc_geth_mii.h" |
32 | 32 | ||
33 | #define DRV_DESC "QE UCC Gigabit Ethernet Controller" | ||
34 | #define DRV_NAME "ucc_geth" | ||
35 | #define DRV_VERSION "1.1" | ||
36 | |||
33 | #define NUM_TX_QUEUES 8 | 37 | #define NUM_TX_QUEUES 8 |
34 | #define NUM_RX_QUEUES 8 | 38 | #define NUM_RX_QUEUES 8 |
35 | #define NUM_BDS_IN_PREFETCHED_BDS 4 | 39 | #define NUM_BDS_IN_PREFETCHED_BDS 4 |
@@ -896,6 +900,7 @@ struct ucc_geth_hardware_statistics { | |||
896 | #define UCC_GETH_TX_VTAG_TABLE_ENTRY_MAX 8 | 900 | #define UCC_GETH_TX_VTAG_TABLE_ENTRY_MAX 8 |
897 | #define UCC_GETH_RX_BD_RING_SIZE_MIN 8 | 901 | #define UCC_GETH_RX_BD_RING_SIZE_MIN 8 |
898 | #define UCC_GETH_TX_BD_RING_SIZE_MIN 2 | 902 | #define UCC_GETH_TX_BD_RING_SIZE_MIN 2 |
903 | #define UCC_GETH_BD_RING_SIZE_MAX 0xffff | ||
899 | 904 | ||
900 | #define UCC_GETH_SIZE_OF_BD QE_SIZEOF_BD | 905 | #define UCC_GETH_SIZE_OF_BD QE_SIZEOF_BD |
901 | 906 | ||
@@ -1135,6 +1140,7 @@ struct ucc_geth_info { | |||
1135 | int bro; | 1140 | int bro; |
1136 | int ecm; | 1141 | int ecm; |
1137 | int receiveFlowControl; | 1142 | int receiveFlowControl; |
1143 | int transmitFlowControl; | ||
1138 | u8 maxGroupAddrInHash; | 1144 | u8 maxGroupAddrInHash; |
1139 | u8 maxIndAddrInHash; | 1145 | u8 maxIndAddrInHash; |
1140 | u8 prel; | 1146 | u8 prel; |
diff --git a/drivers/net/ucc_geth_ethtool.c b/drivers/net/ucc_geth_ethtool.c new file mode 100644 index 000000000000..a8994c7b8583 --- /dev/null +++ b/drivers/net/ucc_geth_ethtool.c | |||
@@ -0,0 +1,388 @@ | |||
1 | /* | ||
2 | * Copyright (c) 2007 Freescale Semiconductor, Inc. All rights reserved. | ||
3 | * | ||
4 | * Description: QE UCC Gigabit Ethernet Ethtool API Set | ||
5 | * | ||
6 | * Author: Li Yang <leoli@freescale.com> | ||
7 | * | ||
8 | * Limitation: | ||
9 | * Can only get/set setttings of the first queue. | ||
10 | * Need to re-open the interface manually after changing some paramters. | ||
11 | * | ||
12 | * This program is free software; you can redistribute it and/or modify it | ||
13 | * under the terms of the GNU General Public License as published by the | ||
14 | * Free Software Foundation; either version 2 of the License, or (at your | ||
15 | * option) any later version. | ||
16 | */ | ||
17 | |||
18 | #include <linux/kernel.h> | ||
19 | #include <linux/init.h> | ||
20 | #include <linux/errno.h> | ||
21 | #include <linux/slab.h> | ||
22 | #include <linux/stddef.h> | ||
23 | #include <linux/interrupt.h> | ||
24 | #include <linux/netdevice.h> | ||
25 | #include <linux/etherdevice.h> | ||
26 | #include <linux/skbuff.h> | ||
27 | #include <linux/spinlock.h> | ||
28 | #include <linux/mm.h> | ||
29 | #include <linux/delay.h> | ||
30 | #include <linux/dma-mapping.h> | ||
31 | #include <linux/fsl_devices.h> | ||
32 | #include <linux/ethtool.h> | ||
33 | #include <linux/mii.h> | ||
34 | #include <linux/phy.h> | ||
35 | |||
36 | #include <asm/io.h> | ||
37 | #include <asm/irq.h> | ||
38 | #include <asm/uaccess.h> | ||
39 | #include <asm/types.h> | ||
40 | #include <asm/uaccess.h> | ||
41 | |||
42 | #include "ucc_geth.h" | ||
43 | #include "ucc_geth_mii.h" | ||
44 | |||
45 | static char hw_stat_gstrings[][ETH_GSTRING_LEN] = { | ||
46 | "tx-64-frames", | ||
47 | "tx-65-127-frames", | ||
48 | "tx-128-255-frames", | ||
49 | "rx-64-frames", | ||
50 | "rx-65-127-frames", | ||
51 | "rx-128-255-frames", | ||
52 | "tx-bytes-ok", | ||
53 | "tx-pause-frames", | ||
54 | "tx-multicast-frames", | ||
55 | "tx-broadcast-frames", | ||
56 | "rx-frames", | ||
57 | "rx-bytes-ok", | ||
58 | "rx-bytes-all", | ||
59 | "rx-multicast-frames", | ||
60 | "rx-broadcast-frames", | ||
61 | "stats-counter-carry", | ||
62 | "stats-counter-mask", | ||
63 | "rx-dropped-frames", | ||
64 | }; | ||
65 | |||
66 | static char tx_fw_stat_gstrings[][ETH_GSTRING_LEN] = { | ||
67 | "tx-single-collision", | ||
68 | "tx-multiple-collision", | ||
69 | "tx-late-collsion", | ||
70 | "tx-aborted-frames", | ||
71 | "tx-lost-frames", | ||
72 | "tx-carrier-sense-errors", | ||
73 | "tx-frames-ok", | ||
74 | "tx-excessive-differ-frames", | ||
75 | "tx-256-511-frames", | ||
76 | "tx-1024-1518-frames", | ||
77 | "tx-jumbo-frames", | ||
78 | }; | ||
79 | |||
80 | static char rx_fw_stat_gstrings[][ETH_GSTRING_LEN] = { | ||
81 | "rx-crc-errors", | ||
82 | "rx-alignment-errors", | ||
83 | "rx-in-range-length-errors", | ||
84 | "rx-out-of-range-length-errors", | ||
85 | "rx-too-long-frames", | ||
86 | "rx-runt", | ||
87 | "rx-very-long-event", | ||
88 | "rx-symbol-errors", | ||
89 | "rx-busy-drop-frames", | ||
90 | "reserved", | ||
91 | "reserved", | ||
92 | "rx-mismatch-drop-frames", | ||
93 | "rx-small-than-64", | ||
94 | "rx-256-511-frames", | ||
95 | "rx-512-1023-frames", | ||
96 | "rx-1024-1518-frames", | ||
97 | "rx-jumbo-frames", | ||
98 | "rx-mac-error-loss", | ||
99 | "rx-pause-frames", | ||
100 | "reserved", | ||
101 | "rx-vlan-removed", | ||
102 | "rx-vlan-replaced", | ||
103 | "rx-vlan-inserted", | ||
104 | "rx-ip-checksum-errors", | ||
105 | }; | ||
106 | |||
107 | #define UEC_HW_STATS_LEN ARRAY_SIZE(hw_stat_gstrings) | ||
108 | #define UEC_TX_FW_STATS_LEN ARRAY_SIZE(tx_fw_stat_gstrings) | ||
109 | #define UEC_RX_FW_STATS_LEN ARRAY_SIZE(rx_fw_stat_gstrings) | ||
110 | |||
111 | extern int init_flow_control_params(u32 automatic_flow_control_mode, | ||
112 | int rx_flow_control_enable, | ||
113 | int tx_flow_control_enable, u16 pause_period, | ||
114 | u16 extension_field, volatile u32 *upsmr_register, | ||
115 | volatile u32 *uempr_register, volatile u32 *maccfg1_register); | ||
116 | |||
117 | static int | ||
118 | uec_get_settings(struct net_device *netdev, struct ethtool_cmd *ecmd) | ||
119 | { | ||
120 | struct ucc_geth_private *ugeth = netdev_priv(netdev); | ||
121 | struct phy_device *phydev = ugeth->phydev; | ||
122 | struct ucc_geth_info *ug_info = ugeth->ug_info; | ||
123 | |||
124 | if (!phydev) | ||
125 | return -ENODEV; | ||
126 | |||
127 | ecmd->maxtxpkt = 1; | ||
128 | ecmd->maxrxpkt = ug_info->interruptcoalescingmaxvalue[0]; | ||
129 | |||
130 | return phy_ethtool_gset(phydev, ecmd); | ||
131 | } | ||
132 | |||
133 | static int | ||
134 | uec_set_settings(struct net_device *netdev, struct ethtool_cmd *ecmd) | ||
135 | { | ||
136 | struct ucc_geth_private *ugeth = netdev_priv(netdev); | ||
137 | struct phy_device *phydev = ugeth->phydev; | ||
138 | |||
139 | if (!phydev) | ||
140 | return -ENODEV; | ||
141 | |||
142 | return phy_ethtool_sset(phydev, ecmd); | ||
143 | } | ||
144 | |||
145 | static void | ||
146 | uec_get_pauseparam(struct net_device *netdev, | ||
147 | struct ethtool_pauseparam *pause) | ||
148 | { | ||
149 | struct ucc_geth_private *ugeth = netdev_priv(netdev); | ||
150 | |||
151 | pause->autoneg = ugeth->phydev->autoneg; | ||
152 | |||
153 | if (ugeth->ug_info->receiveFlowControl) | ||
154 | pause->rx_pause = 1; | ||
155 | if (ugeth->ug_info->transmitFlowControl) | ||
156 | pause->tx_pause = 1; | ||
157 | } | ||
158 | |||
159 | static int | ||
160 | uec_set_pauseparam(struct net_device *netdev, | ||
161 | struct ethtool_pauseparam *pause) | ||
162 | { | ||
163 | struct ucc_geth_private *ugeth = netdev_priv(netdev); | ||
164 | int ret = 0; | ||
165 | |||
166 | ugeth->ug_info->receiveFlowControl = pause->rx_pause; | ||
167 | ugeth->ug_info->transmitFlowControl = pause->tx_pause; | ||
168 | |||
169 | if (ugeth->phydev->autoneg) { | ||
170 | if (netif_running(netdev)) { | ||
171 | /* FIXME: automatically restart */ | ||
172 | printk(KERN_INFO | ||
173 | "Please re-open the interface.\n"); | ||
174 | } | ||
175 | } else { | ||
176 | struct ucc_geth_info *ug_info = ugeth->ug_info; | ||
177 | |||
178 | ret = init_flow_control_params(ug_info->aufc, | ||
179 | ug_info->receiveFlowControl, | ||
180 | ug_info->transmitFlowControl, | ||
181 | ug_info->pausePeriod, | ||
182 | ug_info->extensionField, | ||
183 | &ugeth->uccf->uf_regs->upsmr, | ||
184 | &ugeth->ug_regs->uempr, | ||
185 | &ugeth->ug_regs->maccfg1); | ||
186 | } | ||
187 | |||
188 | return ret; | ||
189 | } | ||
190 | |||
191 | static uint32_t | ||
192 | uec_get_msglevel(struct net_device *netdev) | ||
193 | { | ||
194 | struct ucc_geth_private *ugeth = netdev_priv(netdev); | ||
195 | return ugeth->msg_enable; | ||
196 | } | ||
197 | |||
198 | static void | ||
199 | uec_set_msglevel(struct net_device *netdev, uint32_t data) | ||
200 | { | ||
201 | struct ucc_geth_private *ugeth = netdev_priv(netdev); | ||
202 | ugeth->msg_enable = data; | ||
203 | } | ||
204 | |||
205 | static int | ||
206 | uec_get_regs_len(struct net_device *netdev) | ||
207 | { | ||
208 | return sizeof(struct ucc_geth); | ||
209 | } | ||
210 | |||
211 | static void | ||
212 | uec_get_regs(struct net_device *netdev, | ||
213 | struct ethtool_regs *regs, void *p) | ||
214 | { | ||
215 | int i; | ||
216 | struct ucc_geth_private *ugeth = netdev_priv(netdev); | ||
217 | u32 __iomem *ug_regs = (u32 __iomem *)ugeth->ug_regs; | ||
218 | u32 *buff = p; | ||
219 | |||
220 | for (i = 0; i < sizeof(struct ucc_geth) / sizeof(u32); i++) | ||
221 | buff[i] = in_be32(&ug_regs[i]); | ||
222 | } | ||
223 | |||
224 | static void | ||
225 | uec_get_ringparam(struct net_device *netdev, | ||
226 | struct ethtool_ringparam *ring) | ||
227 | { | ||
228 | struct ucc_geth_private *ugeth = netdev_priv(netdev); | ||
229 | struct ucc_geth_info *ug_info = ugeth->ug_info; | ||
230 | int queue = 0; | ||
231 | |||
232 | ring->rx_max_pending = UCC_GETH_BD_RING_SIZE_MAX; | ||
233 | ring->rx_mini_max_pending = UCC_GETH_BD_RING_SIZE_MAX; | ||
234 | ring->rx_jumbo_max_pending = UCC_GETH_BD_RING_SIZE_MAX; | ||
235 | ring->tx_max_pending = UCC_GETH_BD_RING_SIZE_MAX; | ||
236 | |||
237 | ring->rx_pending = ug_info->bdRingLenRx[queue]; | ||
238 | ring->rx_mini_pending = ug_info->bdRingLenRx[queue]; | ||
239 | ring->rx_jumbo_pending = ug_info->bdRingLenRx[queue]; | ||
240 | ring->tx_pending = ug_info->bdRingLenTx[queue]; | ||
241 | } | ||
242 | |||
243 | static int | ||
244 | uec_set_ringparam(struct net_device *netdev, | ||
245 | struct ethtool_ringparam *ring) | ||
246 | { | ||
247 | struct ucc_geth_private *ugeth = netdev_priv(netdev); | ||
248 | struct ucc_geth_info *ug_info = ugeth->ug_info; | ||
249 | int queue = 0, ret = 0; | ||
250 | |||
251 | if (ring->rx_pending < UCC_GETH_RX_BD_RING_SIZE_MIN) { | ||
252 | printk("%s: RxBD ring size must be no smaller than %d.\n", | ||
253 | netdev->name, UCC_GETH_RX_BD_RING_SIZE_MIN); | ||
254 | return -EINVAL; | ||
255 | } | ||
256 | if (ring->rx_pending % UCC_GETH_RX_BD_RING_SIZE_ALIGNMENT) { | ||
257 | printk("%s: RxBD ring size must be multiple of %d.\n", | ||
258 | netdev->name, UCC_GETH_RX_BD_RING_SIZE_ALIGNMENT); | ||
259 | return -EINVAL; | ||
260 | } | ||
261 | if (ring->tx_pending < UCC_GETH_TX_BD_RING_SIZE_MIN) { | ||
262 | printk("%s: TxBD ring size must be no smaller than %d.\n", | ||
263 | netdev->name, UCC_GETH_TX_BD_RING_SIZE_MIN); | ||
264 | return -EINVAL; | ||
265 | } | ||
266 | |||
267 | ug_info->bdRingLenRx[queue] = ring->rx_pending; | ||
268 | ug_info->bdRingLenTx[queue] = ring->tx_pending; | ||
269 | |||
270 | if (netif_running(netdev)) { | ||
271 | /* FIXME: restart automatically */ | ||
272 | printk(KERN_INFO | ||
273 | "Please re-open the interface.\n"); | ||
274 | } | ||
275 | |||
276 | return ret; | ||
277 | } | ||
278 | |||
279 | static int uec_get_stats_count(struct net_device *netdev) | ||
280 | { | ||
281 | struct ucc_geth_private *ugeth = netdev_priv(netdev); | ||
282 | u32 stats_mode = ugeth->ug_info->statisticsMode; | ||
283 | int len = 0; | ||
284 | |||
285 | if (stats_mode & UCC_GETH_STATISTICS_GATHERING_MODE_HARDWARE) | ||
286 | len += UEC_HW_STATS_LEN; | ||
287 | if (stats_mode & UCC_GETH_STATISTICS_GATHERING_MODE_FIRMWARE_TX) | ||
288 | len += UEC_TX_FW_STATS_LEN; | ||
289 | if (stats_mode & UCC_GETH_STATISTICS_GATHERING_MODE_FIRMWARE_RX) | ||
290 | len += UEC_RX_FW_STATS_LEN; | ||
291 | |||
292 | return len; | ||
293 | } | ||
294 | |||
295 | static void uec_get_strings(struct net_device *netdev, u32 stringset, u8 *buf) | ||
296 | { | ||
297 | struct ucc_geth_private *ugeth = netdev_priv(netdev); | ||
298 | u32 stats_mode = ugeth->ug_info->statisticsMode; | ||
299 | |||
300 | if (stats_mode & UCC_GETH_STATISTICS_GATHERING_MODE_HARDWARE) { | ||
301 | memcpy(buf, hw_stat_gstrings, UEC_HW_STATS_LEN * | ||
302 | ETH_GSTRING_LEN); | ||
303 | buf += UEC_HW_STATS_LEN * ETH_GSTRING_LEN; | ||
304 | } | ||
305 | if (stats_mode & UCC_GETH_STATISTICS_GATHERING_MODE_FIRMWARE_TX) { | ||
306 | memcpy(buf, tx_fw_stat_gstrings, UEC_TX_FW_STATS_LEN * | ||
307 | ETH_GSTRING_LEN); | ||
308 | buf += UEC_TX_FW_STATS_LEN * ETH_GSTRING_LEN; | ||
309 | } | ||
310 | if (stats_mode & UCC_GETH_STATISTICS_GATHERING_MODE_FIRMWARE_RX) | ||
311 | memcpy(buf, tx_fw_stat_gstrings, UEC_RX_FW_STATS_LEN * | ||
312 | ETH_GSTRING_LEN); | ||
313 | } | ||
314 | |||
315 | static void uec_get_ethtool_stats(struct net_device *netdev, | ||
316 | struct ethtool_stats *stats, uint64_t *data) | ||
317 | { | ||
318 | struct ucc_geth_private *ugeth = netdev_priv(netdev); | ||
319 | u32 stats_mode = ugeth->ug_info->statisticsMode; | ||
320 | u32 __iomem *base; | ||
321 | int i, j = 0; | ||
322 | |||
323 | if (stats_mode & UCC_GETH_STATISTICS_GATHERING_MODE_HARDWARE) { | ||
324 | base = (u32 __iomem *)&ugeth->ug_regs->tx64; | ||
325 | for (i = 0; i < UEC_HW_STATS_LEN; i++) | ||
326 | data[j++] = (u64)in_be32(&base[i]); | ||
327 | } | ||
328 | if (stats_mode & UCC_GETH_STATISTICS_GATHERING_MODE_FIRMWARE_TX) { | ||
329 | base = (u32 __iomem *)ugeth->p_tx_fw_statistics_pram; | ||
330 | for (i = 0; i < UEC_TX_FW_STATS_LEN; i++) | ||
331 | data[j++] = (u64)in_be32(&base[i]); | ||
332 | } | ||
333 | if (stats_mode & UCC_GETH_STATISTICS_GATHERING_MODE_FIRMWARE_RX) { | ||
334 | base = (u32 __iomem *)ugeth->p_rx_fw_statistics_pram; | ||
335 | for (i = 0; i < UEC_RX_FW_STATS_LEN; i++) | ||
336 | data[j++] = (u64)in_be32(&base[i]); | ||
337 | } | ||
338 | } | ||
339 | |||
340 | static int uec_nway_reset(struct net_device *netdev) | ||
341 | { | ||
342 | struct ucc_geth_private *ugeth = netdev_priv(netdev); | ||
343 | |||
344 | return phy_start_aneg(ugeth->phydev); | ||
345 | } | ||
346 | |||
347 | /* Report driver information */ | ||
348 | static void | ||
349 | uec_get_drvinfo(struct net_device *netdev, | ||
350 | struct ethtool_drvinfo *drvinfo) | ||
351 | { | ||
352 | strncpy(drvinfo->driver, DRV_NAME, 32); | ||
353 | strncpy(drvinfo->version, DRV_VERSION, 32); | ||
354 | strncpy(drvinfo->fw_version, "N/A", 32); | ||
355 | strncpy(drvinfo->bus_info, "QUICC ENGINE", 32); | ||
356 | drvinfo->n_stats = uec_get_stats_count(netdev); | ||
357 | drvinfo->testinfo_len = 0; | ||
358 | drvinfo->eedump_len = 0; | ||
359 | drvinfo->regdump_len = uec_get_regs_len(netdev); | ||
360 | } | ||
361 | |||
362 | static const struct ethtool_ops uec_ethtool_ops = { | ||
363 | .get_settings = uec_get_settings, | ||
364 | .set_settings = uec_set_settings, | ||
365 | .get_drvinfo = uec_get_drvinfo, | ||
366 | .get_regs_len = uec_get_regs_len, | ||
367 | .get_regs = uec_get_regs, | ||
368 | .get_msglevel = uec_get_msglevel, | ||
369 | .set_msglevel = uec_set_msglevel, | ||
370 | .nway_reset = uec_nway_reset, | ||
371 | .get_link = ethtool_op_get_link, | ||
372 | .get_ringparam = uec_get_ringparam, | ||
373 | .set_ringparam = uec_set_ringparam, | ||
374 | .get_pauseparam = uec_get_pauseparam, | ||
375 | .set_pauseparam = uec_set_pauseparam, | ||
376 | .get_sg = ethtool_op_get_sg, | ||
377 | .set_sg = ethtool_op_set_sg, | ||
378 | .get_tso = ethtool_op_get_tso, | ||
379 | .get_stats_count = uec_get_stats_count, | ||
380 | .get_strings = uec_get_strings, | ||
381 | .get_ethtool_stats = uec_get_ethtool_stats, | ||
382 | .get_perm_addr = ethtool_op_get_perm_addr, | ||
383 | }; | ||
384 | |||
385 | void uec_set_ethtool_ops(struct net_device *netdev) | ||
386 | { | ||
387 | SET_ETHTOOL_OPS(netdev, &uec_ethtool_ops); | ||
388 | } | ||
diff --git a/drivers/net/ucc_geth_mii.c b/drivers/net/ucc_geth_mii.c index 7bcb82f50cf7..5f8c2d30a328 100644 --- a/drivers/net/ucc_geth_mii.c +++ b/drivers/net/ucc_geth_mii.c | |||
@@ -54,8 +54,8 @@ | |||
54 | #define vdbg(format, arg...) do {} while(0) | 54 | #define vdbg(format, arg...) do {} while(0) |
55 | #endif | 55 | #endif |
56 | 56 | ||
57 | #define DRV_DESC "QE UCC Ethernet Controller MII Bus" | 57 | #define MII_DRV_DESC "QE UCC Ethernet Controller MII Bus" |
58 | #define DRV_NAME "fsl-uec_mdio" | 58 | #define MII_DRV_NAME "fsl-uec_mdio" |
59 | 59 | ||
60 | /* Write value to the PHY for this device to the register at regnum, */ | 60 | /* Write value to the PHY for this device to the register at regnum, */ |
61 | /* waiting until the write is done before it returns. All PHY */ | 61 | /* waiting until the write is done before it returns. All PHY */ |
@@ -261,7 +261,7 @@ static struct of_device_id uec_mdio_match[] = { | |||
261 | }; | 261 | }; |
262 | 262 | ||
263 | static struct of_platform_driver uec_mdio_driver = { | 263 | static struct of_platform_driver uec_mdio_driver = { |
264 | .name = DRV_NAME, | 264 | .name = MII_DRV_NAME, |
265 | .probe = uec_mdio_probe, | 265 | .probe = uec_mdio_probe, |
266 | .remove = uec_mdio_remove, | 266 | .remove = uec_mdio_remove, |
267 | .match_table = uec_mdio_match, | 267 | .match_table = uec_mdio_match, |
diff --git a/drivers/pci/pci-acpi.c b/drivers/pci/pci-acpi.c index c8062494009f..67c63d1f1582 100644 --- a/drivers/pci/pci-acpi.c +++ b/drivers/pci/pci-acpi.c | |||
@@ -245,16 +245,33 @@ EXPORT_SYMBOL(pci_osc_control_set); | |||
245 | * currently we simply return _SxD, if present. | 245 | * currently we simply return _SxD, if present. |
246 | */ | 246 | */ |
247 | 247 | ||
248 | static int acpi_pci_choose_state(struct pci_dev *pdev, pm_message_t state) | 248 | static pci_power_t acpi_pci_choose_state(struct pci_dev *pdev, |
249 | pm_message_t state) | ||
249 | { | 250 | { |
250 | /* TBD */ | 251 | int acpi_state; |
251 | 252 | ||
252 | return -ENODEV; | 253 | acpi_state = acpi_pm_device_sleep_state(&pdev->dev, |
254 | device_may_wakeup(&pdev->dev), NULL); | ||
255 | if (acpi_state < 0) | ||
256 | return PCI_POWER_ERROR; | ||
257 | |||
258 | switch (acpi_state) { | ||
259 | case ACPI_STATE_D0: | ||
260 | return PCI_D0; | ||
261 | case ACPI_STATE_D1: | ||
262 | return PCI_D1; | ||
263 | case ACPI_STATE_D2: | ||
264 | return PCI_D2; | ||
265 | case ACPI_STATE_D3: | ||
266 | return PCI_D3hot; | ||
267 | } | ||
268 | return PCI_POWER_ERROR; | ||
253 | } | 269 | } |
254 | 270 | ||
255 | static int acpi_pci_set_power_state(struct pci_dev *dev, pci_power_t state) | 271 | static int acpi_pci_set_power_state(struct pci_dev *dev, pci_power_t state) |
256 | { | 272 | { |
257 | acpi_handle handle = DEVICE_ACPI_HANDLE(&dev->dev); | 273 | acpi_handle handle = DEVICE_ACPI_HANDLE(&dev->dev); |
274 | acpi_handle tmp; | ||
258 | static int state_conv[] = { | 275 | static int state_conv[] = { |
259 | [0] = 0, | 276 | [0] = 0, |
260 | [1] = 1, | 277 | [1] = 1, |
@@ -266,6 +283,9 @@ static int acpi_pci_set_power_state(struct pci_dev *dev, pci_power_t state) | |||
266 | 283 | ||
267 | if (!handle) | 284 | if (!handle) |
268 | return -ENODEV; | 285 | return -ENODEV; |
286 | /* If the ACPI device has _EJ0, ignore the device */ | ||
287 | if (ACPI_SUCCESS(acpi_get_handle(handle, "_EJ0", &tmp))) | ||
288 | return 0; | ||
269 | return acpi_bus_set_power(handle, acpi_state); | 289 | return acpi_bus_set_power(handle, acpi_state); |
270 | } | 290 | } |
271 | 291 | ||
diff --git a/drivers/pci/pci.c b/drivers/pci/pci.c index 03fd59e80fef..fba319d6fcc8 100644 --- a/drivers/pci/pci.c +++ b/drivers/pci/pci.c | |||
@@ -499,7 +499,7 @@ pci_set_power_state(struct pci_dev *dev, pci_power_t state) | |||
499 | return 0; | 499 | return 0; |
500 | } | 500 | } |
501 | 501 | ||
502 | int (*platform_pci_choose_state)(struct pci_dev *dev, pm_message_t state); | 502 | pci_power_t (*platform_pci_choose_state)(struct pci_dev *dev, pm_message_t state); |
503 | 503 | ||
504 | /** | 504 | /** |
505 | * pci_choose_state - Choose the power state of a PCI device | 505 | * pci_choose_state - Choose the power state of a PCI device |
@@ -513,15 +513,15 @@ int (*platform_pci_choose_state)(struct pci_dev *dev, pm_message_t state); | |||
513 | 513 | ||
514 | pci_power_t pci_choose_state(struct pci_dev *dev, pm_message_t state) | 514 | pci_power_t pci_choose_state(struct pci_dev *dev, pm_message_t state) |
515 | { | 515 | { |
516 | int ret; | 516 | pci_power_t ret; |
517 | 517 | ||
518 | if (!pci_find_capability(dev, PCI_CAP_ID_PM)) | 518 | if (!pci_find_capability(dev, PCI_CAP_ID_PM)) |
519 | return PCI_D0; | 519 | return PCI_D0; |
520 | 520 | ||
521 | if (platform_pci_choose_state) { | 521 | if (platform_pci_choose_state) { |
522 | ret = platform_pci_choose_state(dev, state); | 522 | ret = platform_pci_choose_state(dev, state); |
523 | if (ret >= 0) | 523 | if (ret != PCI_POWER_ERROR) |
524 | state.event = ret; | 524 | return ret; |
525 | } | 525 | } |
526 | 526 | ||
527 | switch (state.event) { | 527 | switch (state.event) { |
@@ -1604,6 +1604,7 @@ early_param("pci", pci_setup); | |||
1604 | device_initcall(pci_init); | 1604 | device_initcall(pci_init); |
1605 | 1605 | ||
1606 | EXPORT_SYMBOL_GPL(pci_restore_bars); | 1606 | EXPORT_SYMBOL_GPL(pci_restore_bars); |
1607 | EXPORT_SYMBOL(__pci_reenable_device); | ||
1607 | EXPORT_SYMBOL(pci_enable_device_bars); | 1608 | EXPORT_SYMBOL(pci_enable_device_bars); |
1608 | EXPORT_SYMBOL(pci_enable_device); | 1609 | EXPORT_SYMBOL(pci_enable_device); |
1609 | EXPORT_SYMBOL(pcim_enable_device); | 1610 | EXPORT_SYMBOL(pcim_enable_device); |
diff --git a/drivers/pci/pci.h b/drivers/pci/pci.h index 3fec13d3add7..c6e132d7c0f7 100644 --- a/drivers/pci/pci.h +++ b/drivers/pci/pci.h | |||
@@ -1,6 +1,5 @@ | |||
1 | /* Functions internal to the PCI core code */ | 1 | /* Functions internal to the PCI core code */ |
2 | 2 | ||
3 | extern int __must_check __pci_reenable_device(struct pci_dev *); | ||
4 | extern int pci_uevent(struct device *dev, char **envp, int num_envp, | 3 | extern int pci_uevent(struct device *dev, char **envp, int num_envp, |
5 | char *buffer, int buffer_size); | 4 | char *buffer, int buffer_size); |
6 | extern int pci_create_sysfs_dev_files(struct pci_dev *pdev); | 5 | extern int pci_create_sysfs_dev_files(struct pci_dev *pdev); |
@@ -13,7 +12,7 @@ extern int pci_bus_alloc_resource(struct pci_bus *bus, struct resource *res, | |||
13 | resource_size_t, resource_size_t), | 12 | resource_size_t, resource_size_t), |
14 | void *alignf_data); | 13 | void *alignf_data); |
15 | /* Firmware callbacks */ | 14 | /* Firmware callbacks */ |
16 | extern int (*platform_pci_choose_state)(struct pci_dev *dev, pm_message_t state); | 15 | extern pci_power_t (*platform_pci_choose_state)(struct pci_dev *dev, pm_message_t state); |
17 | extern int (*platform_pci_set_power_state)(struct pci_dev *dev, pci_power_t state); | 16 | extern int (*platform_pci_set_power_state)(struct pci_dev *dev, pci_power_t state); |
18 | 17 | ||
19 | extern int pci_user_read_config_byte(struct pci_dev *dev, int where, u8 *val); | 18 | extern int pci_user_read_config_byte(struct pci_dev *dev, int where, u8 *val); |
diff --git a/drivers/pnp/card.c b/drivers/pnp/card.c index dd6384b1efce..b6a4f02b01d1 100644 --- a/drivers/pnp/card.c +++ b/drivers/pnp/card.c | |||
@@ -2,7 +2,6 @@ | |||
2 | * card.c - contains functions for managing groups of PnP devices | 2 | * card.c - contains functions for managing groups of PnP devices |
3 | * | 3 | * |
4 | * Copyright 2002 Adam Belay <ambx1@neo.rr.com> | 4 | * Copyright 2002 Adam Belay <ambx1@neo.rr.com> |
5 | * | ||
6 | */ | 5 | */ |
7 | 6 | ||
8 | #include <linux/module.h> | 7 | #include <linux/module.h> |
@@ -13,26 +12,31 @@ | |||
13 | LIST_HEAD(pnp_cards); | 12 | LIST_HEAD(pnp_cards); |
14 | static LIST_HEAD(pnp_card_drivers); | 13 | static LIST_HEAD(pnp_card_drivers); |
15 | 14 | ||
16 | 15 | static const struct pnp_card_device_id *match_card(struct pnp_card_driver *drv, | |
17 | static const struct pnp_card_device_id * match_card(struct pnp_card_driver * drv, struct pnp_card * card) | 16 | struct pnp_card *card) |
18 | { | 17 | { |
19 | const struct pnp_card_device_id * drv_id = drv->id_table; | 18 | const struct pnp_card_device_id *drv_id = drv->id_table; |
20 | while (*drv_id->id){ | 19 | |
21 | if (compare_pnp_id(card->id,drv_id->id)) { | 20 | while (*drv_id->id) { |
21 | if (compare_pnp_id(card->id, drv_id->id)) { | ||
22 | int i = 0; | 22 | int i = 0; |
23 | |||
23 | for (;;) { | 24 | for (;;) { |
24 | int found; | 25 | int found; |
25 | struct pnp_dev *dev; | 26 | struct pnp_dev *dev; |
26 | if (i == PNP_MAX_DEVICES || ! *drv_id->devs[i].id) | 27 | |
28 | if (i == PNP_MAX_DEVICES | ||
29 | || !*drv_id->devs[i].id) | ||
27 | return drv_id; | 30 | return drv_id; |
28 | found = 0; | 31 | found = 0; |
29 | card_for_each_dev(card, dev) { | 32 | card_for_each_dev(card, dev) { |
30 | if (compare_pnp_id(dev->id, drv_id->devs[i].id)) { | 33 | if (compare_pnp_id |
34 | (dev->id, drv_id->devs[i].id)) { | ||
31 | found = 1; | 35 | found = 1; |
32 | break; | 36 | break; |
33 | } | 37 | } |
34 | } | 38 | } |
35 | if (! found) | 39 | if (!found) |
36 | break; | 40 | break; |
37 | i++; | 41 | i++; |
38 | } | 42 | } |
@@ -42,14 +46,15 @@ static const struct pnp_card_device_id * match_card(struct pnp_card_driver * drv | |||
42 | return NULL; | 46 | return NULL; |
43 | } | 47 | } |
44 | 48 | ||
45 | static void card_remove(struct pnp_dev * dev) | 49 | static void card_remove(struct pnp_dev *dev) |
46 | { | 50 | { |
47 | dev->card_link = NULL; | 51 | dev->card_link = NULL; |
48 | } | 52 | } |
49 | 53 | ||
50 | static void card_remove_first(struct pnp_dev * dev) | 54 | static void card_remove_first(struct pnp_dev *dev) |
51 | { | 55 | { |
52 | struct pnp_card_driver * drv = to_pnp_card_driver(dev->driver); | 56 | struct pnp_card_driver *drv = to_pnp_card_driver(dev->driver); |
57 | |||
53 | if (!dev->card || !drv) | 58 | if (!dev->card || !drv) |
54 | return; | 59 | return; |
55 | if (drv->remove) | 60 | if (drv->remove) |
@@ -67,7 +72,7 @@ static int card_probe(struct pnp_card *card, struct pnp_card_driver *drv) | |||
67 | 72 | ||
68 | if (!drv->probe) | 73 | if (!drv->probe) |
69 | return 0; | 74 | return 0; |
70 | id = match_card(drv,card); | 75 | id = match_card(drv, card); |
71 | if (!id) | 76 | if (!id) |
72 | return 0; | 77 | return 0; |
73 | 78 | ||
@@ -94,12 +99,11 @@ static int card_probe(struct pnp_card *card, struct pnp_card_driver *drv) | |||
94 | * pnp_add_card_id - adds an EISA id to the specified card | 99 | * pnp_add_card_id - adds an EISA id to the specified card |
95 | * @id: pointer to a pnp_id structure | 100 | * @id: pointer to a pnp_id structure |
96 | * @card: pointer to the desired card | 101 | * @card: pointer to the desired card |
97 | * | ||
98 | */ | 102 | */ |
99 | 103 | int pnp_add_card_id(struct pnp_id *id, struct pnp_card *card) | |
100 | int pnp_add_card_id(struct pnp_id *id, struct pnp_card * card) | ||
101 | { | 104 | { |
102 | struct pnp_id * ptr; | 105 | struct pnp_id *ptr; |
106 | |||
103 | if (!id) | 107 | if (!id) |
104 | return -EINVAL; | 108 | return -EINVAL; |
105 | if (!card) | 109 | if (!card) |
@@ -115,10 +119,11 @@ int pnp_add_card_id(struct pnp_id *id, struct pnp_card * card) | |||
115 | return 0; | 119 | return 0; |
116 | } | 120 | } |
117 | 121 | ||
118 | static void pnp_free_card_ids(struct pnp_card * card) | 122 | static void pnp_free_card_ids(struct pnp_card *card) |
119 | { | 123 | { |
120 | struct pnp_id * id; | 124 | struct pnp_id *id; |
121 | struct pnp_id *next; | 125 | struct pnp_id *next; |
126 | |||
122 | if (!card) | 127 | if (!card) |
123 | return; | 128 | return; |
124 | id = card->id; | 129 | id = card->id; |
@@ -131,49 +136,55 @@ static void pnp_free_card_ids(struct pnp_card * card) | |||
131 | 136 | ||
132 | static void pnp_release_card(struct device *dmdev) | 137 | static void pnp_release_card(struct device *dmdev) |
133 | { | 138 | { |
134 | struct pnp_card * card = to_pnp_card(dmdev); | 139 | struct pnp_card *card = to_pnp_card(dmdev); |
140 | |||
135 | pnp_free_card_ids(card); | 141 | pnp_free_card_ids(card); |
136 | kfree(card); | 142 | kfree(card); |
137 | } | 143 | } |
138 | 144 | ||
139 | 145 | static ssize_t pnp_show_card_name(struct device *dmdev, | |
140 | static ssize_t pnp_show_card_name(struct device *dmdev, struct device_attribute *attr, char *buf) | 146 | struct device_attribute *attr, char *buf) |
141 | { | 147 | { |
142 | char *str = buf; | 148 | char *str = buf; |
143 | struct pnp_card *card = to_pnp_card(dmdev); | 149 | struct pnp_card *card = to_pnp_card(dmdev); |
144 | str += sprintf(str,"%s\n", card->name); | 150 | |
151 | str += sprintf(str, "%s\n", card->name); | ||
145 | return (str - buf); | 152 | return (str - buf); |
146 | } | 153 | } |
147 | 154 | ||
148 | static DEVICE_ATTR(name,S_IRUGO,pnp_show_card_name,NULL); | 155 | static DEVICE_ATTR(name, S_IRUGO, pnp_show_card_name, NULL); |
149 | 156 | ||
150 | static ssize_t pnp_show_card_ids(struct device *dmdev, struct device_attribute *attr, char *buf) | 157 | static ssize_t pnp_show_card_ids(struct device *dmdev, |
158 | struct device_attribute *attr, char *buf) | ||
151 | { | 159 | { |
152 | char *str = buf; | 160 | char *str = buf; |
153 | struct pnp_card *card = to_pnp_card(dmdev); | 161 | struct pnp_card *card = to_pnp_card(dmdev); |
154 | struct pnp_id * pos = card->id; | 162 | struct pnp_id *pos = card->id; |
155 | 163 | ||
156 | while (pos) { | 164 | while (pos) { |
157 | str += sprintf(str,"%s\n", pos->id); | 165 | str += sprintf(str, "%s\n", pos->id); |
158 | pos = pos->next; | 166 | pos = pos->next; |
159 | } | 167 | } |
160 | return (str - buf); | 168 | return (str - buf); |
161 | } | 169 | } |
162 | 170 | ||
163 | static DEVICE_ATTR(card_id,S_IRUGO,pnp_show_card_ids,NULL); | 171 | static DEVICE_ATTR(card_id, S_IRUGO, pnp_show_card_ids, NULL); |
164 | 172 | ||
165 | static int pnp_interface_attach_card(struct pnp_card *card) | 173 | static int pnp_interface_attach_card(struct pnp_card *card) |
166 | { | 174 | { |
167 | int rc = device_create_file(&card->dev,&dev_attr_name); | 175 | int rc = device_create_file(&card->dev, &dev_attr_name); |
168 | if (rc) return rc; | ||
169 | 176 | ||
170 | rc = device_create_file(&card->dev,&dev_attr_card_id); | 177 | if (rc) |
171 | if (rc) goto err_name; | 178 | return rc; |
179 | |||
180 | rc = device_create_file(&card->dev, &dev_attr_card_id); | ||
181 | if (rc) | ||
182 | goto err_name; | ||
172 | 183 | ||
173 | return 0; | 184 | return 0; |
174 | 185 | ||
175 | err_name: | 186 | err_name: |
176 | device_remove_file(&card->dev,&dev_attr_name); | 187 | device_remove_file(&card->dev, &dev_attr_name); |
177 | return rc; | 188 | return rc; |
178 | } | 189 | } |
179 | 190 | ||
@@ -181,15 +192,16 @@ err_name: | |||
181 | * pnp_add_card - adds a PnP card to the PnP Layer | 192 | * pnp_add_card - adds a PnP card to the PnP Layer |
182 | * @card: pointer to the card to add | 193 | * @card: pointer to the card to add |
183 | */ | 194 | */ |
184 | 195 | int pnp_add_card(struct pnp_card *card) | |
185 | int pnp_add_card(struct pnp_card * card) | ||
186 | { | 196 | { |
187 | int error; | 197 | int error; |
188 | struct list_head * pos, * temp; | 198 | struct list_head *pos, *temp; |
199 | |||
189 | if (!card || !card->protocol) | 200 | if (!card || !card->protocol) |
190 | return -EINVAL; | 201 | return -EINVAL; |
191 | 202 | ||
192 | sprintf(card->dev.bus_id, "%02x:%02x", card->protocol->number, card->number); | 203 | sprintf(card->dev.bus_id, "%02x:%02x", card->protocol->number, |
204 | card->number); | ||
193 | card->dev.parent = &card->protocol->dev; | 205 | card->dev.parent = &card->protocol->dev; |
194 | card->dev.bus = NULL; | 206 | card->dev.bus = NULL; |
195 | card->dev.release = &pnp_release_card; | 207 | card->dev.release = &pnp_release_card; |
@@ -205,18 +217,21 @@ int pnp_add_card(struct pnp_card * card) | |||
205 | /* we wait until now to add devices in order to ensure the drivers | 217 | /* we wait until now to add devices in order to ensure the drivers |
206 | * will be able to use all of the related devices on the card | 218 | * will be able to use all of the related devices on the card |
207 | * without waiting any unresonable length of time */ | 219 | * without waiting any unresonable length of time */ |
208 | list_for_each(pos,&card->devices){ | 220 | list_for_each(pos, &card->devices) { |
209 | struct pnp_dev *dev = card_to_pnp_dev(pos); | 221 | struct pnp_dev *dev = card_to_pnp_dev(pos); |
210 | __pnp_add_device(dev); | 222 | __pnp_add_device(dev); |
211 | } | 223 | } |
212 | 224 | ||
213 | /* match with card drivers */ | 225 | /* match with card drivers */ |
214 | list_for_each_safe(pos,temp,&pnp_card_drivers){ | 226 | list_for_each_safe(pos, temp, &pnp_card_drivers) { |
215 | struct pnp_card_driver * drv = list_entry(pos, struct pnp_card_driver, global_list); | 227 | struct pnp_card_driver *drv = |
216 | card_probe(card,drv); | 228 | list_entry(pos, struct pnp_card_driver, |
229 | global_list); | ||
230 | card_probe(card, drv); | ||
217 | } | 231 | } |
218 | } else | 232 | } else |
219 | pnp_err("sysfs failure, card '%s' will be unavailable", card->dev.bus_id); | 233 | pnp_err("sysfs failure, card '%s' will be unavailable", |
234 | card->dev.bus_id); | ||
220 | return error; | 235 | return error; |
221 | } | 236 | } |
222 | 237 | ||
@@ -224,10 +239,10 @@ int pnp_add_card(struct pnp_card * card) | |||
224 | * pnp_remove_card - removes a PnP card from the PnP Layer | 239 | * pnp_remove_card - removes a PnP card from the PnP Layer |
225 | * @card: pointer to the card to remove | 240 | * @card: pointer to the card to remove |
226 | */ | 241 | */ |
227 | 242 | void pnp_remove_card(struct pnp_card *card) | |
228 | void pnp_remove_card(struct pnp_card * card) | ||
229 | { | 243 | { |
230 | struct list_head *pos, *temp; | 244 | struct list_head *pos, *temp; |
245 | |||
231 | if (!card) | 246 | if (!card) |
232 | return; | 247 | return; |
233 | device_unregister(&card->dev); | 248 | device_unregister(&card->dev); |
@@ -235,7 +250,7 @@ void pnp_remove_card(struct pnp_card * card) | |||
235 | list_del(&card->global_list); | 250 | list_del(&card->global_list); |
236 | list_del(&card->protocol_list); | 251 | list_del(&card->protocol_list); |
237 | spin_unlock(&pnp_lock); | 252 | spin_unlock(&pnp_lock); |
238 | list_for_each_safe(pos,temp,&card->devices){ | 253 | list_for_each_safe(pos, temp, &card->devices) { |
239 | struct pnp_dev *dev = card_to_pnp_dev(pos); | 254 | struct pnp_dev *dev = card_to_pnp_dev(pos); |
240 | pnp_remove_card_device(dev); | 255 | pnp_remove_card_device(dev); |
241 | } | 256 | } |
@@ -246,15 +261,14 @@ void pnp_remove_card(struct pnp_card * card) | |||
246 | * @card: pointer to the card to add to | 261 | * @card: pointer to the card to add to |
247 | * @dev: pointer to the device to add | 262 | * @dev: pointer to the device to add |
248 | */ | 263 | */ |
249 | 264 | int pnp_add_card_device(struct pnp_card *card, struct pnp_dev *dev) | |
250 | int pnp_add_card_device(struct pnp_card * card, struct pnp_dev * dev) | ||
251 | { | 265 | { |
252 | if (!card || !dev || !dev->protocol) | 266 | if (!card || !dev || !dev->protocol) |
253 | return -EINVAL; | 267 | return -EINVAL; |
254 | dev->dev.parent = &card->dev; | 268 | dev->dev.parent = &card->dev; |
255 | dev->card_link = NULL; | 269 | dev->card_link = NULL; |
256 | snprintf(dev->dev.bus_id, BUS_ID_SIZE, "%02x:%02x.%02x", dev->protocol->number, | 270 | snprintf(dev->dev.bus_id, BUS_ID_SIZE, "%02x:%02x.%02x", |
257 | card->number,dev->number); | 271 | dev->protocol->number, card->number, dev->number); |
258 | spin_lock(&pnp_lock); | 272 | spin_lock(&pnp_lock); |
259 | dev->card = card; | 273 | dev->card = card; |
260 | list_add_tail(&dev->card_list, &card->devices); | 274 | list_add_tail(&dev->card_list, &card->devices); |
@@ -266,8 +280,7 @@ int pnp_add_card_device(struct pnp_card * card, struct pnp_dev * dev) | |||
266 | * pnp_remove_card_device- removes a device from the specified card | 280 | * pnp_remove_card_device- removes a device from the specified card |
267 | * @dev: pointer to the device to remove | 281 | * @dev: pointer to the device to remove |
268 | */ | 282 | */ |
269 | 283 | void pnp_remove_card_device(struct pnp_dev *dev) | |
270 | void pnp_remove_card_device(struct pnp_dev * dev) | ||
271 | { | 284 | { |
272 | spin_lock(&pnp_lock); | 285 | spin_lock(&pnp_lock); |
273 | dev->card = NULL; | 286 | dev->card = NULL; |
@@ -282,13 +295,14 @@ void pnp_remove_card_device(struct pnp_dev * dev) | |||
282 | * @id: pointer to a PnP ID structure that explains the rules for finding the device | 295 | * @id: pointer to a PnP ID structure that explains the rules for finding the device |
283 | * @from: Starting place to search from. If NULL it will start from the begining. | 296 | * @from: Starting place to search from. If NULL it will start from the begining. |
284 | */ | 297 | */ |
285 | 298 | struct pnp_dev *pnp_request_card_device(struct pnp_card_link *clink, | |
286 | struct pnp_dev * pnp_request_card_device(struct pnp_card_link *clink, const char * id, struct pnp_dev * from) | 299 | const char *id, struct pnp_dev *from) |
287 | { | 300 | { |
288 | struct list_head * pos; | 301 | struct list_head *pos; |
289 | struct pnp_dev * dev; | 302 | struct pnp_dev *dev; |
290 | struct pnp_card_driver * drv; | 303 | struct pnp_card_driver *drv; |
291 | struct pnp_card * card; | 304 | struct pnp_card *card; |
305 | |||
292 | if (!clink || !id) | 306 | if (!clink || !id) |
293 | goto done; | 307 | goto done; |
294 | card = clink->card; | 308 | card = clink->card; |
@@ -302,15 +316,15 @@ struct pnp_dev * pnp_request_card_device(struct pnp_card_link *clink, const char | |||
302 | } | 316 | } |
303 | while (pos != &card->devices) { | 317 | while (pos != &card->devices) { |
304 | dev = card_to_pnp_dev(pos); | 318 | dev = card_to_pnp_dev(pos); |
305 | if ((!dev->card_link) && compare_pnp_id(dev->id,id)) | 319 | if ((!dev->card_link) && compare_pnp_id(dev->id, id)) |
306 | goto found; | 320 | goto found; |
307 | pos = pos->next; | 321 | pos = pos->next; |
308 | } | 322 | } |
309 | 323 | ||
310 | done: | 324 | done: |
311 | return NULL; | 325 | return NULL; |
312 | 326 | ||
313 | found: | 327 | found: |
314 | dev->card_link = clink; | 328 | dev->card_link = clink; |
315 | dev->dev.driver = &drv->link.driver; | 329 | dev->dev.driver = &drv->link.driver; |
316 | if (pnp_bus_type.probe(&dev->dev)) | 330 | if (pnp_bus_type.probe(&dev->dev)) |
@@ -320,7 +334,7 @@ found: | |||
320 | 334 | ||
321 | return dev; | 335 | return dev; |
322 | 336 | ||
323 | err_out: | 337 | err_out: |
324 | dev->dev.driver = NULL; | 338 | dev->dev.driver = NULL; |
325 | dev->card_link = NULL; | 339 | dev->card_link = NULL; |
326 | return NULL; | 340 | return NULL; |
@@ -330,10 +344,10 @@ err_out: | |||
330 | * pnp_release_card_device - call this when the driver no longer needs the device | 344 | * pnp_release_card_device - call this when the driver no longer needs the device |
331 | * @dev: pointer to the PnP device stucture | 345 | * @dev: pointer to the PnP device stucture |
332 | */ | 346 | */ |
333 | 347 | void pnp_release_card_device(struct pnp_dev *dev) | |
334 | void pnp_release_card_device(struct pnp_dev * dev) | ||
335 | { | 348 | { |
336 | struct pnp_card_driver * drv = dev->card_link->driver; | 349 | struct pnp_card_driver *drv = dev->card_link->driver; |
350 | |||
337 | if (!drv) | 351 | if (!drv) |
338 | return; | 352 | return; |
339 | drv->link.remove = &card_remove; | 353 | drv->link.remove = &card_remove; |
@@ -347,6 +361,7 @@ void pnp_release_card_device(struct pnp_dev * dev) | |||
347 | static int card_suspend(struct pnp_dev *dev, pm_message_t state) | 361 | static int card_suspend(struct pnp_dev *dev, pm_message_t state) |
348 | { | 362 | { |
349 | struct pnp_card_link *link = dev->card_link; | 363 | struct pnp_card_link *link = dev->card_link; |
364 | |||
350 | if (link->pm_state.event == state.event) | 365 | if (link->pm_state.event == state.event) |
351 | return 0; | 366 | return 0; |
352 | link->pm_state = state; | 367 | link->pm_state = state; |
@@ -356,6 +371,7 @@ static int card_suspend(struct pnp_dev *dev, pm_message_t state) | |||
356 | static int card_resume(struct pnp_dev *dev) | 371 | static int card_resume(struct pnp_dev *dev) |
357 | { | 372 | { |
358 | struct pnp_card_link *link = dev->card_link; | 373 | struct pnp_card_link *link = dev->card_link; |
374 | |||
359 | if (link->pm_state.event == PM_EVENT_ON) | 375 | if (link->pm_state.event == PM_EVENT_ON) |
360 | return 0; | 376 | return 0; |
361 | link->pm_state = PMSG_ON; | 377 | link->pm_state = PMSG_ON; |
@@ -367,8 +383,7 @@ static int card_resume(struct pnp_dev *dev) | |||
367 | * pnp_register_card_driver - registers a PnP card driver with the PnP Layer | 383 | * pnp_register_card_driver - registers a PnP card driver with the PnP Layer |
368 | * @drv: pointer to the driver to register | 384 | * @drv: pointer to the driver to register |
369 | */ | 385 | */ |
370 | 386 | int pnp_register_card_driver(struct pnp_card_driver *drv) | |
371 | int pnp_register_card_driver(struct pnp_card_driver * drv) | ||
372 | { | 387 | { |
373 | int error; | 388 | int error; |
374 | struct list_head *pos, *temp; | 389 | struct list_head *pos, *temp; |
@@ -389,9 +404,10 @@ int pnp_register_card_driver(struct pnp_card_driver * drv) | |||
389 | list_add_tail(&drv->global_list, &pnp_card_drivers); | 404 | list_add_tail(&drv->global_list, &pnp_card_drivers); |
390 | spin_unlock(&pnp_lock); | 405 | spin_unlock(&pnp_lock); |
391 | 406 | ||
392 | list_for_each_safe(pos,temp,&pnp_cards){ | 407 | list_for_each_safe(pos, temp, &pnp_cards) { |
393 | struct pnp_card *card = list_entry(pos, struct pnp_card, global_list); | 408 | struct pnp_card *card = |
394 | card_probe(card,drv); | 409 | list_entry(pos, struct pnp_card, global_list); |
410 | card_probe(card, drv); | ||
395 | } | 411 | } |
396 | return 0; | 412 | return 0; |
397 | } | 413 | } |
@@ -400,8 +416,7 @@ int pnp_register_card_driver(struct pnp_card_driver * drv) | |||
400 | * pnp_unregister_card_driver - unregisters a PnP card driver from the PnP Layer | 416 | * pnp_unregister_card_driver - unregisters a PnP card driver from the PnP Layer |
401 | * @drv: pointer to the driver to unregister | 417 | * @drv: pointer to the driver to unregister |
402 | */ | 418 | */ |
403 | 419 | void pnp_unregister_card_driver(struct pnp_card_driver *drv) | |
404 | void pnp_unregister_card_driver(struct pnp_card_driver * drv) | ||
405 | { | 420 | { |
406 | spin_lock(&pnp_lock); | 421 | spin_lock(&pnp_lock); |
407 | list_del(&drv->global_list); | 422 | list_del(&drv->global_list); |
@@ -409,13 +424,6 @@ void pnp_unregister_card_driver(struct pnp_card_driver * drv) | |||
409 | pnp_unregister_driver(&drv->link); | 424 | pnp_unregister_driver(&drv->link); |
410 | } | 425 | } |
411 | 426 | ||
412 | #if 0 | ||
413 | EXPORT_SYMBOL(pnp_add_card); | ||
414 | EXPORT_SYMBOL(pnp_remove_card); | ||
415 | EXPORT_SYMBOL(pnp_add_card_device); | ||
416 | EXPORT_SYMBOL(pnp_remove_card_device); | ||
417 | EXPORT_SYMBOL(pnp_add_card_id); | ||
418 | #endif /* 0 */ | ||
419 | EXPORT_SYMBOL(pnp_request_card_device); | 427 | EXPORT_SYMBOL(pnp_request_card_device); |
420 | EXPORT_SYMBOL(pnp_release_card_device); | 428 | EXPORT_SYMBOL(pnp_release_card_device); |
421 | EXPORT_SYMBOL(pnp_register_card_driver); | 429 | EXPORT_SYMBOL(pnp_register_card_driver); |
diff --git a/drivers/pnp/core.c b/drivers/pnp/core.c index 8e7b2dd38810..61066fdb9e6d 100644 --- a/drivers/pnp/core.c +++ b/drivers/pnp/core.c | |||
@@ -2,7 +2,6 @@ | |||
2 | * core.c - contains all core device and protocol registration functions | 2 | * core.c - contains all core device and protocol registration functions |
3 | * | 3 | * |
4 | * Copyright 2002 Adam Belay <ambx1@neo.rr.com> | 4 | * Copyright 2002 Adam Belay <ambx1@neo.rr.com> |
5 | * | ||
6 | */ | 5 | */ |
7 | 6 | ||
8 | #include <linux/pnp.h> | 7 | #include <linux/pnp.h> |
@@ -18,7 +17,6 @@ | |||
18 | 17 | ||
19 | #include "base.h" | 18 | #include "base.h" |
20 | 19 | ||
21 | |||
22 | static LIST_HEAD(pnp_protocols); | 20 | static LIST_HEAD(pnp_protocols); |
23 | LIST_HEAD(pnp_global); | 21 | LIST_HEAD(pnp_global); |
24 | DEFINE_SPINLOCK(pnp_lock); | 22 | DEFINE_SPINLOCK(pnp_lock); |
@@ -36,7 +34,7 @@ void *pnp_alloc(long size) | |||
36 | void *result; | 34 | void *result; |
37 | 35 | ||
38 | result = kzalloc(size, GFP_KERNEL); | 36 | result = kzalloc(size, GFP_KERNEL); |
39 | if (!result){ | 37 | if (!result) { |
40 | printk(KERN_ERR "pnp: Out of Memory\n"); | 38 | printk(KERN_ERR "pnp: Out of Memory\n"); |
41 | return NULL; | 39 | return NULL; |
42 | } | 40 | } |
@@ -49,11 +47,10 @@ void *pnp_alloc(long size) | |||
49 | * | 47 | * |
50 | * Ex protocols: ISAPNP, PNPBIOS, etc | 48 | * Ex protocols: ISAPNP, PNPBIOS, etc |
51 | */ | 49 | */ |
52 | |||
53 | int pnp_register_protocol(struct pnp_protocol *protocol) | 50 | int pnp_register_protocol(struct pnp_protocol *protocol) |
54 | { | 51 | { |
55 | int nodenum; | 52 | int nodenum; |
56 | struct list_head * pos; | 53 | struct list_head *pos; |
57 | 54 | ||
58 | if (!protocol) | 55 | if (!protocol) |
59 | return -EINVAL; | 56 | return -EINVAL; |
@@ -64,9 +61,9 @@ int pnp_register_protocol(struct pnp_protocol *protocol) | |||
64 | spin_lock(&pnp_lock); | 61 | spin_lock(&pnp_lock); |
65 | 62 | ||
66 | /* assign the lowest unused number */ | 63 | /* assign the lowest unused number */ |
67 | list_for_each(pos,&pnp_protocols) { | 64 | list_for_each(pos, &pnp_protocols) { |
68 | struct pnp_protocol * cur = to_pnp_protocol(pos); | 65 | struct pnp_protocol *cur = to_pnp_protocol(pos); |
69 | if (cur->number == nodenum){ | 66 | if (cur->number == nodenum) { |
70 | pos = &pnp_protocols; | 67 | pos = &pnp_protocols; |
71 | nodenum++; | 68 | nodenum++; |
72 | } | 69 | } |
@@ -83,7 +80,6 @@ int pnp_register_protocol(struct pnp_protocol *protocol) | |||
83 | /** | 80 | /** |
84 | * pnp_protocol_unregister - removes a pnp protocol from the pnp layer | 81 | * pnp_protocol_unregister - removes a pnp protocol from the pnp layer |
85 | * @protocol: pointer to the corresponding pnp_protocol structure | 82 | * @protocol: pointer to the corresponding pnp_protocol structure |
86 | * | ||
87 | */ | 83 | */ |
88 | void pnp_unregister_protocol(struct pnp_protocol *protocol) | 84 | void pnp_unregister_protocol(struct pnp_protocol *protocol) |
89 | { | 85 | { |
@@ -93,11 +89,11 @@ void pnp_unregister_protocol(struct pnp_protocol *protocol) | |||
93 | device_unregister(&protocol->dev); | 89 | device_unregister(&protocol->dev); |
94 | } | 90 | } |
95 | 91 | ||
96 | |||
97 | static void pnp_free_ids(struct pnp_dev *dev) | 92 | static void pnp_free_ids(struct pnp_dev *dev) |
98 | { | 93 | { |
99 | struct pnp_id * id; | 94 | struct pnp_id *id; |
100 | struct pnp_id * next; | 95 | struct pnp_id *next; |
96 | |||
101 | if (!dev) | 97 | if (!dev) |
102 | return; | 98 | return; |
103 | id = dev->id; | 99 | id = dev->id; |
@@ -110,7 +106,8 @@ static void pnp_free_ids(struct pnp_dev *dev) | |||
110 | 106 | ||
111 | static void pnp_release_device(struct device *dmdev) | 107 | static void pnp_release_device(struct device *dmdev) |
112 | { | 108 | { |
113 | struct pnp_dev * dev = to_pnp_dev(dmdev); | 109 | struct pnp_dev *dev = to_pnp_dev(dmdev); |
110 | |||
114 | pnp_free_option(dev->independent); | 111 | pnp_free_option(dev->independent); |
115 | pnp_free_option(dev->dependent); | 112 | pnp_free_option(dev->dependent); |
116 | pnp_free_ids(dev); | 113 | pnp_free_ids(dev); |
@@ -120,6 +117,7 @@ static void pnp_release_device(struct device *dmdev) | |||
120 | int __pnp_add_device(struct pnp_dev *dev) | 117 | int __pnp_add_device(struct pnp_dev *dev) |
121 | { | 118 | { |
122 | int ret; | 119 | int ret; |
120 | |||
123 | pnp_fixup_device(dev); | 121 | pnp_fixup_device(dev); |
124 | dev->dev.bus = &pnp_bus_type; | 122 | dev->dev.bus = &pnp_bus_type; |
125 | dev->dev.dma_mask = &dev->dma_mask; | 123 | dev->dev.dma_mask = &dev->dma_mask; |
@@ -143,13 +141,13 @@ int __pnp_add_device(struct pnp_dev *dev) | |||
143 | * | 141 | * |
144 | * adds to driver model, name database, fixups, interface, etc. | 142 | * adds to driver model, name database, fixups, interface, etc. |
145 | */ | 143 | */ |
146 | |||
147 | int pnp_add_device(struct pnp_dev *dev) | 144 | int pnp_add_device(struct pnp_dev *dev) |
148 | { | 145 | { |
149 | if (!dev || !dev->protocol || dev->card) | 146 | if (!dev || !dev->protocol || dev->card) |
150 | return -EINVAL; | 147 | return -EINVAL; |
151 | dev->dev.parent = &dev->protocol->dev; | 148 | dev->dev.parent = &dev->protocol->dev; |
152 | sprintf(dev->dev.bus_id, "%02x:%02x", dev->protocol->number, dev->number); | 149 | sprintf(dev->dev.bus_id, "%02x:%02x", dev->protocol->number, |
150 | dev->number); | ||
153 | return __pnp_add_device(dev); | 151 | return __pnp_add_device(dev); |
154 | } | 152 | } |
155 | 153 | ||
@@ -162,21 +160,6 @@ void __pnp_remove_device(struct pnp_dev *dev) | |||
162 | device_unregister(&dev->dev); | 160 | device_unregister(&dev->dev); |
163 | } | 161 | } |
164 | 162 | ||
165 | /** | ||
166 | * pnp_remove_device - removes a pnp device from the pnp layer | ||
167 | * @dev: pointer to dev to add | ||
168 | * | ||
169 | * this function will free all mem used by dev | ||
170 | */ | ||
171 | #if 0 | ||
172 | void pnp_remove_device(struct pnp_dev *dev) | ||
173 | { | ||
174 | if (!dev || dev->card) | ||
175 | return; | ||
176 | __pnp_remove_device(dev); | ||
177 | } | ||
178 | #endif /* 0 */ | ||
179 | |||
180 | static int __init pnp_init(void) | 163 | static int __init pnp_init(void) |
181 | { | 164 | { |
182 | printk(KERN_INFO "Linux Plug and Play Support v0.97 (c) Adam Belay\n"); | 165 | printk(KERN_INFO "Linux Plug and Play Support v0.97 (c) Adam Belay\n"); |
@@ -184,10 +167,3 @@ static int __init pnp_init(void) | |||
184 | } | 167 | } |
185 | 168 | ||
186 | subsys_initcall(pnp_init); | 169 | subsys_initcall(pnp_init); |
187 | |||
188 | #if 0 | ||
189 | EXPORT_SYMBOL(pnp_register_protocol); | ||
190 | EXPORT_SYMBOL(pnp_unregister_protocol); | ||
191 | EXPORT_SYMBOL(pnp_add_device); | ||
192 | EXPORT_SYMBOL(pnp_remove_device); | ||
193 | #endif /* 0 */ | ||
diff --git a/drivers/pnp/driver.c b/drivers/pnp/driver.c index e161423b4300..30b8f6f3258a 100644 --- a/drivers/pnp/driver.c +++ b/drivers/pnp/driver.c | |||
@@ -2,7 +2,6 @@ | |||
2 | * driver.c - device id matching, driver model, etc. | 2 | * driver.c - device id matching, driver model, etc. |
3 | * | 3 | * |
4 | * Copyright 2002 Adam Belay <ambx1@neo.rr.com> | 4 | * Copyright 2002 Adam Belay <ambx1@neo.rr.com> |
5 | * | ||
6 | */ | 5 | */ |
7 | 6 | ||
8 | #include <linux/string.h> | 7 | #include <linux/string.h> |
@@ -16,12 +15,11 @@ | |||
16 | static int compare_func(const char *ida, const char *idb) | 15 | static int compare_func(const char *ida, const char *idb) |
17 | { | 16 | { |
18 | int i; | 17 | int i; |
18 | |||
19 | /* we only need to compare the last 4 chars */ | 19 | /* we only need to compare the last 4 chars */ |
20 | for (i=3; i<7; i++) | 20 | for (i = 3; i < 7; i++) { |
21 | { | ||
22 | if (ida[i] != 'X' && | 21 | if (ida[i] != 'X' && |
23 | idb[i] != 'X' && | 22 | idb[i] != 'X' && toupper(ida[i]) != toupper(idb[i])) |
24 | toupper(ida[i]) != toupper(idb[i])) | ||
25 | return 0; | 23 | return 0; |
26 | } | 24 | } |
27 | return 1; | 25 | return 1; |
@@ -31,20 +29,22 @@ int compare_pnp_id(struct pnp_id *pos, const char *id) | |||
31 | { | 29 | { |
32 | if (!pos || !id || (strlen(id) != 7)) | 30 | if (!pos || !id || (strlen(id) != 7)) |
33 | return 0; | 31 | return 0; |
34 | if (memcmp(id,"ANYDEVS",7)==0) | 32 | if (memcmp(id, "ANYDEVS", 7) == 0) |
35 | return 1; | 33 | return 1; |
36 | while (pos){ | 34 | while (pos) { |
37 | if (memcmp(pos->id,id,3)==0) | 35 | if (memcmp(pos->id, id, 3) == 0) |
38 | if (compare_func(pos->id,id)==1) | 36 | if (compare_func(pos->id, id) == 1) |
39 | return 1; | 37 | return 1; |
40 | pos = pos->next; | 38 | pos = pos->next; |
41 | } | 39 | } |
42 | return 0; | 40 | return 0; |
43 | } | 41 | } |
44 | 42 | ||
45 | static const struct pnp_device_id * match_device(struct pnp_driver *drv, struct pnp_dev *dev) | 43 | static const struct pnp_device_id *match_device(struct pnp_driver *drv, |
44 | struct pnp_dev *dev) | ||
46 | { | 45 | { |
47 | const struct pnp_device_id *drv_id = drv->id_table; | 46 | const struct pnp_device_id *drv_id = drv->id_table; |
47 | |||
48 | if (!drv_id) | 48 | if (!drv_id) |
49 | return NULL; | 49 | return NULL; |
50 | 50 | ||
@@ -59,7 +59,7 @@ static const struct pnp_device_id * match_device(struct pnp_driver *drv, struct | |||
59 | int pnp_device_attach(struct pnp_dev *pnp_dev) | 59 | int pnp_device_attach(struct pnp_dev *pnp_dev) |
60 | { | 60 | { |
61 | spin_lock(&pnp_lock); | 61 | spin_lock(&pnp_lock); |
62 | if(pnp_dev->status != PNP_READY){ | 62 | if (pnp_dev->status != PNP_READY) { |
63 | spin_unlock(&pnp_lock); | 63 | spin_unlock(&pnp_lock); |
64 | return -EBUSY; | 64 | return -EBUSY; |
65 | } | 65 | } |
@@ -86,7 +86,8 @@ static int pnp_device_probe(struct device *dev) | |||
86 | pnp_dev = to_pnp_dev(dev); | 86 | pnp_dev = to_pnp_dev(dev); |
87 | pnp_drv = to_pnp_driver(dev->driver); | 87 | pnp_drv = to_pnp_driver(dev->driver); |
88 | 88 | ||
89 | pnp_dbg("match found with the PnP device '%s' and the driver '%s'", dev->bus_id,pnp_drv->name); | 89 | pnp_dbg("match found with the PnP device '%s' and the driver '%s'", |
90 | dev->bus_id, pnp_drv->name); | ||
90 | 91 | ||
91 | error = pnp_device_attach(pnp_dev); | 92 | error = pnp_device_attach(pnp_dev); |
92 | if (error < 0) | 93 | if (error < 0) |
@@ -99,7 +100,7 @@ static int pnp_device_probe(struct device *dev) | |||
99 | return error; | 100 | return error; |
100 | } | 101 | } |
101 | } else if ((pnp_drv->flags & PNP_DRIVER_RES_DISABLE) | 102 | } else if ((pnp_drv->flags & PNP_DRIVER_RES_DISABLE) |
102 | == PNP_DRIVER_RES_DISABLE) { | 103 | == PNP_DRIVER_RES_DISABLE) { |
103 | error = pnp_disable_dev(pnp_dev); | 104 | error = pnp_disable_dev(pnp_dev); |
104 | if (error < 0) | 105 | if (error < 0) |
105 | return error; | 106 | return error; |
@@ -110,22 +111,22 @@ static int pnp_device_probe(struct device *dev) | |||
110 | if (dev_id != NULL) | 111 | if (dev_id != NULL) |
111 | error = pnp_drv->probe(pnp_dev, dev_id); | 112 | error = pnp_drv->probe(pnp_dev, dev_id); |
112 | } | 113 | } |
113 | if (error >= 0){ | 114 | if (error >= 0) { |
114 | pnp_dev->driver = pnp_drv; | 115 | pnp_dev->driver = pnp_drv; |
115 | error = 0; | 116 | error = 0; |
116 | } else | 117 | } else |
117 | goto fail; | 118 | goto fail; |
118 | return error; | 119 | return error; |
119 | 120 | ||
120 | fail: | 121 | fail: |
121 | pnp_device_detach(pnp_dev); | 122 | pnp_device_detach(pnp_dev); |
122 | return error; | 123 | return error; |
123 | } | 124 | } |
124 | 125 | ||
125 | static int pnp_device_remove(struct device *dev) | 126 | static int pnp_device_remove(struct device *dev) |
126 | { | 127 | { |
127 | struct pnp_dev * pnp_dev = to_pnp_dev(dev); | 128 | struct pnp_dev *pnp_dev = to_pnp_dev(dev); |
128 | struct pnp_driver * drv = pnp_dev->driver; | 129 | struct pnp_driver *drv = pnp_dev->driver; |
129 | 130 | ||
130 | if (drv) { | 131 | if (drv) { |
131 | if (drv->remove) | 132 | if (drv->remove) |
@@ -138,8 +139,9 @@ static int pnp_device_remove(struct device *dev) | |||
138 | 139 | ||
139 | static int pnp_bus_match(struct device *dev, struct device_driver *drv) | 140 | static int pnp_bus_match(struct device *dev, struct device_driver *drv) |
140 | { | 141 | { |
141 | struct pnp_dev * pnp_dev = to_pnp_dev(dev); | 142 | struct pnp_dev *pnp_dev = to_pnp_dev(dev); |
142 | struct pnp_driver * pnp_drv = to_pnp_driver(drv); | 143 | struct pnp_driver *pnp_drv = to_pnp_driver(drv); |
144 | |||
143 | if (match_device(pnp_drv, pnp_dev) == NULL) | 145 | if (match_device(pnp_drv, pnp_dev) == NULL) |
144 | return 0; | 146 | return 0; |
145 | return 1; | 147 | return 1; |
@@ -147,8 +149,8 @@ static int pnp_bus_match(struct device *dev, struct device_driver *drv) | |||
147 | 149 | ||
148 | static int pnp_bus_suspend(struct device *dev, pm_message_t state) | 150 | static int pnp_bus_suspend(struct device *dev, pm_message_t state) |
149 | { | 151 | { |
150 | struct pnp_dev * pnp_dev = to_pnp_dev(dev); | 152 | struct pnp_dev *pnp_dev = to_pnp_dev(dev); |
151 | struct pnp_driver * pnp_drv = pnp_dev->driver; | 153 | struct pnp_driver *pnp_drv = pnp_dev->driver; |
152 | int error; | 154 | int error; |
153 | 155 | ||
154 | if (!pnp_drv) | 156 | if (!pnp_drv) |
@@ -162,23 +164,28 @@ static int pnp_bus_suspend(struct device *dev, pm_message_t state) | |||
162 | 164 | ||
163 | if (!(pnp_drv->flags & PNP_DRIVER_RES_DO_NOT_CHANGE) && | 165 | if (!(pnp_drv->flags & PNP_DRIVER_RES_DO_NOT_CHANGE) && |
164 | pnp_can_disable(pnp_dev)) { | 166 | pnp_can_disable(pnp_dev)) { |
165 | error = pnp_stop_dev(pnp_dev); | 167 | error = pnp_stop_dev(pnp_dev); |
166 | if (error) | 168 | if (error) |
167 | return error; | 169 | return error; |
168 | } | 170 | } |
169 | 171 | ||
172 | if (pnp_dev->protocol && pnp_dev->protocol->suspend) | ||
173 | pnp_dev->protocol->suspend(pnp_dev, state); | ||
170 | return 0; | 174 | return 0; |
171 | } | 175 | } |
172 | 176 | ||
173 | static int pnp_bus_resume(struct device *dev) | 177 | static int pnp_bus_resume(struct device *dev) |
174 | { | 178 | { |
175 | struct pnp_dev * pnp_dev = to_pnp_dev(dev); | 179 | struct pnp_dev *pnp_dev = to_pnp_dev(dev); |
176 | struct pnp_driver * pnp_drv = pnp_dev->driver; | 180 | struct pnp_driver *pnp_drv = pnp_dev->driver; |
177 | int error; | 181 | int error; |
178 | 182 | ||
179 | if (!pnp_drv) | 183 | if (!pnp_drv) |
180 | return 0; | 184 | return 0; |
181 | 185 | ||
186 | if (pnp_dev->protocol && pnp_dev->protocol->resume) | ||
187 | pnp_dev->protocol->resume(pnp_dev); | ||
188 | |||
182 | if (!(pnp_drv->flags & PNP_DRIVER_RES_DO_NOT_CHANGE)) { | 189 | if (!(pnp_drv->flags & PNP_DRIVER_RES_DO_NOT_CHANGE)) { |
183 | error = pnp_start_dev(pnp_dev); | 190 | error = pnp_start_dev(pnp_dev); |
184 | if (error) | 191 | if (error) |
@@ -192,12 +199,12 @@ static int pnp_bus_resume(struct device *dev) | |||
192 | } | 199 | } |
193 | 200 | ||
194 | struct bus_type pnp_bus_type = { | 201 | struct bus_type pnp_bus_type = { |
195 | .name = "pnp", | 202 | .name = "pnp", |
196 | .match = pnp_bus_match, | 203 | .match = pnp_bus_match, |
197 | .probe = pnp_device_probe, | 204 | .probe = pnp_device_probe, |
198 | .remove = pnp_device_remove, | 205 | .remove = pnp_device_remove, |
199 | .suspend = pnp_bus_suspend, | 206 | .suspend = pnp_bus_suspend, |
200 | .resume = pnp_bus_resume, | 207 | .resume = pnp_bus_resume, |
201 | }; | 208 | }; |
202 | 209 | ||
203 | int pnp_register_driver(struct pnp_driver *drv) | 210 | int pnp_register_driver(struct pnp_driver *drv) |
@@ -220,12 +227,11 @@ void pnp_unregister_driver(struct pnp_driver *drv) | |||
220 | * pnp_add_id - adds an EISA id to the specified device | 227 | * pnp_add_id - adds an EISA id to the specified device |
221 | * @id: pointer to a pnp_id structure | 228 | * @id: pointer to a pnp_id structure |
222 | * @dev: pointer to the desired device | 229 | * @dev: pointer to the desired device |
223 | * | ||
224 | */ | 230 | */ |
225 | |||
226 | int pnp_add_id(struct pnp_id *id, struct pnp_dev *dev) | 231 | int pnp_add_id(struct pnp_id *id, struct pnp_dev *dev) |
227 | { | 232 | { |
228 | struct pnp_id *ptr; | 233 | struct pnp_id *ptr; |
234 | |||
229 | if (!id) | 235 | if (!id) |
230 | return -EINVAL; | 236 | return -EINVAL; |
231 | if (!dev) | 237 | if (!dev) |
@@ -243,8 +249,5 @@ int pnp_add_id(struct pnp_id *id, struct pnp_dev *dev) | |||
243 | 249 | ||
244 | EXPORT_SYMBOL(pnp_register_driver); | 250 | EXPORT_SYMBOL(pnp_register_driver); |
245 | EXPORT_SYMBOL(pnp_unregister_driver); | 251 | EXPORT_SYMBOL(pnp_unregister_driver); |
246 | #if 0 | ||
247 | EXPORT_SYMBOL(pnp_add_id); | ||
248 | #endif | ||
249 | EXPORT_SYMBOL(pnp_device_attach); | 252 | EXPORT_SYMBOL(pnp_device_attach); |
250 | EXPORT_SYMBOL(pnp_device_detach); | 253 | EXPORT_SYMBOL(pnp_device_detach); |
diff --git a/drivers/pnp/interface.c b/drivers/pnp/interface.c index ac9fcd499f3f..fe6684e13e82 100644 --- a/drivers/pnp/interface.c +++ b/drivers/pnp/interface.c | |||
@@ -3,7 +3,6 @@ | |||
3 | * | 3 | * |
4 | * Some code, especially possible resource dumping is based on isapnp_proc.c (c) Jaroslav Kysela <perex@suse.cz> | 4 | * Some code, especially possible resource dumping is based on isapnp_proc.c (c) Jaroslav Kysela <perex@suse.cz> |
5 | * Copyright 2002 Adam Belay <ambx1@neo.rr.com> | 5 | * Copyright 2002 Adam Belay <ambx1@neo.rr.com> |
6 | * | ||
7 | */ | 6 | */ |
8 | 7 | ||
9 | #include <linux/pnp.h> | 8 | #include <linux/pnp.h> |
@@ -29,7 +28,7 @@ struct pnp_info_buffer { | |||
29 | 28 | ||
30 | typedef struct pnp_info_buffer pnp_info_buffer_t; | 29 | typedef struct pnp_info_buffer pnp_info_buffer_t; |
31 | 30 | ||
32 | static int pnp_printf(pnp_info_buffer_t * buffer, char *fmt,...) | 31 | static int pnp_printf(pnp_info_buffer_t * buffer, char *fmt, ...) |
33 | { | 32 | { |
34 | va_list args; | 33 | va_list args; |
35 | int res; | 34 | int res; |
@@ -48,14 +47,18 @@ static int pnp_printf(pnp_info_buffer_t * buffer, char *fmt,...) | |||
48 | return res; | 47 | return res; |
49 | } | 48 | } |
50 | 49 | ||
51 | static void pnp_print_port(pnp_info_buffer_t *buffer, char *space, struct pnp_port *port) | 50 | static void pnp_print_port(pnp_info_buffer_t * buffer, char *space, |
51 | struct pnp_port *port) | ||
52 | { | 52 | { |
53 | pnp_printf(buffer, "%sport 0x%x-0x%x, align 0x%x, size 0x%x, %i-bit address decoding\n", | 53 | pnp_printf(buffer, |
54 | space, port->min, port->max, port->align ? (port->align-1) : 0, port->size, | 54 | "%sport 0x%x-0x%x, align 0x%x, size 0x%x, %i-bit address decoding\n", |
55 | port->flags & PNP_PORT_FLAG_16BITADDR ? 16 : 10); | 55 | space, port->min, port->max, |
56 | port->align ? (port->align - 1) : 0, port->size, | ||
57 | port->flags & PNP_PORT_FLAG_16BITADDR ? 16 : 10); | ||
56 | } | 58 | } |
57 | 59 | ||
58 | static void pnp_print_irq(pnp_info_buffer_t *buffer, char *space, struct pnp_irq *irq) | 60 | static void pnp_print_irq(pnp_info_buffer_t * buffer, char *space, |
61 | struct pnp_irq *irq) | ||
59 | { | 62 | { |
60 | int first = 1, i; | 63 | int first = 1, i; |
61 | 64 | ||
@@ -85,14 +88,15 @@ static void pnp_print_irq(pnp_info_buffer_t *buffer, char *space, struct pnp_irq | |||
85 | pnp_printf(buffer, "\n"); | 88 | pnp_printf(buffer, "\n"); |
86 | } | 89 | } |
87 | 90 | ||
88 | static void pnp_print_dma(pnp_info_buffer_t *buffer, char *space, struct pnp_dma *dma) | 91 | static void pnp_print_dma(pnp_info_buffer_t * buffer, char *space, |
92 | struct pnp_dma *dma) | ||
89 | { | 93 | { |
90 | int first = 1, i; | 94 | int first = 1, i; |
91 | char *s; | 95 | char *s; |
92 | 96 | ||
93 | pnp_printf(buffer, "%sdma ", space); | 97 | pnp_printf(buffer, "%sdma ", space); |
94 | for (i = 0; i < 8; i++) | 98 | for (i = 0; i < 8; i++) |
95 | if (dma->map & (1<<i)) { | 99 | if (dma->map & (1 << i)) { |
96 | if (!first) { | 100 | if (!first) { |
97 | pnp_printf(buffer, ","); | 101 | pnp_printf(buffer, ","); |
98 | } else { | 102 | } else { |
@@ -136,12 +140,13 @@ static void pnp_print_dma(pnp_info_buffer_t *buffer, char *space, struct pnp_dma | |||
136 | pnp_printf(buffer, " %s\n", s); | 140 | pnp_printf(buffer, " %s\n", s); |
137 | } | 141 | } |
138 | 142 | ||
139 | static void pnp_print_mem(pnp_info_buffer_t *buffer, char *space, struct pnp_mem *mem) | 143 | static void pnp_print_mem(pnp_info_buffer_t * buffer, char *space, |
144 | struct pnp_mem *mem) | ||
140 | { | 145 | { |
141 | char *s; | 146 | char *s; |
142 | 147 | ||
143 | pnp_printf(buffer, "%sMemory 0x%x-0x%x, align 0x%x, size 0x%x", | 148 | pnp_printf(buffer, "%sMemory 0x%x-0x%x, align 0x%x, size 0x%x", |
144 | space, mem->min, mem->max, mem->align, mem->size); | 149 | space, mem->min, mem->max, mem->align, mem->size); |
145 | if (mem->flags & IORESOURCE_MEM_WRITEABLE) | 150 | if (mem->flags & IORESOURCE_MEM_WRITEABLE) |
146 | pnp_printf(buffer, ", writeable"); | 151 | pnp_printf(buffer, ", writeable"); |
147 | if (mem->flags & IORESOURCE_MEM_CACHEABLE) | 152 | if (mem->flags & IORESOURCE_MEM_CACHEABLE) |
@@ -168,7 +173,7 @@ static void pnp_print_mem(pnp_info_buffer_t *buffer, char *space, struct pnp_mem | |||
168 | pnp_printf(buffer, ", %s\n", s); | 173 | pnp_printf(buffer, ", %s\n", s); |
169 | } | 174 | } |
170 | 175 | ||
171 | static void pnp_print_option(pnp_info_buffer_t *buffer, char *space, | 176 | static void pnp_print_option(pnp_info_buffer_t * buffer, char *space, |
172 | struct pnp_option *option, int dep) | 177 | struct pnp_option *option, int dep) |
173 | { | 178 | { |
174 | char *s; | 179 | char *s; |
@@ -179,19 +184,19 @@ static void pnp_print_option(pnp_info_buffer_t *buffer, char *space, | |||
179 | 184 | ||
180 | if (dep) { | 185 | if (dep) { |
181 | switch (option->priority) { | 186 | switch (option->priority) { |
182 | case PNP_RES_PRIORITY_PREFERRED: | 187 | case PNP_RES_PRIORITY_PREFERRED: |
183 | s = "preferred"; | 188 | s = "preferred"; |
184 | break; | 189 | break; |
185 | case PNP_RES_PRIORITY_ACCEPTABLE: | 190 | case PNP_RES_PRIORITY_ACCEPTABLE: |
186 | s = "acceptable"; | 191 | s = "acceptable"; |
187 | break; | 192 | break; |
188 | case PNP_RES_PRIORITY_FUNCTIONAL: | 193 | case PNP_RES_PRIORITY_FUNCTIONAL: |
189 | s = "functional"; | 194 | s = "functional"; |
190 | break; | 195 | break; |
191 | default: | 196 | default: |
192 | s = "invalid"; | 197 | s = "invalid"; |
193 | } | 198 | } |
194 | pnp_printf(buffer, "Dependent: %02i - Priority %s\n",dep, s); | 199 | pnp_printf(buffer, "Dependent: %02i - Priority %s\n", dep, s); |
195 | } | 200 | } |
196 | 201 | ||
197 | for (port = option->port; port; port = port->next) | 202 | for (port = option->port; port; port = port->next) |
@@ -204,16 +209,16 @@ static void pnp_print_option(pnp_info_buffer_t *buffer, char *space, | |||
204 | pnp_print_mem(buffer, space, mem); | 209 | pnp_print_mem(buffer, space, mem); |
205 | } | 210 | } |
206 | 211 | ||
207 | 212 | static ssize_t pnp_show_options(struct device *dmdev, | |
208 | static ssize_t pnp_show_options(struct device *dmdev, struct device_attribute *attr, char *buf) | 213 | struct device_attribute *attr, char *buf) |
209 | { | 214 | { |
210 | struct pnp_dev *dev = to_pnp_dev(dmdev); | 215 | struct pnp_dev *dev = to_pnp_dev(dmdev); |
211 | struct pnp_option * independent = dev->independent; | 216 | struct pnp_option *independent = dev->independent; |
212 | struct pnp_option * dependent = dev->dependent; | 217 | struct pnp_option *dependent = dev->dependent; |
213 | int ret, dep = 1; | 218 | int ret, dep = 1; |
214 | 219 | ||
215 | pnp_info_buffer_t *buffer = (pnp_info_buffer_t *) | 220 | pnp_info_buffer_t *buffer = (pnp_info_buffer_t *) |
216 | pnp_alloc(sizeof(pnp_info_buffer_t)); | 221 | pnp_alloc(sizeof(pnp_info_buffer_t)); |
217 | if (!buffer) | 222 | if (!buffer) |
218 | return -ENOMEM; | 223 | return -ENOMEM; |
219 | 224 | ||
@@ -223,7 +228,7 @@ static ssize_t pnp_show_options(struct device *dmdev, struct device_attribute *a | |||
223 | if (independent) | 228 | if (independent) |
224 | pnp_print_option(buffer, "", independent, 0); | 229 | pnp_print_option(buffer, "", independent, 0); |
225 | 230 | ||
226 | while (dependent){ | 231 | while (dependent) { |
227 | pnp_print_option(buffer, " ", dependent, dep); | 232 | pnp_print_option(buffer, " ", dependent, dep); |
228 | dependent = dependent->next; | 233 | dependent = dependent->next; |
229 | dep++; | 234 | dep++; |
@@ -233,10 +238,11 @@ static ssize_t pnp_show_options(struct device *dmdev, struct device_attribute *a | |||
233 | return ret; | 238 | return ret; |
234 | } | 239 | } |
235 | 240 | ||
236 | static DEVICE_ATTR(options,S_IRUGO,pnp_show_options,NULL); | 241 | static DEVICE_ATTR(options, S_IRUGO, pnp_show_options, NULL); |
237 | 242 | ||
238 | 243 | static ssize_t pnp_show_current_resources(struct device *dmdev, | |
239 | static ssize_t pnp_show_current_resources(struct device *dmdev, struct device_attribute *attr, char *buf) | 244 | struct device_attribute *attr, |
245 | char *buf) | ||
240 | { | 246 | { |
241 | struct pnp_dev *dev = to_pnp_dev(dmdev); | 247 | struct pnp_dev *dev = to_pnp_dev(dmdev); |
242 | int i, ret; | 248 | int i, ret; |
@@ -252,52 +258,56 @@ static ssize_t pnp_show_current_resources(struct device *dmdev, struct device_at | |||
252 | buffer->buffer = buf; | 258 | buffer->buffer = buf; |
253 | buffer->curr = buffer->buffer; | 259 | buffer->curr = buffer->buffer; |
254 | 260 | ||
255 | pnp_printf(buffer,"state = "); | 261 | pnp_printf(buffer, "state = "); |
256 | if (dev->active) | 262 | if (dev->active) |
257 | pnp_printf(buffer,"active\n"); | 263 | pnp_printf(buffer, "active\n"); |
258 | else | 264 | else |
259 | pnp_printf(buffer,"disabled\n"); | 265 | pnp_printf(buffer, "disabled\n"); |
260 | 266 | ||
261 | for (i = 0; i < PNP_MAX_PORT; i++) { | 267 | for (i = 0; i < PNP_MAX_PORT; i++) { |
262 | if (pnp_port_valid(dev, i)) { | 268 | if (pnp_port_valid(dev, i)) { |
263 | pnp_printf(buffer,"io"); | 269 | pnp_printf(buffer, "io"); |
264 | if (pnp_port_flags(dev, i) & IORESOURCE_DISABLED) | 270 | if (pnp_port_flags(dev, i) & IORESOURCE_DISABLED) |
265 | pnp_printf(buffer," disabled\n"); | 271 | pnp_printf(buffer, " disabled\n"); |
266 | else | 272 | else |
267 | pnp_printf(buffer," 0x%llx-0x%llx\n", | 273 | pnp_printf(buffer, " 0x%llx-0x%llx\n", |
268 | (unsigned long long)pnp_port_start(dev, i), | 274 | (unsigned long long) |
269 | (unsigned long long)pnp_port_end(dev, i)); | 275 | pnp_port_start(dev, i), |
276 | (unsigned long long)pnp_port_end(dev, | ||
277 | i)); | ||
270 | } | 278 | } |
271 | } | 279 | } |
272 | for (i = 0; i < PNP_MAX_MEM; i++) { | 280 | for (i = 0; i < PNP_MAX_MEM; i++) { |
273 | if (pnp_mem_valid(dev, i)) { | 281 | if (pnp_mem_valid(dev, i)) { |
274 | pnp_printf(buffer,"mem"); | 282 | pnp_printf(buffer, "mem"); |
275 | if (pnp_mem_flags(dev, i) & IORESOURCE_DISABLED) | 283 | if (pnp_mem_flags(dev, i) & IORESOURCE_DISABLED) |
276 | pnp_printf(buffer," disabled\n"); | 284 | pnp_printf(buffer, " disabled\n"); |
277 | else | 285 | else |
278 | pnp_printf(buffer," 0x%llx-0x%llx\n", | 286 | pnp_printf(buffer, " 0x%llx-0x%llx\n", |
279 | (unsigned long long)pnp_mem_start(dev, i), | 287 | (unsigned long long) |
280 | (unsigned long long)pnp_mem_end(dev, i)); | 288 | pnp_mem_start(dev, i), |
289 | (unsigned long long)pnp_mem_end(dev, | ||
290 | i)); | ||
281 | } | 291 | } |
282 | } | 292 | } |
283 | for (i = 0; i < PNP_MAX_IRQ; i++) { | 293 | for (i = 0; i < PNP_MAX_IRQ; i++) { |
284 | if (pnp_irq_valid(dev, i)) { | 294 | if (pnp_irq_valid(dev, i)) { |
285 | pnp_printf(buffer,"irq"); | 295 | pnp_printf(buffer, "irq"); |
286 | if (pnp_irq_flags(dev, i) & IORESOURCE_DISABLED) | 296 | if (pnp_irq_flags(dev, i) & IORESOURCE_DISABLED) |
287 | pnp_printf(buffer," disabled\n"); | 297 | pnp_printf(buffer, " disabled\n"); |
288 | else | 298 | else |
289 | pnp_printf(buffer," %lld\n", | 299 | pnp_printf(buffer, " %lld\n", |
290 | (unsigned long long)pnp_irq(dev, i)); | 300 | (unsigned long long)pnp_irq(dev, i)); |
291 | } | 301 | } |
292 | } | 302 | } |
293 | for (i = 0; i < PNP_MAX_DMA; i++) { | 303 | for (i = 0; i < PNP_MAX_DMA; i++) { |
294 | if (pnp_dma_valid(dev, i)) { | 304 | if (pnp_dma_valid(dev, i)) { |
295 | pnp_printf(buffer,"dma"); | 305 | pnp_printf(buffer, "dma"); |
296 | if (pnp_dma_flags(dev, i) & IORESOURCE_DISABLED) | 306 | if (pnp_dma_flags(dev, i) & IORESOURCE_DISABLED) |
297 | pnp_printf(buffer," disabled\n"); | 307 | pnp_printf(buffer, " disabled\n"); |
298 | else | 308 | else |
299 | pnp_printf(buffer," %lld\n", | 309 | pnp_printf(buffer, " %lld\n", |
300 | (unsigned long long)pnp_dma(dev, i)); | 310 | (unsigned long long)pnp_dma(dev, i)); |
301 | } | 311 | } |
302 | } | 312 | } |
303 | ret = (buffer->curr - buf); | 313 | ret = (buffer->curr - buf); |
@@ -308,55 +318,57 @@ static ssize_t pnp_show_current_resources(struct device *dmdev, struct device_at | |||
308 | extern struct semaphore pnp_res_mutex; | 318 | extern struct semaphore pnp_res_mutex; |
309 | 319 | ||
310 | static ssize_t | 320 | static ssize_t |
311 | pnp_set_current_resources(struct device * dmdev, struct device_attribute *attr, const char * ubuf, size_t count) | 321 | pnp_set_current_resources(struct device *dmdev, struct device_attribute *attr, |
322 | const char *ubuf, size_t count) | ||
312 | { | 323 | { |
313 | struct pnp_dev *dev = to_pnp_dev(dmdev); | 324 | struct pnp_dev *dev = to_pnp_dev(dmdev); |
314 | char *buf = (void *)ubuf; | 325 | char *buf = (void *)ubuf; |
315 | int retval = 0; | 326 | int retval = 0; |
316 | 327 | ||
317 | if (dev->status & PNP_ATTACHED) { | 328 | if (dev->status & PNP_ATTACHED) { |
318 | retval = -EBUSY; | 329 | retval = -EBUSY; |
319 | pnp_info("Device %s cannot be configured because it is in use.", dev->dev.bus_id); | 330 | pnp_info("Device %s cannot be configured because it is in use.", |
331 | dev->dev.bus_id); | ||
320 | goto done; | 332 | goto done; |
321 | } | 333 | } |
322 | 334 | ||
323 | while (isspace(*buf)) | 335 | while (isspace(*buf)) |
324 | ++buf; | 336 | ++buf; |
325 | if (!strnicmp(buf,"disable",7)) { | 337 | if (!strnicmp(buf, "disable", 7)) { |
326 | retval = pnp_disable_dev(dev); | 338 | retval = pnp_disable_dev(dev); |
327 | goto done; | 339 | goto done; |
328 | } | 340 | } |
329 | if (!strnicmp(buf,"activate",8)) { | 341 | if (!strnicmp(buf, "activate", 8)) { |
330 | retval = pnp_activate_dev(dev); | 342 | retval = pnp_activate_dev(dev); |
331 | goto done; | 343 | goto done; |
332 | } | 344 | } |
333 | if (!strnicmp(buf,"fill",4)) { | 345 | if (!strnicmp(buf, "fill", 4)) { |
334 | if (dev->active) | 346 | if (dev->active) |
335 | goto done; | 347 | goto done; |
336 | retval = pnp_auto_config_dev(dev); | 348 | retval = pnp_auto_config_dev(dev); |
337 | goto done; | 349 | goto done; |
338 | } | 350 | } |
339 | if (!strnicmp(buf,"auto",4)) { | 351 | if (!strnicmp(buf, "auto", 4)) { |
340 | if (dev->active) | 352 | if (dev->active) |
341 | goto done; | 353 | goto done; |
342 | pnp_init_resource_table(&dev->res); | 354 | pnp_init_resource_table(&dev->res); |
343 | retval = pnp_auto_config_dev(dev); | 355 | retval = pnp_auto_config_dev(dev); |
344 | goto done; | 356 | goto done; |
345 | } | 357 | } |
346 | if (!strnicmp(buf,"clear",5)) { | 358 | if (!strnicmp(buf, "clear", 5)) { |
347 | if (dev->active) | 359 | if (dev->active) |
348 | goto done; | 360 | goto done; |
349 | pnp_init_resource_table(&dev->res); | 361 | pnp_init_resource_table(&dev->res); |
350 | goto done; | 362 | goto done; |
351 | } | 363 | } |
352 | if (!strnicmp(buf,"get",3)) { | 364 | if (!strnicmp(buf, "get", 3)) { |
353 | down(&pnp_res_mutex); | 365 | down(&pnp_res_mutex); |
354 | if (pnp_can_read(dev)) | 366 | if (pnp_can_read(dev)) |
355 | dev->protocol->get(dev, &dev->res); | 367 | dev->protocol->get(dev, &dev->res); |
356 | up(&pnp_res_mutex); | 368 | up(&pnp_res_mutex); |
357 | goto done; | 369 | goto done; |
358 | } | 370 | } |
359 | if (!strnicmp(buf,"set",3)) { | 371 | if (!strnicmp(buf, "set", 3)) { |
360 | int nport = 0, nmem = 0, nirq = 0, ndma = 0; | 372 | int nport = 0, nmem = 0, nirq = 0, ndma = 0; |
361 | if (dev->active) | 373 | if (dev->active) |
362 | goto done; | 374 | goto done; |
@@ -366,65 +378,77 @@ pnp_set_current_resources(struct device * dmdev, struct device_attribute *attr, | |||
366 | while (1) { | 378 | while (1) { |
367 | while (isspace(*buf)) | 379 | while (isspace(*buf)) |
368 | ++buf; | 380 | ++buf; |
369 | if (!strnicmp(buf,"io",2)) { | 381 | if (!strnicmp(buf, "io", 2)) { |
370 | buf += 2; | 382 | buf += 2; |
371 | while (isspace(*buf)) | 383 | while (isspace(*buf)) |
372 | ++buf; | 384 | ++buf; |
373 | dev->res.port_resource[nport].start = simple_strtoul(buf,&buf,0); | 385 | dev->res.port_resource[nport].start = |
386 | simple_strtoul(buf, &buf, 0); | ||
374 | while (isspace(*buf)) | 387 | while (isspace(*buf)) |
375 | ++buf; | 388 | ++buf; |
376 | if(*buf == '-') { | 389 | if (*buf == '-') { |
377 | buf += 1; | 390 | buf += 1; |
378 | while (isspace(*buf)) | 391 | while (isspace(*buf)) |
379 | ++buf; | 392 | ++buf; |
380 | dev->res.port_resource[nport].end = simple_strtoul(buf,&buf,0); | 393 | dev->res.port_resource[nport].end = |
394 | simple_strtoul(buf, &buf, 0); | ||
381 | } else | 395 | } else |
382 | dev->res.port_resource[nport].end = dev->res.port_resource[nport].start; | 396 | dev->res.port_resource[nport].end = |
383 | dev->res.port_resource[nport].flags = IORESOURCE_IO; | 397 | dev->res.port_resource[nport].start; |
398 | dev->res.port_resource[nport].flags = | ||
399 | IORESOURCE_IO; | ||
384 | nport++; | 400 | nport++; |
385 | if (nport >= PNP_MAX_PORT) | 401 | if (nport >= PNP_MAX_PORT) |
386 | break; | 402 | break; |
387 | continue; | 403 | continue; |
388 | } | 404 | } |
389 | if (!strnicmp(buf,"mem",3)) { | 405 | if (!strnicmp(buf, "mem", 3)) { |
390 | buf += 3; | 406 | buf += 3; |
391 | while (isspace(*buf)) | 407 | while (isspace(*buf)) |
392 | ++buf; | 408 | ++buf; |
393 | dev->res.mem_resource[nmem].start = simple_strtoul(buf,&buf,0); | 409 | dev->res.mem_resource[nmem].start = |
410 | simple_strtoul(buf, &buf, 0); | ||
394 | while (isspace(*buf)) | 411 | while (isspace(*buf)) |
395 | ++buf; | 412 | ++buf; |
396 | if(*buf == '-') { | 413 | if (*buf == '-') { |
397 | buf += 1; | 414 | buf += 1; |
398 | while (isspace(*buf)) | 415 | while (isspace(*buf)) |
399 | ++buf; | 416 | ++buf; |
400 | dev->res.mem_resource[nmem].end = simple_strtoul(buf,&buf,0); | 417 | dev->res.mem_resource[nmem].end = |
418 | simple_strtoul(buf, &buf, 0); | ||
401 | } else | 419 | } else |
402 | dev->res.mem_resource[nmem].end = dev->res.mem_resource[nmem].start; | 420 | dev->res.mem_resource[nmem].end = |
403 | dev->res.mem_resource[nmem].flags = IORESOURCE_MEM; | 421 | dev->res.mem_resource[nmem].start; |
422 | dev->res.mem_resource[nmem].flags = | ||
423 | IORESOURCE_MEM; | ||
404 | nmem++; | 424 | nmem++; |
405 | if (nmem >= PNP_MAX_MEM) | 425 | if (nmem >= PNP_MAX_MEM) |
406 | break; | 426 | break; |
407 | continue; | 427 | continue; |
408 | } | 428 | } |
409 | if (!strnicmp(buf,"irq",3)) { | 429 | if (!strnicmp(buf, "irq", 3)) { |
410 | buf += 3; | 430 | buf += 3; |
411 | while (isspace(*buf)) | 431 | while (isspace(*buf)) |
412 | ++buf; | 432 | ++buf; |
413 | dev->res.irq_resource[nirq].start = | 433 | dev->res.irq_resource[nirq].start = |
414 | dev->res.irq_resource[nirq].end = simple_strtoul(buf,&buf,0); | 434 | dev->res.irq_resource[nirq].end = |
415 | dev->res.irq_resource[nirq].flags = IORESOURCE_IRQ; | 435 | simple_strtoul(buf, &buf, 0); |
436 | dev->res.irq_resource[nirq].flags = | ||
437 | IORESOURCE_IRQ; | ||
416 | nirq++; | 438 | nirq++; |
417 | if (nirq >= PNP_MAX_IRQ) | 439 | if (nirq >= PNP_MAX_IRQ) |
418 | break; | 440 | break; |
419 | continue; | 441 | continue; |
420 | } | 442 | } |
421 | if (!strnicmp(buf,"dma",3)) { | 443 | if (!strnicmp(buf, "dma", 3)) { |
422 | buf += 3; | 444 | buf += 3; |
423 | while (isspace(*buf)) | 445 | while (isspace(*buf)) |
424 | ++buf; | 446 | ++buf; |
425 | dev->res.dma_resource[ndma].start = | 447 | dev->res.dma_resource[ndma].start = |
426 | dev->res.dma_resource[ndma].end = simple_strtoul(buf,&buf,0); | 448 | dev->res.dma_resource[ndma].end = |
427 | dev->res.dma_resource[ndma].flags = IORESOURCE_DMA; | 449 | simple_strtoul(buf, &buf, 0); |
450 | dev->res.dma_resource[ndma].flags = | ||
451 | IORESOURCE_DMA; | ||
428 | ndma++; | 452 | ndma++; |
429 | if (ndma >= PNP_MAX_DMA) | 453 | if (ndma >= PNP_MAX_DMA) |
430 | break; | 454 | break; |
@@ -435,45 +459,50 @@ pnp_set_current_resources(struct device * dmdev, struct device_attribute *attr, | |||
435 | up(&pnp_res_mutex); | 459 | up(&pnp_res_mutex); |
436 | goto done; | 460 | goto done; |
437 | } | 461 | } |
438 | done: | 462 | done: |
439 | if (retval < 0) | 463 | if (retval < 0) |
440 | return retval; | 464 | return retval; |
441 | return count; | 465 | return count; |
442 | } | 466 | } |
443 | 467 | ||
444 | static DEVICE_ATTR(resources,S_IRUGO | S_IWUSR, | 468 | static DEVICE_ATTR(resources, S_IRUGO | S_IWUSR, |
445 | pnp_show_current_resources,pnp_set_current_resources); | 469 | pnp_show_current_resources, pnp_set_current_resources); |
446 | 470 | ||
447 | static ssize_t pnp_show_current_ids(struct device *dmdev, struct device_attribute *attr, char *buf) | 471 | static ssize_t pnp_show_current_ids(struct device *dmdev, |
472 | struct device_attribute *attr, char *buf) | ||
448 | { | 473 | { |
449 | char *str = buf; | 474 | char *str = buf; |
450 | struct pnp_dev *dev = to_pnp_dev(dmdev); | 475 | struct pnp_dev *dev = to_pnp_dev(dmdev); |
451 | struct pnp_id * pos = dev->id; | 476 | struct pnp_id *pos = dev->id; |
452 | 477 | ||
453 | while (pos) { | 478 | while (pos) { |
454 | str += sprintf(str,"%s\n", pos->id); | 479 | str += sprintf(str, "%s\n", pos->id); |
455 | pos = pos->next; | 480 | pos = pos->next; |
456 | } | 481 | } |
457 | return (str - buf); | 482 | return (str - buf); |
458 | } | 483 | } |
459 | 484 | ||
460 | static DEVICE_ATTR(id,S_IRUGO,pnp_show_current_ids,NULL); | 485 | static DEVICE_ATTR(id, S_IRUGO, pnp_show_current_ids, NULL); |
461 | 486 | ||
462 | int pnp_interface_attach_device(struct pnp_dev *dev) | 487 | int pnp_interface_attach_device(struct pnp_dev *dev) |
463 | { | 488 | { |
464 | int rc = device_create_file(&dev->dev,&dev_attr_options); | 489 | int rc = device_create_file(&dev->dev, &dev_attr_options); |
465 | if (rc) goto err; | 490 | |
466 | rc = device_create_file(&dev->dev,&dev_attr_resources); | 491 | if (rc) |
467 | if (rc) goto err_opt; | 492 | goto err; |
468 | rc = device_create_file(&dev->dev,&dev_attr_id); | 493 | rc = device_create_file(&dev->dev, &dev_attr_resources); |
469 | if (rc) goto err_res; | 494 | if (rc) |
495 | goto err_opt; | ||
496 | rc = device_create_file(&dev->dev, &dev_attr_id); | ||
497 | if (rc) | ||
498 | goto err_res; | ||
470 | 499 | ||
471 | return 0; | 500 | return 0; |
472 | 501 | ||
473 | err_res: | 502 | err_res: |
474 | device_remove_file(&dev->dev,&dev_attr_resources); | 503 | device_remove_file(&dev->dev, &dev_attr_resources); |
475 | err_opt: | 504 | err_opt: |
476 | device_remove_file(&dev->dev,&dev_attr_options); | 505 | device_remove_file(&dev->dev, &dev_attr_options); |
477 | err: | 506 | err: |
478 | return rc; | 507 | return rc; |
479 | } | 508 | } |
diff --git a/drivers/pnp/isapnp/compat.c b/drivers/pnp/isapnp/compat.c index 0697ab88a9ac..10bdcc4d4f7b 100644 --- a/drivers/pnp/isapnp/compat.c +++ b/drivers/pnp/isapnp/compat.c | |||
@@ -3,34 +3,30 @@ | |||
3 | * the old isapnp APIs. If possible use the new APIs instead. | 3 | * the old isapnp APIs. If possible use the new APIs instead. |
4 | * | 4 | * |
5 | * Copyright 2002 Adam Belay <ambx1@neo.rr.com> | 5 | * Copyright 2002 Adam Belay <ambx1@neo.rr.com> |
6 | * | ||
7 | */ | 6 | */ |
8 | |||
9 | /* TODO: see if more isapnp functions are needed here */ | ||
10 | 7 | ||
11 | #include <linux/module.h> | 8 | #include <linux/module.h> |
12 | #include <linux/isapnp.h> | 9 | #include <linux/isapnp.h> |
13 | #include <linux/string.h> | 10 | #include <linux/string.h> |
14 | 11 | ||
15 | static void pnp_convert_id(char *buf, unsigned short vendor, unsigned short device) | 12 | static void pnp_convert_id(char *buf, unsigned short vendor, |
13 | unsigned short device) | ||
16 | { | 14 | { |
17 | sprintf(buf, "%c%c%c%x%x%x%x", | 15 | sprintf(buf, "%c%c%c%x%x%x%x", |
18 | 'A' + ((vendor >> 2) & 0x3f) - 1, | 16 | 'A' + ((vendor >> 2) & 0x3f) - 1, |
19 | 'A' + (((vendor & 3) << 3) | ((vendor >> 13) & 7)) - 1, | 17 | 'A' + (((vendor & 3) << 3) | ((vendor >> 13) & 7)) - 1, |
20 | 'A' + ((vendor >> 8) & 0x1f) - 1, | 18 | 'A' + ((vendor >> 8) & 0x1f) - 1, |
21 | (device >> 4) & 0x0f, | 19 | (device >> 4) & 0x0f, device & 0x0f, |
22 | device & 0x0f, | 20 | (device >> 12) & 0x0f, (device >> 8) & 0x0f); |
23 | (device >> 12) & 0x0f, | ||
24 | (device >> 8) & 0x0f); | ||
25 | } | 21 | } |
26 | 22 | ||
27 | struct pnp_card *pnp_find_card(unsigned short vendor, | 23 | struct pnp_card *pnp_find_card(unsigned short vendor, unsigned short device, |
28 | unsigned short device, | ||
29 | struct pnp_card *from) | 24 | struct pnp_card *from) |
30 | { | 25 | { |
31 | char id[8]; | 26 | char id[8]; |
32 | char any[8]; | 27 | char any[8]; |
33 | struct list_head *list; | 28 | struct list_head *list; |
29 | |||
34 | pnp_convert_id(id, vendor, device); | 30 | pnp_convert_id(id, vendor, device); |
35 | pnp_convert_id(any, ISAPNP_ANY_ID, ISAPNP_ANY_ID); | 31 | pnp_convert_id(any, ISAPNP_ANY_ID, ISAPNP_ANY_ID); |
36 | 32 | ||
@@ -38,20 +34,20 @@ struct pnp_card *pnp_find_card(unsigned short vendor, | |||
38 | 34 | ||
39 | while (list != &pnp_cards) { | 35 | while (list != &pnp_cards) { |
40 | struct pnp_card *card = global_to_pnp_card(list); | 36 | struct pnp_card *card = global_to_pnp_card(list); |
41 | if (compare_pnp_id(card->id,id) || (memcmp(id,any,7)==0)) | 37 | |
38 | if (compare_pnp_id(card->id, id) || (memcmp(id, any, 7) == 0)) | ||
42 | return card; | 39 | return card; |
43 | list = list->next; | 40 | list = list->next; |
44 | } | 41 | } |
45 | return NULL; | 42 | return NULL; |
46 | } | 43 | } |
47 | 44 | ||
48 | struct pnp_dev *pnp_find_dev(struct pnp_card *card, | 45 | struct pnp_dev *pnp_find_dev(struct pnp_card *card, unsigned short vendor, |
49 | unsigned short vendor, | 46 | unsigned short function, struct pnp_dev *from) |
50 | unsigned short function, | ||
51 | struct pnp_dev *from) | ||
52 | { | 47 | { |
53 | char id[8]; | 48 | char id[8]; |
54 | char any[8]; | 49 | char any[8]; |
50 | |||
55 | pnp_convert_id(id, vendor, function); | 51 | pnp_convert_id(id, vendor, function); |
56 | pnp_convert_id(any, ISAPNP_ANY_ID, ISAPNP_ANY_ID); | 52 | pnp_convert_id(any, ISAPNP_ANY_ID, ISAPNP_ANY_ID); |
57 | if (card == NULL) { /* look for a logical device from all cards */ | 53 | if (card == NULL) { /* look for a logical device from all cards */ |
@@ -63,7 +59,9 @@ struct pnp_dev *pnp_find_dev(struct pnp_card *card, | |||
63 | 59 | ||
64 | while (list != &pnp_global) { | 60 | while (list != &pnp_global) { |
65 | struct pnp_dev *dev = global_to_pnp_dev(list); | 61 | struct pnp_dev *dev = global_to_pnp_dev(list); |
66 | if (compare_pnp_id(dev->id,id) || (memcmp(id,any,7)==0)) | 62 | |
63 | if (compare_pnp_id(dev->id, id) || | ||
64 | (memcmp(id, any, 7) == 0)) | ||
67 | return dev; | 65 | return dev; |
68 | list = list->next; | 66 | list = list->next; |
69 | } | 67 | } |
@@ -78,7 +76,8 @@ struct pnp_dev *pnp_find_dev(struct pnp_card *card, | |||
78 | } | 76 | } |
79 | while (list != &card->devices) { | 77 | while (list != &card->devices) { |
80 | struct pnp_dev *dev = card_to_pnp_dev(list); | 78 | struct pnp_dev *dev = card_to_pnp_dev(list); |
81 | if (compare_pnp_id(dev->id,id)) | 79 | |
80 | if (compare_pnp_id(dev->id, id)) | ||
82 | return dev; | 81 | return dev; |
83 | list = list->next; | 82 | list = list->next; |
84 | } | 83 | } |
diff --git a/drivers/pnp/isapnp/core.c b/drivers/pnp/isapnp/core.c index 914d00c423ad..b4e2aa995b53 100644 --- a/drivers/pnp/isapnp/core.c +++ b/drivers/pnp/isapnp/core.c | |||
@@ -51,10 +51,10 @@ | |||
51 | #define ISAPNP_DEBUG | 51 | #define ISAPNP_DEBUG |
52 | #endif | 52 | #endif |
53 | 53 | ||
54 | int isapnp_disable; /* Disable ISA PnP */ | 54 | int isapnp_disable; /* Disable ISA PnP */ |
55 | static int isapnp_rdp; /* Read Data Port */ | 55 | static int isapnp_rdp; /* Read Data Port */ |
56 | static int isapnp_reset = 1; /* reset all PnP cards (deactivate) */ | 56 | static int isapnp_reset = 1; /* reset all PnP cards (deactivate) */ |
57 | static int isapnp_verbose = 1; /* verbose mode */ | 57 | static int isapnp_verbose = 1; /* verbose mode */ |
58 | 58 | ||
59 | MODULE_AUTHOR("Jaroslav Kysela <perex@suse.cz>"); | 59 | MODULE_AUTHOR("Jaroslav Kysela <perex@suse.cz>"); |
60 | MODULE_DESCRIPTION("Generic ISA Plug & Play support"); | 60 | MODULE_DESCRIPTION("Generic ISA Plug & Play support"); |
@@ -126,7 +126,7 @@ static unsigned short isapnp_read_word(unsigned char idx) | |||
126 | unsigned short val; | 126 | unsigned short val; |
127 | 127 | ||
128 | val = isapnp_read_byte(idx); | 128 | val = isapnp_read_byte(idx); |
129 | val = (val << 8) + isapnp_read_byte(idx+1); | 129 | val = (val << 8) + isapnp_read_byte(idx + 1); |
130 | return val; | 130 | return val; |
131 | } | 131 | } |
132 | 132 | ||
@@ -139,7 +139,7 @@ void isapnp_write_byte(unsigned char idx, unsigned char val) | |||
139 | static void isapnp_write_word(unsigned char idx, unsigned short val) | 139 | static void isapnp_write_word(unsigned char idx, unsigned short val) |
140 | { | 140 | { |
141 | isapnp_write_byte(idx, val >> 8); | 141 | isapnp_write_byte(idx, val >> 8); |
142 | isapnp_write_byte(idx+1, val); | 142 | isapnp_write_byte(idx + 1, val); |
143 | } | 143 | } |
144 | 144 | ||
145 | static void isapnp_key(void) | 145 | static void isapnp_key(void) |
@@ -193,7 +193,7 @@ static void isapnp_deactivate(unsigned char logdev) | |||
193 | static void __init isapnp_peek(unsigned char *data, int bytes) | 193 | static void __init isapnp_peek(unsigned char *data, int bytes) |
194 | { | 194 | { |
195 | int i, j; | 195 | int i, j; |
196 | unsigned char d=0; | 196 | unsigned char d = 0; |
197 | 197 | ||
198 | for (i = 1; i <= bytes; i++) { | 198 | for (i = 1; i <= bytes; i++) { |
199 | for (j = 0; j < 20; j++) { | 199 | for (j = 0; j < 20; j++) { |
@@ -220,19 +220,18 @@ static int isapnp_next_rdp(void) | |||
220 | { | 220 | { |
221 | int rdp = isapnp_rdp; | 221 | int rdp = isapnp_rdp; |
222 | static int old_rdp = 0; | 222 | static int old_rdp = 0; |
223 | 223 | ||
224 | if(old_rdp) | 224 | if (old_rdp) { |
225 | { | ||
226 | release_region(old_rdp, 1); | 225 | release_region(old_rdp, 1); |
227 | old_rdp = 0; | 226 | old_rdp = 0; |
228 | } | 227 | } |
229 | while (rdp <= 0x3ff) { | 228 | while (rdp <= 0x3ff) { |
230 | /* | 229 | /* |
231 | * We cannot use NE2000 probe spaces for ISAPnP or we | 230 | * We cannot use NE2000 probe spaces for ISAPnP or we |
232 | * will lock up machines. | 231 | * will lock up machines. |
233 | */ | 232 | */ |
234 | if ((rdp < 0x280 || rdp > 0x380) && request_region(rdp, 1, "ISAPnP")) | 233 | if ((rdp < 0x280 || rdp > 0x380) |
235 | { | 234 | && request_region(rdp, 1, "ISAPnP")) { |
236 | isapnp_rdp = rdp; | 235 | isapnp_rdp = rdp; |
237 | old_rdp = rdp; | 236 | old_rdp = rdp; |
238 | return 0; | 237 | return 0; |
@@ -253,7 +252,6 @@ static inline void isapnp_set_rdp(void) | |||
253 | * Perform an isolation. The port selection code now tries to avoid | 252 | * Perform an isolation. The port selection code now tries to avoid |
254 | * "dangerous to read" ports. | 253 | * "dangerous to read" ports. |
255 | */ | 254 | */ |
256 | |||
257 | static int __init isapnp_isolate_rdp_select(void) | 255 | static int __init isapnp_isolate_rdp_select(void) |
258 | { | 256 | { |
259 | isapnp_wait(); | 257 | isapnp_wait(); |
@@ -282,7 +280,6 @@ static int __init isapnp_isolate_rdp_select(void) | |||
282 | /* | 280 | /* |
283 | * Isolate (assign uniqued CSN) to all ISA PnP devices. | 281 | * Isolate (assign uniqued CSN) to all ISA PnP devices. |
284 | */ | 282 | */ |
285 | |||
286 | static int __init isapnp_isolate(void) | 283 | static int __init isapnp_isolate(void) |
287 | { | 284 | { |
288 | unsigned char checksum = 0x6a; | 285 | unsigned char checksum = 0x6a; |
@@ -305,7 +302,9 @@ static int __init isapnp_isolate(void) | |||
305 | udelay(250); | 302 | udelay(250); |
306 | if (data == 0x55aa) | 303 | if (data == 0x55aa) |
307 | bit = 0x01; | 304 | bit = 0x01; |
308 | checksum = ((((checksum ^ (checksum >> 1)) & 0x01) ^ bit) << 7) | (checksum >> 1); | 305 | checksum = |
306 | ((((checksum ^ (checksum >> 1)) & 0x01) ^ bit) << 7) | ||
307 | | (checksum >> 1); | ||
309 | bit = 0x00; | 308 | bit = 0x00; |
310 | } | 309 | } |
311 | for (i = 65; i <= 72; i++) { | 310 | for (i = 65; i <= 72; i++) { |
@@ -351,13 +350,12 @@ static int __init isapnp_isolate(void) | |||
351 | /* | 350 | /* |
352 | * Read one tag from stream. | 351 | * Read one tag from stream. |
353 | */ | 352 | */ |
354 | |||
355 | static int __init isapnp_read_tag(unsigned char *type, unsigned short *size) | 353 | static int __init isapnp_read_tag(unsigned char *type, unsigned short *size) |
356 | { | 354 | { |
357 | unsigned char tag, tmp[2]; | 355 | unsigned char tag, tmp[2]; |
358 | 356 | ||
359 | isapnp_peek(&tag, 1); | 357 | isapnp_peek(&tag, 1); |
360 | if (tag == 0) /* invalid tag */ | 358 | if (tag == 0) /* invalid tag */ |
361 | return -1; | 359 | return -1; |
362 | if (tag & 0x80) { /* large item */ | 360 | if (tag & 0x80) { /* large item */ |
363 | *type = tag; | 361 | *type = tag; |
@@ -368,7 +366,8 @@ static int __init isapnp_read_tag(unsigned char *type, unsigned short *size) | |||
368 | *size = tag & 0x07; | 366 | *size = tag & 0x07; |
369 | } | 367 | } |
370 | #if 0 | 368 | #if 0 |
371 | printk(KERN_DEBUG "tag = 0x%x, type = 0x%x, size = %i\n", tag, *type, *size); | 369 | printk(KERN_DEBUG "tag = 0x%x, type = 0x%x, size = %i\n", tag, *type, |
370 | *size); | ||
372 | #endif | 371 | #endif |
373 | if (*type == 0xff && *size == 0xffff) /* probably invalid data */ | 372 | if (*type == 0xff && *size == 0xffff) /* probably invalid data */ |
374 | return -1; | 373 | return -1; |
@@ -378,7 +377,6 @@ static int __init isapnp_read_tag(unsigned char *type, unsigned short *size) | |||
378 | /* | 377 | /* |
379 | * Skip specified number of bytes from stream. | 378 | * Skip specified number of bytes from stream. |
380 | */ | 379 | */ |
381 | |||
382 | static void __init isapnp_skip_bytes(int count) | 380 | static void __init isapnp_skip_bytes(int count) |
383 | { | 381 | { |
384 | isapnp_peek(NULL, count); | 382 | isapnp_peek(NULL, count); |
@@ -387,31 +385,30 @@ static void __init isapnp_skip_bytes(int count) | |||
387 | /* | 385 | /* |
388 | * Parse EISA id. | 386 | * Parse EISA id. |
389 | */ | 387 | */ |
390 | 388 | static void isapnp_parse_id(struct pnp_dev *dev, unsigned short vendor, | |
391 | static void isapnp_parse_id(struct pnp_dev * dev, unsigned short vendor, unsigned short device) | 389 | unsigned short device) |
392 | { | 390 | { |
393 | struct pnp_id * id; | 391 | struct pnp_id *id; |
392 | |||
394 | if (!dev) | 393 | if (!dev) |
395 | return; | 394 | return; |
396 | id = kzalloc(sizeof(struct pnp_id), GFP_KERNEL); | 395 | id = kzalloc(sizeof(struct pnp_id), GFP_KERNEL); |
397 | if (!id) | 396 | if (!id) |
398 | return; | 397 | return; |
399 | sprintf(id->id, "%c%c%c%x%x%x%x", | 398 | sprintf(id->id, "%c%c%c%x%x%x%x", |
400 | 'A' + ((vendor >> 2) & 0x3f) - 1, | 399 | 'A' + ((vendor >> 2) & 0x3f) - 1, |
401 | 'A' + (((vendor & 3) << 3) | ((vendor >> 13) & 7)) - 1, | 400 | 'A' + (((vendor & 3) << 3) | ((vendor >> 13) & 7)) - 1, |
402 | 'A' + ((vendor >> 8) & 0x1f) - 1, | 401 | 'A' + ((vendor >> 8) & 0x1f) - 1, |
403 | (device >> 4) & 0x0f, | 402 | (device >> 4) & 0x0f, |
404 | device & 0x0f, | 403 | device & 0x0f, (device >> 12) & 0x0f, (device >> 8) & 0x0f); |
405 | (device >> 12) & 0x0f, | ||
406 | (device >> 8) & 0x0f); | ||
407 | pnp_add_id(id, dev); | 404 | pnp_add_id(id, dev); |
408 | } | 405 | } |
409 | 406 | ||
410 | /* | 407 | /* |
411 | * Parse logical device tag. | 408 | * Parse logical device tag. |
412 | */ | 409 | */ |
413 | 410 | static struct pnp_dev *__init isapnp_parse_device(struct pnp_card *card, | |
414 | static struct pnp_dev * __init isapnp_parse_device(struct pnp_card *card, int size, int number) | 411 | int size, int number) |
415 | { | 412 | { |
416 | unsigned char tmp[6]; | 413 | unsigned char tmp[6]; |
417 | struct pnp_dev *dev; | 414 | struct pnp_dev *dev; |
@@ -435,13 +432,11 @@ static struct pnp_dev * __init isapnp_parse_device(struct pnp_card *card, int si | |||
435 | return dev; | 432 | return dev; |
436 | } | 433 | } |
437 | 434 | ||
438 | |||
439 | /* | 435 | /* |
440 | * Add IRQ resource to resources list. | 436 | * Add IRQ resource to resources list. |
441 | */ | 437 | */ |
442 | |||
443 | static void __init isapnp_parse_irq_resource(struct pnp_option *option, | 438 | static void __init isapnp_parse_irq_resource(struct pnp_option *option, |
444 | int size) | 439 | int size) |
445 | { | 440 | { |
446 | unsigned char tmp[3]; | 441 | unsigned char tmp[3]; |
447 | struct pnp_irq *irq; | 442 | struct pnp_irq *irq; |
@@ -458,15 +453,13 @@ static void __init isapnp_parse_irq_resource(struct pnp_option *option, | |||
458 | else | 453 | else |
459 | irq->flags = IORESOURCE_IRQ_HIGHEDGE; | 454 | irq->flags = IORESOURCE_IRQ_HIGHEDGE; |
460 | pnp_register_irq_resource(option, irq); | 455 | pnp_register_irq_resource(option, irq); |
461 | return; | ||
462 | } | 456 | } |
463 | 457 | ||
464 | /* | 458 | /* |
465 | * Add DMA resource to resources list. | 459 | * Add DMA resource to resources list. |
466 | */ | 460 | */ |
467 | |||
468 | static void __init isapnp_parse_dma_resource(struct pnp_option *option, | 461 | static void __init isapnp_parse_dma_resource(struct pnp_option *option, |
469 | int size) | 462 | int size) |
470 | { | 463 | { |
471 | unsigned char tmp[2]; | 464 | unsigned char tmp[2]; |
472 | struct pnp_dma *dma; | 465 | struct pnp_dma *dma; |
@@ -478,15 +471,13 @@ static void __init isapnp_parse_dma_resource(struct pnp_option *option, | |||
478 | dma->map = tmp[0]; | 471 | dma->map = tmp[0]; |
479 | dma->flags = tmp[1]; | 472 | dma->flags = tmp[1]; |
480 | pnp_register_dma_resource(option, dma); | 473 | pnp_register_dma_resource(option, dma); |
481 | return; | ||
482 | } | 474 | } |
483 | 475 | ||
484 | /* | 476 | /* |
485 | * Add port resource to resources list. | 477 | * Add port resource to resources list. |
486 | */ | 478 | */ |
487 | |||
488 | static void __init isapnp_parse_port_resource(struct pnp_option *option, | 479 | static void __init isapnp_parse_port_resource(struct pnp_option *option, |
489 | int size) | 480 | int size) |
490 | { | 481 | { |
491 | unsigned char tmp[7]; | 482 | unsigned char tmp[7]; |
492 | struct pnp_port *port; | 483 | struct pnp_port *port; |
@@ -500,16 +491,14 @@ static void __init isapnp_parse_port_resource(struct pnp_option *option, | |||
500 | port->align = tmp[5]; | 491 | port->align = tmp[5]; |
501 | port->size = tmp[6]; | 492 | port->size = tmp[6]; |
502 | port->flags = tmp[0] ? PNP_PORT_FLAG_16BITADDR : 0; | 493 | port->flags = tmp[0] ? PNP_PORT_FLAG_16BITADDR : 0; |
503 | pnp_register_port_resource(option,port); | 494 | pnp_register_port_resource(option, port); |
504 | return; | ||
505 | } | 495 | } |
506 | 496 | ||
507 | /* | 497 | /* |
508 | * Add fixed port resource to resources list. | 498 | * Add fixed port resource to resources list. |
509 | */ | 499 | */ |
510 | |||
511 | static void __init isapnp_parse_fixed_port_resource(struct pnp_option *option, | 500 | static void __init isapnp_parse_fixed_port_resource(struct pnp_option *option, |
512 | int size) | 501 | int size) |
513 | { | 502 | { |
514 | unsigned char tmp[3]; | 503 | unsigned char tmp[3]; |
515 | struct pnp_port *port; | 504 | struct pnp_port *port; |
@@ -522,16 +511,14 @@ static void __init isapnp_parse_fixed_port_resource(struct pnp_option *option, | |||
522 | port->size = tmp[2]; | 511 | port->size = tmp[2]; |
523 | port->align = 0; | 512 | port->align = 0; |
524 | port->flags = PNP_PORT_FLAG_FIXED; | 513 | port->flags = PNP_PORT_FLAG_FIXED; |
525 | pnp_register_port_resource(option,port); | 514 | pnp_register_port_resource(option, port); |
526 | return; | ||
527 | } | 515 | } |
528 | 516 | ||
529 | /* | 517 | /* |
530 | * Add memory resource to resources list. | 518 | * Add memory resource to resources list. |
531 | */ | 519 | */ |
532 | |||
533 | static void __init isapnp_parse_mem_resource(struct pnp_option *option, | 520 | static void __init isapnp_parse_mem_resource(struct pnp_option *option, |
534 | int size) | 521 | int size) |
535 | { | 522 | { |
536 | unsigned char tmp[9]; | 523 | unsigned char tmp[9]; |
537 | struct pnp_mem *mem; | 524 | struct pnp_mem *mem; |
@@ -545,16 +532,14 @@ static void __init isapnp_parse_mem_resource(struct pnp_option *option, | |||
545 | mem->align = (tmp[6] << 8) | tmp[5]; | 532 | mem->align = (tmp[6] << 8) | tmp[5]; |
546 | mem->size = ((tmp[8] << 8) | tmp[7]) << 8; | 533 | mem->size = ((tmp[8] << 8) | tmp[7]) << 8; |
547 | mem->flags = tmp[0]; | 534 | mem->flags = tmp[0]; |
548 | pnp_register_mem_resource(option,mem); | 535 | pnp_register_mem_resource(option, mem); |
549 | return; | ||
550 | } | 536 | } |
551 | 537 | ||
552 | /* | 538 | /* |
553 | * Add 32-bit memory resource to resources list. | 539 | * Add 32-bit memory resource to resources list. |
554 | */ | 540 | */ |
555 | |||
556 | static void __init isapnp_parse_mem32_resource(struct pnp_option *option, | 541 | static void __init isapnp_parse_mem32_resource(struct pnp_option *option, |
557 | int size) | 542 | int size) |
558 | { | 543 | { |
559 | unsigned char tmp[17]; | 544 | unsigned char tmp[17]; |
560 | struct pnp_mem *mem; | 545 | struct pnp_mem *mem; |
@@ -565,18 +550,19 @@ static void __init isapnp_parse_mem32_resource(struct pnp_option *option, | |||
565 | return; | 550 | return; |
566 | mem->min = (tmp[4] << 24) | (tmp[3] << 16) | (tmp[2] << 8) | tmp[1]; | 551 | mem->min = (tmp[4] << 24) | (tmp[3] << 16) | (tmp[2] << 8) | tmp[1]; |
567 | mem->max = (tmp[8] << 24) | (tmp[7] << 16) | (tmp[6] << 8) | tmp[5]; | 552 | mem->max = (tmp[8] << 24) | (tmp[7] << 16) | (tmp[6] << 8) | tmp[5]; |
568 | mem->align = (tmp[12] << 24) | (tmp[11] << 16) | (tmp[10] << 8) | tmp[9]; | 553 | mem->align = |
569 | mem->size = (tmp[16] << 24) | (tmp[15] << 16) | (tmp[14] << 8) | tmp[13]; | 554 | (tmp[12] << 24) | (tmp[11] << 16) | (tmp[10] << 8) | tmp[9]; |
555 | mem->size = | ||
556 | (tmp[16] << 24) | (tmp[15] << 16) | (tmp[14] << 8) | tmp[13]; | ||
570 | mem->flags = tmp[0]; | 557 | mem->flags = tmp[0]; |
571 | pnp_register_mem_resource(option,mem); | 558 | pnp_register_mem_resource(option, mem); |
572 | } | 559 | } |
573 | 560 | ||
574 | /* | 561 | /* |
575 | * Add 32-bit fixed memory resource to resources list. | 562 | * Add 32-bit fixed memory resource to resources list. |
576 | */ | 563 | */ |
577 | |||
578 | static void __init isapnp_parse_fixed_mem32_resource(struct pnp_option *option, | 564 | static void __init isapnp_parse_fixed_mem32_resource(struct pnp_option *option, |
579 | int size) | 565 | int size) |
580 | { | 566 | { |
581 | unsigned char tmp[9]; | 567 | unsigned char tmp[9]; |
582 | struct pnp_mem *mem; | 568 | struct pnp_mem *mem; |
@@ -585,28 +571,29 @@ static void __init isapnp_parse_fixed_mem32_resource(struct pnp_option *option, | |||
585 | mem = kzalloc(sizeof(struct pnp_mem), GFP_KERNEL); | 571 | mem = kzalloc(sizeof(struct pnp_mem), GFP_KERNEL); |
586 | if (!mem) | 572 | if (!mem) |
587 | return; | 573 | return; |
588 | mem->min = mem->max = (tmp[4] << 24) | (tmp[3] << 16) | (tmp[2] << 8) | tmp[1]; | 574 | mem->min = mem->max = |
575 | (tmp[4] << 24) | (tmp[3] << 16) | (tmp[2] << 8) | tmp[1]; | ||
589 | mem->size = (tmp[8] << 24) | (tmp[7] << 16) | (tmp[6] << 8) | tmp[5]; | 576 | mem->size = (tmp[8] << 24) | (tmp[7] << 16) | (tmp[6] << 8) | tmp[5]; |
590 | mem->align = 0; | 577 | mem->align = 0; |
591 | mem->flags = tmp[0]; | 578 | mem->flags = tmp[0]; |
592 | pnp_register_mem_resource(option,mem); | 579 | pnp_register_mem_resource(option, mem); |
593 | } | 580 | } |
594 | 581 | ||
595 | /* | 582 | /* |
596 | * Parse card name for ISA PnP device. | 583 | * Parse card name for ISA PnP device. |
597 | */ | 584 | */ |
598 | |||
599 | static void __init | 585 | static void __init |
600 | isapnp_parse_name(char *name, unsigned int name_max, unsigned short *size) | 586 | isapnp_parse_name(char *name, unsigned int name_max, unsigned short *size) |
601 | { | 587 | { |
602 | if (name[0] == '\0') { | 588 | if (name[0] == '\0') { |
603 | unsigned short size1 = *size >= name_max ? (name_max - 1) : *size; | 589 | unsigned short size1 = |
590 | *size >= name_max ? (name_max - 1) : *size; | ||
604 | isapnp_peek(name, size1); | 591 | isapnp_peek(name, size1); |
605 | name[size1] = '\0'; | 592 | name[size1] = '\0'; |
606 | *size -= size1; | 593 | *size -= size1; |
607 | 594 | ||
608 | /* clean whitespace from end of string */ | 595 | /* clean whitespace from end of string */ |
609 | while (size1 > 0 && name[--size1] == ' ') | 596 | while (size1 > 0 && name[--size1] == ' ') |
610 | name[size1] = '\0'; | 597 | name[size1] = '\0'; |
611 | } | 598 | } |
612 | } | 599 | } |
@@ -614,7 +601,6 @@ isapnp_parse_name(char *name, unsigned int name_max, unsigned short *size) | |||
614 | /* | 601 | /* |
615 | * Parse resource map for logical device. | 602 | * Parse resource map for logical device. |
616 | */ | 603 | */ |
617 | |||
618 | static int __init isapnp_create_device(struct pnp_card *card, | 604 | static int __init isapnp_create_device(struct pnp_card *card, |
619 | unsigned short size) | 605 | unsigned short size) |
620 | { | 606 | { |
@@ -622,6 +608,7 @@ static int __init isapnp_create_device(struct pnp_card *card, | |||
622 | unsigned char type, tmp[17]; | 608 | unsigned char type, tmp[17]; |
623 | struct pnp_option *option; | 609 | struct pnp_option *option; |
624 | struct pnp_dev *dev; | 610 | struct pnp_dev *dev; |
611 | |||
625 | if ((dev = isapnp_parse_device(card, size, number++)) == NULL) | 612 | if ((dev = isapnp_parse_device(card, size, number++)) == NULL) |
626 | return 1; | 613 | return 1; |
627 | option = pnp_register_independent_option(dev); | 614 | option = pnp_register_independent_option(dev); |
@@ -629,17 +616,19 @@ static int __init isapnp_create_device(struct pnp_card *card, | |||
629 | kfree(dev); | 616 | kfree(dev); |
630 | return 1; | 617 | return 1; |
631 | } | 618 | } |
632 | pnp_add_card_device(card,dev); | 619 | pnp_add_card_device(card, dev); |
633 | 620 | ||
634 | while (1) { | 621 | while (1) { |
635 | if (isapnp_read_tag(&type, &size)<0) | 622 | if (isapnp_read_tag(&type, &size) < 0) |
636 | return 1; | 623 | return 1; |
637 | if (skip && type != _STAG_LOGDEVID && type != _STAG_END) | 624 | if (skip && type != _STAG_LOGDEVID && type != _STAG_END) |
638 | goto __skip; | 625 | goto __skip; |
639 | switch (type) { | 626 | switch (type) { |
640 | case _STAG_LOGDEVID: | 627 | case _STAG_LOGDEVID: |
641 | if (size >= 5 && size <= 6) { | 628 | if (size >= 5 && size <= 6) { |
642 | if ((dev = isapnp_parse_device(card, size, number++)) == NULL) | 629 | if ((dev = |
630 | isapnp_parse_device(card, size, | ||
631 | number++)) == NULL) | ||
643 | return 1; | 632 | return 1; |
644 | size = 0; | 633 | size = 0; |
645 | skip = 0; | 634 | skip = 0; |
@@ -648,7 +637,7 @@ static int __init isapnp_create_device(struct pnp_card *card, | |||
648 | kfree(dev); | 637 | kfree(dev); |
649 | return 1; | 638 | return 1; |
650 | } | 639 | } |
651 | pnp_add_card_device(card,dev); | 640 | pnp_add_card_device(card, dev); |
652 | } else { | 641 | } else { |
653 | skip = 1; | 642 | skip = 1; |
654 | } | 643 | } |
@@ -658,7 +647,8 @@ static int __init isapnp_create_device(struct pnp_card *card, | |||
658 | case _STAG_COMPATDEVID: | 647 | case _STAG_COMPATDEVID: |
659 | if (size == 4 && compat < DEVICE_COUNT_COMPATIBLE) { | 648 | if (size == 4 && compat < DEVICE_COUNT_COMPATIBLE) { |
660 | isapnp_peek(tmp, 4); | 649 | isapnp_peek(tmp, 4); |
661 | isapnp_parse_id(dev,(tmp[1] << 8) | tmp[0], (tmp[3] << 8) | tmp[2]); | 650 | isapnp_parse_id(dev, (tmp[1] << 8) | tmp[0], |
651 | (tmp[3] << 8) | tmp[2]); | ||
662 | compat++; | 652 | compat++; |
663 | size = 0; | 653 | size = 0; |
664 | } | 654 | } |
@@ -684,7 +674,7 @@ static int __init isapnp_create_device(struct pnp_card *card, | |||
684 | priority = 0x100 | tmp[0]; | 674 | priority = 0x100 | tmp[0]; |
685 | size = 0; | 675 | size = 0; |
686 | } | 676 | } |
687 | option = pnp_register_dependent_option(dev,priority); | 677 | option = pnp_register_dependent_option(dev, priority); |
688 | if (!option) | 678 | if (!option) |
689 | return 1; | 679 | return 1; |
690 | break; | 680 | break; |
@@ -739,11 +729,13 @@ static int __init isapnp_create_device(struct pnp_card *card, | |||
739 | isapnp_skip_bytes(size); | 729 | isapnp_skip_bytes(size); |
740 | return 1; | 730 | return 1; |
741 | default: | 731 | default: |
742 | printk(KERN_ERR "isapnp: unexpected or unknown tag type 0x%x for logical device %i (device %i), ignored\n", type, dev->number, card->number); | 732 | printk(KERN_ERR |
733 | "isapnp: unexpected or unknown tag type 0x%x for logical device %i (device %i), ignored\n", | ||
734 | type, dev->number, card->number); | ||
743 | } | 735 | } |
744 | __skip: | 736 | __skip: |
745 | if (size > 0) | 737 | if (size > 0) |
746 | isapnp_skip_bytes(size); | 738 | isapnp_skip_bytes(size); |
747 | } | 739 | } |
748 | return 0; | 740 | return 0; |
749 | } | 741 | } |
@@ -751,14 +743,13 @@ static int __init isapnp_create_device(struct pnp_card *card, | |||
751 | /* | 743 | /* |
752 | * Parse resource map for ISA PnP card. | 744 | * Parse resource map for ISA PnP card. |
753 | */ | 745 | */ |
754 | |||
755 | static void __init isapnp_parse_resource_map(struct pnp_card *card) | 746 | static void __init isapnp_parse_resource_map(struct pnp_card *card) |
756 | { | 747 | { |
757 | unsigned char type, tmp[17]; | 748 | unsigned char type, tmp[17]; |
758 | unsigned short size; | 749 | unsigned short size; |
759 | 750 | ||
760 | while (1) { | 751 | while (1) { |
761 | if (isapnp_read_tag(&type, &size)<0) | 752 | if (isapnp_read_tag(&type, &size) < 0) |
762 | return; | 753 | return; |
763 | switch (type) { | 754 | switch (type) { |
764 | case _STAG_PNPVERNO: | 755 | case _STAG_PNPVERNO: |
@@ -771,7 +762,7 @@ static void __init isapnp_parse_resource_map(struct pnp_card *card) | |||
771 | break; | 762 | break; |
772 | case _STAG_LOGDEVID: | 763 | case _STAG_LOGDEVID: |
773 | if (size >= 5 && size <= 6) { | 764 | if (size >= 5 && size <= 6) { |
774 | if (isapnp_create_device(card, size)==1) | 765 | if (isapnp_create_device(card, size) == 1) |
775 | return; | 766 | return; |
776 | size = 0; | 767 | size = 0; |
777 | } | 768 | } |
@@ -779,7 +770,8 @@ static void __init isapnp_parse_resource_map(struct pnp_card *card) | |||
779 | case _STAG_VENDOR: | 770 | case _STAG_VENDOR: |
780 | break; | 771 | break; |
781 | case _LTAG_ANSISTR: | 772 | case _LTAG_ANSISTR: |
782 | isapnp_parse_name(card->name, sizeof(card->name), &size); | 773 | isapnp_parse_name(card->name, sizeof(card->name), |
774 | &size); | ||
783 | break; | 775 | break; |
784 | case _LTAG_UNICODESTR: | 776 | case _LTAG_UNICODESTR: |
785 | /* silently ignore */ | 777 | /* silently ignore */ |
@@ -792,18 +784,19 @@ static void __init isapnp_parse_resource_map(struct pnp_card *card) | |||
792 | isapnp_skip_bytes(size); | 784 | isapnp_skip_bytes(size); |
793 | return; | 785 | return; |
794 | default: | 786 | default: |
795 | printk(KERN_ERR "isapnp: unexpected or unknown tag type 0x%x for device %i, ignored\n", type, card->number); | 787 | printk(KERN_ERR |
788 | "isapnp: unexpected or unknown tag type 0x%x for device %i, ignored\n", | ||
789 | type, card->number); | ||
796 | } | 790 | } |
797 | __skip: | 791 | __skip: |
798 | if (size > 0) | 792 | if (size > 0) |
799 | isapnp_skip_bytes(size); | 793 | isapnp_skip_bytes(size); |
800 | } | 794 | } |
801 | } | 795 | } |
802 | 796 | ||
803 | /* | 797 | /* |
804 | * Compute ISA PnP checksum for first eight bytes. | 798 | * Compute ISA PnP checksum for first eight bytes. |
805 | */ | 799 | */ |
806 | |||
807 | static unsigned char __init isapnp_checksum(unsigned char *data) | 800 | static unsigned char __init isapnp_checksum(unsigned char *data) |
808 | { | 801 | { |
809 | int i, j; | 802 | int i, j; |
@@ -815,7 +808,9 @@ static unsigned char __init isapnp_checksum(unsigned char *data) | |||
815 | bit = 0; | 808 | bit = 0; |
816 | if (b & (1 << j)) | 809 | if (b & (1 << j)) |
817 | bit = 1; | 810 | bit = 1; |
818 | checksum = ((((checksum ^ (checksum >> 1)) & 0x01) ^ bit) << 7) | (checksum >> 1); | 811 | checksum = |
812 | ((((checksum ^ (checksum >> 1)) & 0x01) ^ bit) << 7) | ||
813 | | (checksum >> 1); | ||
819 | } | 814 | } |
820 | } | 815 | } |
821 | return checksum; | 816 | return checksum; |
@@ -824,27 +819,25 @@ static unsigned char __init isapnp_checksum(unsigned char *data) | |||
824 | /* | 819 | /* |
825 | * Parse EISA id for ISA PnP card. | 820 | * Parse EISA id for ISA PnP card. |
826 | */ | 821 | */ |
827 | 822 | static void isapnp_parse_card_id(struct pnp_card *card, unsigned short vendor, | |
828 | static void isapnp_parse_card_id(struct pnp_card * card, unsigned short vendor, unsigned short device) | 823 | unsigned short device) |
829 | { | 824 | { |
830 | struct pnp_id * id = kzalloc(sizeof(struct pnp_id), GFP_KERNEL); | 825 | struct pnp_id *id = kzalloc(sizeof(struct pnp_id), GFP_KERNEL); |
826 | |||
831 | if (!id) | 827 | if (!id) |
832 | return; | 828 | return; |
833 | sprintf(id->id, "%c%c%c%x%x%x%x", | 829 | sprintf(id->id, "%c%c%c%x%x%x%x", |
834 | 'A' + ((vendor >> 2) & 0x3f) - 1, | 830 | 'A' + ((vendor >> 2) & 0x3f) - 1, |
835 | 'A' + (((vendor & 3) << 3) | ((vendor >> 13) & 7)) - 1, | 831 | 'A' + (((vendor & 3) << 3) | ((vendor >> 13) & 7)) - 1, |
836 | 'A' + ((vendor >> 8) & 0x1f) - 1, | 832 | 'A' + ((vendor >> 8) & 0x1f) - 1, |
837 | (device >> 4) & 0x0f, | 833 | (device >> 4) & 0x0f, |
838 | device & 0x0f, | 834 | device & 0x0f, (device >> 12) & 0x0f, (device >> 8) & 0x0f); |
839 | (device >> 12) & 0x0f, | 835 | pnp_add_card_id(id, card); |
840 | (device >> 8) & 0x0f); | ||
841 | pnp_add_card_id(id,card); | ||
842 | } | 836 | } |
843 | 837 | ||
844 | /* | 838 | /* |
845 | * Build device list for all present ISA PnP devices. | 839 | * Build device list for all present ISA PnP devices. |
846 | */ | 840 | */ |
847 | |||
848 | static int __init isapnp_build_device_list(void) | 841 | static int __init isapnp_build_device_list(void) |
849 | { | 842 | { |
850 | int csn; | 843 | int csn; |
@@ -858,22 +851,29 @@ static int __init isapnp_build_device_list(void) | |||
858 | isapnp_peek(header, 9); | 851 | isapnp_peek(header, 9); |
859 | checksum = isapnp_checksum(header); | 852 | checksum = isapnp_checksum(header); |
860 | #if 0 | 853 | #if 0 |
861 | printk(KERN_DEBUG "vendor: %02x:%02x:%02x:%02x:%02x:%02x:%02x:%02x:%02x\n", | 854 | printk(KERN_DEBUG |
862 | header[0], header[1], header[2], header[3], | 855 | "vendor: %02x:%02x:%02x:%02x:%02x:%02x:%02x:%02x:%02x\n", |
863 | header[4], header[5], header[6], header[7], header[8]); | 856 | header[0], header[1], header[2], header[3], header[4], |
857 | header[5], header[6], header[7], header[8]); | ||
864 | printk(KERN_DEBUG "checksum = 0x%x\n", checksum); | 858 | printk(KERN_DEBUG "checksum = 0x%x\n", checksum); |
865 | #endif | 859 | #endif |
866 | if ((card = kzalloc(sizeof(struct pnp_card), GFP_KERNEL)) == NULL) | 860 | if ((card = |
861 | kzalloc(sizeof(struct pnp_card), GFP_KERNEL)) == NULL) | ||
867 | continue; | 862 | continue; |
868 | 863 | ||
869 | card->number = csn; | 864 | card->number = csn; |
870 | INIT_LIST_HEAD(&card->devices); | 865 | INIT_LIST_HEAD(&card->devices); |
871 | isapnp_parse_card_id(card, (header[1] << 8) | header[0], (header[3] << 8) | header[2]); | 866 | isapnp_parse_card_id(card, (header[1] << 8) | header[0], |
872 | card->serial = (header[7] << 24) | (header[6] << 16) | (header[5] << 8) | header[4]; | 867 | (header[3] << 8) | header[2]); |
868 | card->serial = | ||
869 | (header[7] << 24) | (header[6] << 16) | (header[5] << 8) | | ||
870 | header[4]; | ||
873 | isapnp_checksum_value = 0x00; | 871 | isapnp_checksum_value = 0x00; |
874 | isapnp_parse_resource_map(card); | 872 | isapnp_parse_resource_map(card); |
875 | if (isapnp_checksum_value != 0x00) | 873 | if (isapnp_checksum_value != 0x00) |
876 | printk(KERN_ERR "isapnp: checksum for device %i is not valid (0x%x)\n", csn, isapnp_checksum_value); | 874 | printk(KERN_ERR |
875 | "isapnp: checksum for device %i is not valid (0x%x)\n", | ||
876 | csn, isapnp_checksum_value); | ||
877 | card->checksum = isapnp_checksum_value; | 877 | card->checksum = isapnp_checksum_value; |
878 | card->protocol = &isapnp_protocol; | 878 | card->protocol = &isapnp_protocol; |
879 | 879 | ||
@@ -890,6 +890,7 @@ static int __init isapnp_build_device_list(void) | |||
890 | int isapnp_present(void) | 890 | int isapnp_present(void) |
891 | { | 891 | { |
892 | struct pnp_card *card; | 892 | struct pnp_card *card; |
893 | |||
893 | pnp_for_each_card(card) { | 894 | pnp_for_each_card(card) { |
894 | if (card->protocol == &isapnp_protocol) | 895 | if (card->protocol == &isapnp_protocol) |
895 | return 1; | 896 | return 1; |
@@ -911,13 +912,13 @@ int isapnp_cfg_begin(int csn, int logdev) | |||
911 | /* it is possible to set RDP only in the isolation phase */ | 912 | /* it is possible to set RDP only in the isolation phase */ |
912 | /* Jens Thoms Toerring <Jens.Toerring@physik.fu-berlin.de> */ | 913 | /* Jens Thoms Toerring <Jens.Toerring@physik.fu-berlin.de> */ |
913 | isapnp_write_byte(0x02, 0x04); /* clear CSN of card */ | 914 | isapnp_write_byte(0x02, 0x04); /* clear CSN of card */ |
914 | mdelay(2); /* is this necessary? */ | 915 | mdelay(2); /* is this necessary? */ |
915 | isapnp_wake(csn); /* bring card into sleep state */ | 916 | isapnp_wake(csn); /* bring card into sleep state */ |
916 | isapnp_wake(0); /* bring card into isolation state */ | 917 | isapnp_wake(0); /* bring card into isolation state */ |
917 | isapnp_set_rdp(); /* reset the RDP port */ | 918 | isapnp_set_rdp(); /* reset the RDP port */ |
918 | udelay(1000); /* delay 1000us */ | 919 | udelay(1000); /* delay 1000us */ |
919 | isapnp_write_byte(0x06, csn); /* reset CSN to previous value */ | 920 | isapnp_write_byte(0x06, csn); /* reset CSN to previous value */ |
920 | udelay(250); /* is this necessary? */ | 921 | udelay(250); /* is this necessary? */ |
921 | #endif | 922 | #endif |
922 | if (logdev >= 0) | 923 | if (logdev >= 0) |
923 | isapnp_device(logdev); | 924 | isapnp_device(logdev); |
@@ -931,12 +932,10 @@ int isapnp_cfg_end(void) | |||
931 | return 0; | 932 | return 0; |
932 | } | 933 | } |
933 | 934 | ||
934 | |||
935 | /* | 935 | /* |
936 | * Inititialization. | 936 | * Initialization. |
937 | */ | 937 | */ |
938 | 938 | ||
939 | |||
940 | EXPORT_SYMBOL(isapnp_protocol); | 939 | EXPORT_SYMBOL(isapnp_protocol); |
941 | EXPORT_SYMBOL(isapnp_present); | 940 | EXPORT_SYMBOL(isapnp_present); |
942 | EXPORT_SYMBOL(isapnp_cfg_begin); | 941 | EXPORT_SYMBOL(isapnp_cfg_begin); |
@@ -946,7 +945,8 @@ EXPORT_SYMBOL(isapnp_read_byte); | |||
946 | #endif | 945 | #endif |
947 | EXPORT_SYMBOL(isapnp_write_byte); | 946 | EXPORT_SYMBOL(isapnp_write_byte); |
948 | 947 | ||
949 | static int isapnp_read_resources(struct pnp_dev *dev, struct pnp_resource_table *res) | 948 | static int isapnp_read_resources(struct pnp_dev *dev, |
949 | struct pnp_resource_table *res) | ||
950 | { | 950 | { |
951 | int tmp, ret; | 951 | int tmp, ret; |
952 | 952 | ||
@@ -960,31 +960,37 @@ static int isapnp_read_resources(struct pnp_dev *dev, struct pnp_resource_table | |||
960 | res->port_resource[tmp].flags = IORESOURCE_IO; | 960 | res->port_resource[tmp].flags = IORESOURCE_IO; |
961 | } | 961 | } |
962 | for (tmp = 0; tmp < PNP_MAX_MEM; tmp++) { | 962 | for (tmp = 0; tmp < PNP_MAX_MEM; tmp++) { |
963 | ret = isapnp_read_word(ISAPNP_CFG_MEM + (tmp << 3)) << 8; | 963 | ret = |
964 | isapnp_read_word(ISAPNP_CFG_MEM + (tmp << 3)) << 8; | ||
964 | if (!ret) | 965 | if (!ret) |
965 | continue; | 966 | continue; |
966 | res->mem_resource[tmp].start = ret; | 967 | res->mem_resource[tmp].start = ret; |
967 | res->mem_resource[tmp].flags = IORESOURCE_MEM; | 968 | res->mem_resource[tmp].flags = IORESOURCE_MEM; |
968 | } | 969 | } |
969 | for (tmp = 0; tmp < PNP_MAX_IRQ; tmp++) { | 970 | for (tmp = 0; tmp < PNP_MAX_IRQ; tmp++) { |
970 | ret = (isapnp_read_word(ISAPNP_CFG_IRQ + (tmp << 1)) >> 8); | 971 | ret = |
972 | (isapnp_read_word(ISAPNP_CFG_IRQ + (tmp << 1)) >> | ||
973 | 8); | ||
971 | if (!ret) | 974 | if (!ret) |
972 | continue; | 975 | continue; |
973 | res->irq_resource[tmp].start = res->irq_resource[tmp].end = ret; | 976 | res->irq_resource[tmp].start = |
977 | res->irq_resource[tmp].end = ret; | ||
974 | res->irq_resource[tmp].flags = IORESOURCE_IRQ; | 978 | res->irq_resource[tmp].flags = IORESOURCE_IRQ; |
975 | } | 979 | } |
976 | for (tmp = 0; tmp < PNP_MAX_DMA; tmp++) { | 980 | for (tmp = 0; tmp < PNP_MAX_DMA; tmp++) { |
977 | ret = isapnp_read_byte(ISAPNP_CFG_DMA + tmp); | 981 | ret = isapnp_read_byte(ISAPNP_CFG_DMA + tmp); |
978 | if (ret == 4) | 982 | if (ret == 4) |
979 | continue; | 983 | continue; |
980 | res->dma_resource[tmp].start = res->dma_resource[tmp].end = ret; | 984 | res->dma_resource[tmp].start = |
985 | res->dma_resource[tmp].end = ret; | ||
981 | res->dma_resource[tmp].flags = IORESOURCE_DMA; | 986 | res->dma_resource[tmp].flags = IORESOURCE_DMA; |
982 | } | 987 | } |
983 | } | 988 | } |
984 | return 0; | 989 | return 0; |
985 | } | 990 | } |
986 | 991 | ||
987 | static int isapnp_get_resources(struct pnp_dev *dev, struct pnp_resource_table * res) | 992 | static int isapnp_get_resources(struct pnp_dev *dev, |
993 | struct pnp_resource_table *res) | ||
988 | { | 994 | { |
989 | int ret; | 995 | int ret; |
990 | pnp_init_resource_table(res); | 996 | pnp_init_resource_table(res); |
@@ -994,24 +1000,44 @@ static int isapnp_get_resources(struct pnp_dev *dev, struct pnp_resource_table * | |||
994 | return ret; | 1000 | return ret; |
995 | } | 1001 | } |
996 | 1002 | ||
997 | static int isapnp_set_resources(struct pnp_dev *dev, struct pnp_resource_table * res) | 1003 | static int isapnp_set_resources(struct pnp_dev *dev, |
1004 | struct pnp_resource_table *res) | ||
998 | { | 1005 | { |
999 | int tmp; | 1006 | int tmp; |
1000 | 1007 | ||
1001 | isapnp_cfg_begin(dev->card->number, dev->number); | 1008 | isapnp_cfg_begin(dev->card->number, dev->number); |
1002 | dev->active = 1; | 1009 | dev->active = 1; |
1003 | for (tmp = 0; tmp < PNP_MAX_PORT && (res->port_resource[tmp].flags & (IORESOURCE_IO | IORESOURCE_UNSET)) == IORESOURCE_IO; tmp++) | 1010 | for (tmp = 0; |
1004 | isapnp_write_word(ISAPNP_CFG_PORT+(tmp<<1), res->port_resource[tmp].start); | 1011 | tmp < PNP_MAX_PORT |
1005 | for (tmp = 0; tmp < PNP_MAX_IRQ && (res->irq_resource[tmp].flags & (IORESOURCE_IRQ | IORESOURCE_UNSET)) == IORESOURCE_IRQ; tmp++) { | 1012 | && (res->port_resource[tmp]. |
1013 | flags & (IORESOURCE_IO | IORESOURCE_UNSET)) == IORESOURCE_IO; | ||
1014 | tmp++) | ||
1015 | isapnp_write_word(ISAPNP_CFG_PORT + (tmp << 1), | ||
1016 | res->port_resource[tmp].start); | ||
1017 | for (tmp = 0; | ||
1018 | tmp < PNP_MAX_IRQ | ||
1019 | && (res->irq_resource[tmp]. | ||
1020 | flags & (IORESOURCE_IRQ | IORESOURCE_UNSET)) == IORESOURCE_IRQ; | ||
1021 | tmp++) { | ||
1006 | int irq = res->irq_resource[tmp].start; | 1022 | int irq = res->irq_resource[tmp].start; |
1007 | if (irq == 2) | 1023 | if (irq == 2) |
1008 | irq = 9; | 1024 | irq = 9; |
1009 | isapnp_write_byte(ISAPNP_CFG_IRQ+(tmp<<1), irq); | 1025 | isapnp_write_byte(ISAPNP_CFG_IRQ + (tmp << 1), irq); |
1010 | } | 1026 | } |
1011 | for (tmp = 0; tmp < PNP_MAX_DMA && (res->dma_resource[tmp].flags & (IORESOURCE_DMA | IORESOURCE_UNSET)) == IORESOURCE_DMA; tmp++) | 1027 | for (tmp = 0; |
1012 | isapnp_write_byte(ISAPNP_CFG_DMA+tmp, res->dma_resource[tmp].start); | 1028 | tmp < PNP_MAX_DMA |
1013 | for (tmp = 0; tmp < PNP_MAX_MEM && (res->mem_resource[tmp].flags & (IORESOURCE_MEM | IORESOURCE_UNSET)) == IORESOURCE_MEM; tmp++) | 1029 | && (res->dma_resource[tmp]. |
1014 | isapnp_write_word(ISAPNP_CFG_MEM+(tmp<<3), (res->mem_resource[tmp].start >> 8) & 0xffff); | 1030 | flags & (IORESOURCE_DMA | IORESOURCE_UNSET)) == IORESOURCE_DMA; |
1031 | tmp++) | ||
1032 | isapnp_write_byte(ISAPNP_CFG_DMA + tmp, | ||
1033 | res->dma_resource[tmp].start); | ||
1034 | for (tmp = 0; | ||
1035 | tmp < PNP_MAX_MEM | ||
1036 | && (res->mem_resource[tmp]. | ||
1037 | flags & (IORESOURCE_MEM | IORESOURCE_UNSET)) == IORESOURCE_MEM; | ||
1038 | tmp++) | ||
1039 | isapnp_write_word(ISAPNP_CFG_MEM + (tmp << 3), | ||
1040 | (res->mem_resource[tmp].start >> 8) & 0xffff); | ||
1015 | /* FIXME: We aren't handling 32bit mems properly here */ | 1041 | /* FIXME: We aren't handling 32bit mems properly here */ |
1016 | isapnp_activate(dev->number); | 1042 | isapnp_activate(dev->number); |
1017 | isapnp_cfg_end(); | 1043 | isapnp_cfg_end(); |
@@ -1030,9 +1056,9 @@ static int isapnp_disable_resources(struct pnp_dev *dev) | |||
1030 | } | 1056 | } |
1031 | 1057 | ||
1032 | struct pnp_protocol isapnp_protocol = { | 1058 | struct pnp_protocol isapnp_protocol = { |
1033 | .name = "ISA Plug and Play", | 1059 | .name = "ISA Plug and Play", |
1034 | .get = isapnp_get_resources, | 1060 | .get = isapnp_get_resources, |
1035 | .set = isapnp_set_resources, | 1061 | .set = isapnp_set_resources, |
1036 | .disable = isapnp_disable_resources, | 1062 | .disable = isapnp_disable_resources, |
1037 | }; | 1063 | }; |
1038 | 1064 | ||
@@ -1053,31 +1079,36 @@ static int __init isapnp_init(void) | |||
1053 | #endif | 1079 | #endif |
1054 | #ifdef ISAPNP_REGION_OK | 1080 | #ifdef ISAPNP_REGION_OK |
1055 | if (!request_region(_PIDXR, 1, "isapnp index")) { | 1081 | if (!request_region(_PIDXR, 1, "isapnp index")) { |
1056 | printk(KERN_ERR "isapnp: Index Register 0x%x already used\n", _PIDXR); | 1082 | printk(KERN_ERR "isapnp: Index Register 0x%x already used\n", |
1083 | _PIDXR); | ||
1057 | return -EBUSY; | 1084 | return -EBUSY; |
1058 | } | 1085 | } |
1059 | #endif | 1086 | #endif |
1060 | if (!request_region(_PNPWRP, 1, "isapnp write")) { | 1087 | if (!request_region(_PNPWRP, 1, "isapnp write")) { |
1061 | printk(KERN_ERR "isapnp: Write Data Register 0x%x already used\n", _PNPWRP); | 1088 | printk(KERN_ERR |
1089 | "isapnp: Write Data Register 0x%x already used\n", | ||
1090 | _PNPWRP); | ||
1062 | #ifdef ISAPNP_REGION_OK | 1091 | #ifdef ISAPNP_REGION_OK |
1063 | release_region(_PIDXR, 1); | 1092 | release_region(_PIDXR, 1); |
1064 | #endif | 1093 | #endif |
1065 | return -EBUSY; | 1094 | return -EBUSY; |
1066 | } | 1095 | } |
1067 | 1096 | ||
1068 | if(pnp_register_protocol(&isapnp_protocol)<0) | 1097 | if (pnp_register_protocol(&isapnp_protocol) < 0) |
1069 | return -EBUSY; | 1098 | return -EBUSY; |
1070 | 1099 | ||
1071 | /* | 1100 | /* |
1072 | * Print a message. The existing ISAPnP code is hanging machines | 1101 | * Print a message. The existing ISAPnP code is hanging machines |
1073 | * so let the user know where. | 1102 | * so let the user know where. |
1074 | */ | 1103 | */ |
1075 | 1104 | ||
1076 | printk(KERN_INFO "isapnp: Scanning for PnP cards...\n"); | 1105 | printk(KERN_INFO "isapnp: Scanning for PnP cards...\n"); |
1077 | if (isapnp_rdp >= 0x203 && isapnp_rdp <= 0x3ff) { | 1106 | if (isapnp_rdp >= 0x203 && isapnp_rdp <= 0x3ff) { |
1078 | isapnp_rdp |= 3; | 1107 | isapnp_rdp |= 3; |
1079 | if (!request_region(isapnp_rdp, 1, "isapnp read")) { | 1108 | if (!request_region(isapnp_rdp, 1, "isapnp read")) { |
1080 | printk(KERN_ERR "isapnp: Read Data Register 0x%x already used\n", isapnp_rdp); | 1109 | printk(KERN_ERR |
1110 | "isapnp: Read Data Register 0x%x already used\n", | ||
1111 | isapnp_rdp); | ||
1081 | #ifdef ISAPNP_REGION_OK | 1112 | #ifdef ISAPNP_REGION_OK |
1082 | release_region(_PIDXR, 1); | 1113 | release_region(_PIDXR, 1); |
1083 | #endif | 1114 | #endif |
@@ -1089,14 +1120,14 @@ static int __init isapnp_init(void) | |||
1089 | isapnp_detected = 1; | 1120 | isapnp_detected = 1; |
1090 | if (isapnp_rdp < 0x203 || isapnp_rdp > 0x3ff) { | 1121 | if (isapnp_rdp < 0x203 || isapnp_rdp > 0x3ff) { |
1091 | cards = isapnp_isolate(); | 1122 | cards = isapnp_isolate(); |
1092 | if (cards < 0 || | 1123 | if (cards < 0 || (isapnp_rdp < 0x203 || isapnp_rdp > 0x3ff)) { |
1093 | (isapnp_rdp < 0x203 || isapnp_rdp > 0x3ff)) { | ||
1094 | #ifdef ISAPNP_REGION_OK | 1124 | #ifdef ISAPNP_REGION_OK |
1095 | release_region(_PIDXR, 1); | 1125 | release_region(_PIDXR, 1); |
1096 | #endif | 1126 | #endif |
1097 | release_region(_PNPWRP, 1); | 1127 | release_region(_PNPWRP, 1); |
1098 | isapnp_detected = 0; | 1128 | isapnp_detected = 0; |
1099 | printk(KERN_INFO "isapnp: No Plug & Play device found\n"); | 1129 | printk(KERN_INFO |
1130 | "isapnp: No Plug & Play device found\n"); | ||
1100 | return 0; | 1131 | return 0; |
1101 | } | 1132 | } |
1102 | request_region(isapnp_rdp, 1, "isapnp read"); | 1133 | request_region(isapnp_rdp, 1, "isapnp read"); |
@@ -1104,19 +1135,23 @@ static int __init isapnp_init(void) | |||
1104 | isapnp_build_device_list(); | 1135 | isapnp_build_device_list(); |
1105 | cards = 0; | 1136 | cards = 0; |
1106 | 1137 | ||
1107 | protocol_for_each_card(&isapnp_protocol,card) { | 1138 | protocol_for_each_card(&isapnp_protocol, card) { |
1108 | cards++; | 1139 | cards++; |
1109 | if (isapnp_verbose) { | 1140 | if (isapnp_verbose) { |
1110 | printk(KERN_INFO "isapnp: Card '%s'\n", card->name[0]?card->name:"Unknown"); | 1141 | printk(KERN_INFO "isapnp: Card '%s'\n", |
1142 | card->name[0] ? card->name : "Unknown"); | ||
1111 | if (isapnp_verbose < 2) | 1143 | if (isapnp_verbose < 2) |
1112 | continue; | 1144 | continue; |
1113 | card_for_each_dev(card,dev) { | 1145 | card_for_each_dev(card, dev) { |
1114 | printk(KERN_INFO "isapnp: Device '%s'\n", dev->name[0]?dev->name:"Unknown"); | 1146 | printk(KERN_INFO "isapnp: Device '%s'\n", |
1147 | dev->name[0] ? dev->name : "Unknown"); | ||
1115 | } | 1148 | } |
1116 | } | 1149 | } |
1117 | } | 1150 | } |
1118 | if (cards) { | 1151 | if (cards) { |
1119 | printk(KERN_INFO "isapnp: %i Plug & Play card%s detected total\n", cards, cards>1?"s":""); | 1152 | printk(KERN_INFO |
1153 | "isapnp: %i Plug & Play card%s detected total\n", cards, | ||
1154 | cards > 1 ? "s" : ""); | ||
1120 | } else { | 1155 | } else { |
1121 | printk(KERN_INFO "isapnp: No Plug & Play card found\n"); | 1156 | printk(KERN_INFO "isapnp: No Plug & Play card found\n"); |
1122 | } | 1157 | } |
@@ -1141,11 +1176,10 @@ __setup("noisapnp", isapnp_setup_disable); | |||
1141 | 1176 | ||
1142 | static int __init isapnp_setup_isapnp(char *str) | 1177 | static int __init isapnp_setup_isapnp(char *str) |
1143 | { | 1178 | { |
1144 | (void)((get_option(&str,&isapnp_rdp) == 2) && | 1179 | (void)((get_option(&str, &isapnp_rdp) == 2) && |
1145 | (get_option(&str,&isapnp_reset) == 2) && | 1180 | (get_option(&str, &isapnp_reset) == 2) && |
1146 | (get_option(&str,&isapnp_verbose) == 2)); | 1181 | (get_option(&str, &isapnp_verbose) == 2)); |
1147 | return 1; | 1182 | return 1; |
1148 | } | 1183 | } |
1149 | 1184 | ||
1150 | __setup("isapnp=", isapnp_setup_isapnp); | 1185 | __setup("isapnp=", isapnp_setup_isapnp); |
1151 | |||
diff --git a/drivers/pnp/isapnp/proc.c b/drivers/pnp/isapnp/proc.c index 40b724ebe23b..3fbc0f9ffc26 100644 --- a/drivers/pnp/isapnp/proc.c +++ b/drivers/pnp/isapnp/proc.c | |||
@@ -2,7 +2,6 @@ | |||
2 | * ISA Plug & Play support | 2 | * ISA Plug & Play support |
3 | * Copyright (c) by Jaroslav Kysela <perex@suse.cz> | 3 | * Copyright (c) by Jaroslav Kysela <perex@suse.cz> |
4 | * | 4 | * |
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | 5 | * This program is free software; you can redistribute it and/or modify |
7 | * it under the terms of the GNU General Public License as published by | 6 | * it under the terms of the GNU General Public License as published by |
8 | * the Free Software Foundation; either version 2 of the License, or | 7 | * the Free Software Foundation; either version 2 of the License, or |
@@ -16,7 +15,6 @@ | |||
16 | * You should have received a copy of the GNU General Public License | 15 | * You should have received a copy of the GNU General Public License |
17 | * along with this program; if not, write to the Free Software | 16 | * along with this program; if not, write to the Free Software |
18 | * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. | 17 | * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. |
19 | * | ||
20 | */ | 18 | */ |
21 | 19 | ||
22 | #include <linux/module.h> | 20 | #include <linux/module.h> |
@@ -54,7 +52,8 @@ static loff_t isapnp_proc_bus_lseek(struct file *file, loff_t off, int whence) | |||
54 | return (file->f_pos = new); | 52 | return (file->f_pos = new); |
55 | } | 53 | } |
56 | 54 | ||
57 | static ssize_t isapnp_proc_bus_read(struct file *file, char __user *buf, size_t nbytes, loff_t *ppos) | 55 | static ssize_t isapnp_proc_bus_read(struct file *file, char __user * buf, |
56 | size_t nbytes, loff_t * ppos) | ||
58 | { | 57 | { |
59 | struct inode *ino = file->f_path.dentry->d_inode; | 58 | struct inode *ino = file->f_path.dentry->d_inode; |
60 | struct proc_dir_entry *dp = PDE(ino); | 59 | struct proc_dir_entry *dp = PDE(ino); |
@@ -74,7 +73,7 @@ static ssize_t isapnp_proc_bus_read(struct file *file, char __user *buf, size_t | |||
74 | return -EINVAL; | 73 | return -EINVAL; |
75 | 74 | ||
76 | isapnp_cfg_begin(dev->card->number, dev->number); | 75 | isapnp_cfg_begin(dev->card->number, dev->number); |
77 | for ( ; pos < 256 && cnt > 0; pos++, buf++, cnt--) { | 76 | for (; pos < 256 && cnt > 0; pos++, buf++, cnt--) { |
78 | unsigned char val; | 77 | unsigned char val; |
79 | val = isapnp_read_byte(pos); | 78 | val = isapnp_read_byte(pos); |
80 | __put_user(val, buf); | 79 | __put_user(val, buf); |
@@ -85,10 +84,9 @@ static ssize_t isapnp_proc_bus_read(struct file *file, char __user *buf, size_t | |||
85 | return nbytes; | 84 | return nbytes; |
86 | } | 85 | } |
87 | 86 | ||
88 | static const struct file_operations isapnp_proc_bus_file_operations = | 87 | static const struct file_operations isapnp_proc_bus_file_operations = { |
89 | { | 88 | .llseek = isapnp_proc_bus_lseek, |
90 | .llseek = isapnp_proc_bus_lseek, | 89 | .read = isapnp_proc_bus_read, |
91 | .read = isapnp_proc_bus_read, | ||
92 | }; | 90 | }; |
93 | 91 | ||
94 | static int isapnp_proc_attach_device(struct pnp_dev *dev) | 92 | static int isapnp_proc_attach_device(struct pnp_dev *dev) |
@@ -139,13 +137,14 @@ static int __exit isapnp_proc_detach_bus(struct pnp_card *bus) | |||
139 | remove_proc_entry(name, isapnp_proc_bus_dir); | 137 | remove_proc_entry(name, isapnp_proc_bus_dir); |
140 | return 0; | 138 | return 0; |
141 | } | 139 | } |
142 | #endif /* MODULE */ | 140 | #endif /* MODULE */ |
143 | 141 | ||
144 | int __init isapnp_proc_init(void) | 142 | int __init isapnp_proc_init(void) |
145 | { | 143 | { |
146 | struct pnp_dev *dev; | 144 | struct pnp_dev *dev; |
145 | |||
147 | isapnp_proc_bus_dir = proc_mkdir("isapnp", proc_bus); | 146 | isapnp_proc_bus_dir = proc_mkdir("isapnp", proc_bus); |
148 | protocol_for_each_dev(&isapnp_protocol,dev) { | 147 | protocol_for_each_dev(&isapnp_protocol, dev) { |
149 | isapnp_proc_attach_device(dev); | 148 | isapnp_proc_attach_device(dev); |
150 | } | 149 | } |
151 | return 0; | 150 | return 0; |
@@ -167,4 +166,4 @@ int __exit isapnp_proc_done(void) | |||
167 | remove_proc_entry("isapnp", proc_bus); | 166 | remove_proc_entry("isapnp", proc_bus); |
168 | return 0; | 167 | return 0; |
169 | } | 168 | } |
170 | #endif /* MODULE */ | 169 | #endif /* MODULE */ |
diff --git a/drivers/pnp/manager.c b/drivers/pnp/manager.c index 57e6ab1004d0..3bda513a6bd3 100644 --- a/drivers/pnp/manager.c +++ b/drivers/pnp/manager.c | |||
@@ -3,7 +3,6 @@ | |||
3 | * | 3 | * |
4 | * based on isapnp.c resource management (c) Jaroslav Kysela <perex@suse.cz> | 4 | * based on isapnp.c resource management (c) Jaroslav Kysela <perex@suse.cz> |
5 | * Copyright 2003 Adam Belay <ambx1@neo.rr.com> | 5 | * Copyright 2003 Adam Belay <ambx1@neo.rr.com> |
6 | * | ||
7 | */ | 6 | */ |
8 | 7 | ||
9 | #include <linux/errno.h> | 8 | #include <linux/errno.h> |
@@ -26,7 +25,8 @@ static int pnp_assign_port(struct pnp_dev *dev, struct pnp_port *rule, int idx) | |||
26 | return -EINVAL; | 25 | return -EINVAL; |
27 | 26 | ||
28 | if (idx >= PNP_MAX_PORT) { | 27 | if (idx >= PNP_MAX_PORT) { |
29 | pnp_err("More than 4 ports is incompatible with pnp specifications."); | 28 | pnp_err |
29 | ("More than 4 ports is incompatible with pnp specifications."); | ||
30 | /* pretend we were successful so at least the manager won't try again */ | 30 | /* pretend we were successful so at least the manager won't try again */ |
31 | return 1; | 31 | return 1; |
32 | } | 32 | } |
@@ -41,11 +41,11 @@ static int pnp_assign_port(struct pnp_dev *dev, struct pnp_port *rule, int idx) | |||
41 | 41 | ||
42 | /* set the initial values */ | 42 | /* set the initial values */ |
43 | *flags |= rule->flags | IORESOURCE_IO; | 43 | *flags |= rule->flags | IORESOURCE_IO; |
44 | *flags &= ~IORESOURCE_UNSET; | 44 | *flags &= ~IORESOURCE_UNSET; |
45 | 45 | ||
46 | if (!rule->size) { | 46 | if (!rule->size) { |
47 | *flags |= IORESOURCE_DISABLED; | 47 | *flags |= IORESOURCE_DISABLED; |
48 | return 1; /* skip disabled resource requests */ | 48 | return 1; /* skip disabled resource requests */ |
49 | } | 49 | } |
50 | 50 | ||
51 | *start = rule->min; | 51 | *start = rule->min; |
@@ -70,7 +70,8 @@ static int pnp_assign_mem(struct pnp_dev *dev, struct pnp_mem *rule, int idx) | |||
70 | return -EINVAL; | 70 | return -EINVAL; |
71 | 71 | ||
72 | if (idx >= PNP_MAX_MEM) { | 72 | if (idx >= PNP_MAX_MEM) { |
73 | pnp_err("More than 8 mems is incompatible with pnp specifications."); | 73 | pnp_err |
74 | ("More than 8 mems is incompatible with pnp specifications."); | ||
74 | /* pretend we were successful so at least the manager won't try again */ | 75 | /* pretend we were successful so at least the manager won't try again */ |
75 | return 1; | 76 | return 1; |
76 | } | 77 | } |
@@ -85,7 +86,7 @@ static int pnp_assign_mem(struct pnp_dev *dev, struct pnp_mem *rule, int idx) | |||
85 | 86 | ||
86 | /* set the initial values */ | 87 | /* set the initial values */ |
87 | *flags |= rule->flags | IORESOURCE_MEM; | 88 | *flags |= rule->flags | IORESOURCE_MEM; |
88 | *flags &= ~IORESOURCE_UNSET; | 89 | *flags &= ~IORESOURCE_UNSET; |
89 | 90 | ||
90 | /* convert pnp flags to standard Linux flags */ | 91 | /* convert pnp flags to standard Linux flags */ |
91 | if (!(rule->flags & IORESOURCE_MEM_WRITEABLE)) | 92 | if (!(rule->flags & IORESOURCE_MEM_WRITEABLE)) |
@@ -99,11 +100,11 @@ static int pnp_assign_mem(struct pnp_dev *dev, struct pnp_mem *rule, int idx) | |||
99 | 100 | ||
100 | if (!rule->size) { | 101 | if (!rule->size) { |
101 | *flags |= IORESOURCE_DISABLED; | 102 | *flags |= IORESOURCE_DISABLED; |
102 | return 1; /* skip disabled resource requests */ | 103 | return 1; /* skip disabled resource requests */ |
103 | } | 104 | } |
104 | 105 | ||
105 | *start = rule->min; | 106 | *start = rule->min; |
106 | *end = *start + rule->size -1; | 107 | *end = *start + rule->size - 1; |
107 | 108 | ||
108 | /* run through until pnp_check_mem is happy */ | 109 | /* run through until pnp_check_mem is happy */ |
109 | while (!pnp_check_mem(dev, idx)) { | 110 | while (!pnp_check_mem(dev, idx)) { |
@@ -115,7 +116,7 @@ static int pnp_assign_mem(struct pnp_dev *dev, struct pnp_mem *rule, int idx) | |||
115 | return 1; | 116 | return 1; |
116 | } | 117 | } |
117 | 118 | ||
118 | static int pnp_assign_irq(struct pnp_dev * dev, struct pnp_irq *rule, int idx) | 119 | static int pnp_assign_irq(struct pnp_dev *dev, struct pnp_irq *rule, int idx) |
119 | { | 120 | { |
120 | resource_size_t *start, *end; | 121 | resource_size_t *start, *end; |
121 | unsigned long *flags; | 122 | unsigned long *flags; |
@@ -130,7 +131,8 @@ static int pnp_assign_irq(struct pnp_dev * dev, struct pnp_irq *rule, int idx) | |||
130 | return -EINVAL; | 131 | return -EINVAL; |
131 | 132 | ||
132 | if (idx >= PNP_MAX_IRQ) { | 133 | if (idx >= PNP_MAX_IRQ) { |
133 | pnp_err("More than 2 irqs is incompatible with pnp specifications."); | 134 | pnp_err |
135 | ("More than 2 irqs is incompatible with pnp specifications."); | ||
134 | /* pretend we were successful so at least the manager won't try again */ | 136 | /* pretend we were successful so at least the manager won't try again */ |
135 | return 1; | 137 | return 1; |
136 | } | 138 | } |
@@ -145,11 +147,11 @@ static int pnp_assign_irq(struct pnp_dev * dev, struct pnp_irq *rule, int idx) | |||
145 | 147 | ||
146 | /* set the initial values */ | 148 | /* set the initial values */ |
147 | *flags |= rule->flags | IORESOURCE_IRQ; | 149 | *flags |= rule->flags | IORESOURCE_IRQ; |
148 | *flags &= ~IORESOURCE_UNSET; | 150 | *flags &= ~IORESOURCE_UNSET; |
149 | 151 | ||
150 | if (bitmap_empty(rule->map, PNP_IRQ_NR)) { | 152 | if (bitmap_empty(rule->map, PNP_IRQ_NR)) { |
151 | *flags |= IORESOURCE_DISABLED; | 153 | *flags |= IORESOURCE_DISABLED; |
152 | return 1; /* skip disabled resource requests */ | 154 | return 1; /* skip disabled resource requests */ |
153 | } | 155 | } |
154 | 156 | ||
155 | /* TBD: need check for >16 IRQ */ | 157 | /* TBD: need check for >16 IRQ */ |
@@ -159,9 +161,9 @@ static int pnp_assign_irq(struct pnp_dev * dev, struct pnp_irq *rule, int idx) | |||
159 | return 1; | 161 | return 1; |
160 | } | 162 | } |
161 | for (i = 0; i < 16; i++) { | 163 | for (i = 0; i < 16; i++) { |
162 | if(test_bit(xtab[i], rule->map)) { | 164 | if (test_bit(xtab[i], rule->map)) { |
163 | *start = *end = xtab[i]; | 165 | *start = *end = xtab[i]; |
164 | if(pnp_check_irq(dev, idx)) | 166 | if (pnp_check_irq(dev, idx)) |
165 | return 1; | 167 | return 1; |
166 | } | 168 | } |
167 | } | 169 | } |
@@ -183,7 +185,8 @@ static int pnp_assign_dma(struct pnp_dev *dev, struct pnp_dma *rule, int idx) | |||
183 | return -EINVAL; | 185 | return -EINVAL; |
184 | 186 | ||
185 | if (idx >= PNP_MAX_DMA) { | 187 | if (idx >= PNP_MAX_DMA) { |
186 | pnp_err("More than 2 dmas is incompatible with pnp specifications."); | 188 | pnp_err |
189 | ("More than 2 dmas is incompatible with pnp specifications."); | ||
187 | /* pretend we were successful so at least the manager won't try again */ | 190 | /* pretend we were successful so at least the manager won't try again */ |
188 | return 1; | 191 | return 1; |
189 | } | 192 | } |
@@ -198,17 +201,17 @@ static int pnp_assign_dma(struct pnp_dev *dev, struct pnp_dma *rule, int idx) | |||
198 | 201 | ||
199 | /* set the initial values */ | 202 | /* set the initial values */ |
200 | *flags |= rule->flags | IORESOURCE_DMA; | 203 | *flags |= rule->flags | IORESOURCE_DMA; |
201 | *flags &= ~IORESOURCE_UNSET; | 204 | *flags &= ~IORESOURCE_UNSET; |
202 | 205 | ||
203 | if (!rule->map) { | 206 | if (!rule->map) { |
204 | *flags |= IORESOURCE_DISABLED; | 207 | *flags |= IORESOURCE_DISABLED; |
205 | return 1; /* skip disabled resource requests */ | 208 | return 1; /* skip disabled resource requests */ |
206 | } | 209 | } |
207 | 210 | ||
208 | for (i = 0; i < 8; i++) { | 211 | for (i = 0; i < 8; i++) { |
209 | if(rule->map & (1<<xtab[i])) { | 212 | if (rule->map & (1 << xtab[i])) { |
210 | *start = *end = xtab[i]; | 213 | *start = *end = xtab[i]; |
211 | if(pnp_check_dma(dev, idx)) | 214 | if (pnp_check_dma(dev, idx)) |
212 | return 1; | 215 | return 1; |
213 | } | 216 | } |
214 | } | 217 | } |
@@ -218,72 +221,80 @@ static int pnp_assign_dma(struct pnp_dev *dev, struct pnp_dma *rule, int idx) | |||
218 | /** | 221 | /** |
219 | * pnp_init_resources - Resets a resource table to default values. | 222 | * pnp_init_resources - Resets a resource table to default values. |
220 | * @table: pointer to the desired resource table | 223 | * @table: pointer to the desired resource table |
221 | * | ||
222 | */ | 224 | */ |
223 | void pnp_init_resource_table(struct pnp_resource_table *table) | 225 | void pnp_init_resource_table(struct pnp_resource_table *table) |
224 | { | 226 | { |
225 | int idx; | 227 | int idx; |
228 | |||
226 | for (idx = 0; idx < PNP_MAX_IRQ; idx++) { | 229 | for (idx = 0; idx < PNP_MAX_IRQ; idx++) { |
227 | table->irq_resource[idx].name = NULL; | 230 | table->irq_resource[idx].name = NULL; |
228 | table->irq_resource[idx].start = -1; | 231 | table->irq_resource[idx].start = -1; |
229 | table->irq_resource[idx].end = -1; | 232 | table->irq_resource[idx].end = -1; |
230 | table->irq_resource[idx].flags = IORESOURCE_IRQ | IORESOURCE_AUTO | IORESOURCE_UNSET; | 233 | table->irq_resource[idx].flags = |
234 | IORESOURCE_IRQ | IORESOURCE_AUTO | IORESOURCE_UNSET; | ||
231 | } | 235 | } |
232 | for (idx = 0; idx < PNP_MAX_DMA; idx++) { | 236 | for (idx = 0; idx < PNP_MAX_DMA; idx++) { |
233 | table->dma_resource[idx].name = NULL; | 237 | table->dma_resource[idx].name = NULL; |
234 | table->dma_resource[idx].start = -1; | 238 | table->dma_resource[idx].start = -1; |
235 | table->dma_resource[idx].end = -1; | 239 | table->dma_resource[idx].end = -1; |
236 | table->dma_resource[idx].flags = IORESOURCE_DMA | IORESOURCE_AUTO | IORESOURCE_UNSET; | 240 | table->dma_resource[idx].flags = |
241 | IORESOURCE_DMA | IORESOURCE_AUTO | IORESOURCE_UNSET; | ||
237 | } | 242 | } |
238 | for (idx = 0; idx < PNP_MAX_PORT; idx++) { | 243 | for (idx = 0; idx < PNP_MAX_PORT; idx++) { |
239 | table->port_resource[idx].name = NULL; | 244 | table->port_resource[idx].name = NULL; |
240 | table->port_resource[idx].start = 0; | 245 | table->port_resource[idx].start = 0; |
241 | table->port_resource[idx].end = 0; | 246 | table->port_resource[idx].end = 0; |
242 | table->port_resource[idx].flags = IORESOURCE_IO | IORESOURCE_AUTO | IORESOURCE_UNSET; | 247 | table->port_resource[idx].flags = |
248 | IORESOURCE_IO | IORESOURCE_AUTO | IORESOURCE_UNSET; | ||
243 | } | 249 | } |
244 | for (idx = 0; idx < PNP_MAX_MEM; idx++) { | 250 | for (idx = 0; idx < PNP_MAX_MEM; idx++) { |
245 | table->mem_resource[idx].name = NULL; | 251 | table->mem_resource[idx].name = NULL; |
246 | table->mem_resource[idx].start = 0; | 252 | table->mem_resource[idx].start = 0; |
247 | table->mem_resource[idx].end = 0; | 253 | table->mem_resource[idx].end = 0; |
248 | table->mem_resource[idx].flags = IORESOURCE_MEM | IORESOURCE_AUTO | IORESOURCE_UNSET; | 254 | table->mem_resource[idx].flags = |
255 | IORESOURCE_MEM | IORESOURCE_AUTO | IORESOURCE_UNSET; | ||
249 | } | 256 | } |
250 | } | 257 | } |
251 | 258 | ||
252 | /** | 259 | /** |
253 | * pnp_clean_resources - clears resources that were not manually set | 260 | * pnp_clean_resources - clears resources that were not manually set |
254 | * @res: the resources to clean | 261 | * @res: the resources to clean |
255 | * | ||
256 | */ | 262 | */ |
257 | static void pnp_clean_resource_table(struct pnp_resource_table * res) | 263 | static void pnp_clean_resource_table(struct pnp_resource_table *res) |
258 | { | 264 | { |
259 | int idx; | 265 | int idx; |
266 | |||
260 | for (idx = 0; idx < PNP_MAX_IRQ; idx++) { | 267 | for (idx = 0; idx < PNP_MAX_IRQ; idx++) { |
261 | if (!(res->irq_resource[idx].flags & IORESOURCE_AUTO)) | 268 | if (!(res->irq_resource[idx].flags & IORESOURCE_AUTO)) |
262 | continue; | 269 | continue; |
263 | res->irq_resource[idx].start = -1; | 270 | res->irq_resource[idx].start = -1; |
264 | res->irq_resource[idx].end = -1; | 271 | res->irq_resource[idx].end = -1; |
265 | res->irq_resource[idx].flags = IORESOURCE_IRQ | IORESOURCE_AUTO | IORESOURCE_UNSET; | 272 | res->irq_resource[idx].flags = |
273 | IORESOURCE_IRQ | IORESOURCE_AUTO | IORESOURCE_UNSET; | ||
266 | } | 274 | } |
267 | for (idx = 0; idx < PNP_MAX_DMA; idx++) { | 275 | for (idx = 0; idx < PNP_MAX_DMA; idx++) { |
268 | if (!(res->dma_resource[idx].flags & IORESOURCE_AUTO)) | 276 | if (!(res->dma_resource[idx].flags & IORESOURCE_AUTO)) |
269 | continue; | 277 | continue; |
270 | res->dma_resource[idx].start = -1; | 278 | res->dma_resource[idx].start = -1; |
271 | res->dma_resource[idx].end = -1; | 279 | res->dma_resource[idx].end = -1; |
272 | res->dma_resource[idx].flags = IORESOURCE_DMA | IORESOURCE_AUTO | IORESOURCE_UNSET; | 280 | res->dma_resource[idx].flags = |
281 | IORESOURCE_DMA | IORESOURCE_AUTO | IORESOURCE_UNSET; | ||
273 | } | 282 | } |
274 | for (idx = 0; idx < PNP_MAX_PORT; idx++) { | 283 | for (idx = 0; idx < PNP_MAX_PORT; idx++) { |
275 | if (!(res->port_resource[idx].flags & IORESOURCE_AUTO)) | 284 | if (!(res->port_resource[idx].flags & IORESOURCE_AUTO)) |
276 | continue; | 285 | continue; |
277 | res->port_resource[idx].start = 0; | 286 | res->port_resource[idx].start = 0; |
278 | res->port_resource[idx].end = 0; | 287 | res->port_resource[idx].end = 0; |
279 | res->port_resource[idx].flags = IORESOURCE_IO | IORESOURCE_AUTO | IORESOURCE_UNSET; | 288 | res->port_resource[idx].flags = |
289 | IORESOURCE_IO | IORESOURCE_AUTO | IORESOURCE_UNSET; | ||
280 | } | 290 | } |
281 | for (idx = 0; idx < PNP_MAX_MEM; idx++) { | 291 | for (idx = 0; idx < PNP_MAX_MEM; idx++) { |
282 | if (!(res->mem_resource[idx].flags & IORESOURCE_AUTO)) | 292 | if (!(res->mem_resource[idx].flags & IORESOURCE_AUTO)) |
283 | continue; | 293 | continue; |
284 | res->mem_resource[idx].start = 0; | 294 | res->mem_resource[idx].start = 0; |
285 | res->mem_resource[idx].end = 0; | 295 | res->mem_resource[idx].end = 0; |
286 | res->mem_resource[idx].flags = IORESOURCE_MEM | IORESOURCE_AUTO | IORESOURCE_UNSET; | 296 | res->mem_resource[idx].flags = |
297 | IORESOURCE_MEM | IORESOURCE_AUTO | IORESOURCE_UNSET; | ||
287 | } | 298 | } |
288 | } | 299 | } |
289 | 300 | ||
@@ -306,7 +317,7 @@ static int pnp_assign_resources(struct pnp_dev *dev, int depnum) | |||
306 | return -ENODEV; | 317 | return -ENODEV; |
307 | 318 | ||
308 | down(&pnp_res_mutex); | 319 | down(&pnp_res_mutex); |
309 | pnp_clean_resource_table(&dev->res); /* start with a fresh slate */ | 320 | pnp_clean_resource_table(&dev->res); /* start with a fresh slate */ |
310 | if (dev->independent) { | 321 | if (dev->independent) { |
311 | port = dev->independent->port; | 322 | port = dev->independent->port; |
312 | mem = dev->independent->mem; | 323 | mem = dev->independent->mem; |
@@ -341,10 +352,11 @@ static int pnp_assign_resources(struct pnp_dev *dev, int depnum) | |||
341 | if (depnum) { | 352 | if (depnum) { |
342 | struct pnp_option *dep; | 353 | struct pnp_option *dep; |
343 | int i; | 354 | int i; |
344 | for (i=1,dep=dev->dependent; i<depnum; i++, dep=dep->next) | 355 | for (i = 1, dep = dev->dependent; i < depnum; |
345 | if(!dep) | 356 | i++, dep = dep->next) |
357 | if (!dep) | ||
346 | goto fail; | 358 | goto fail; |
347 | port =dep->port; | 359 | port = dep->port; |
348 | mem = dep->mem; | 360 | mem = dep->mem; |
349 | irq = dep->irq; | 361 | irq = dep->irq; |
350 | dma = dep->dma; | 362 | dma = dep->dma; |
@@ -378,7 +390,7 @@ static int pnp_assign_resources(struct pnp_dev *dev, int depnum) | |||
378 | up(&pnp_res_mutex); | 390 | up(&pnp_res_mutex); |
379 | return 1; | 391 | return 1; |
380 | 392 | ||
381 | fail: | 393 | fail: |
382 | pnp_clean_resource_table(&dev->res); | 394 | pnp_clean_resource_table(&dev->res); |
383 | up(&pnp_res_mutex); | 395 | up(&pnp_res_mutex); |
384 | return 0; | 396 | return 0; |
@@ -392,10 +404,12 @@ fail: | |||
392 | * | 404 | * |
393 | * This function can be used by drivers that want to manually set thier resources. | 405 | * This function can be used by drivers that want to manually set thier resources. |
394 | */ | 406 | */ |
395 | int pnp_manual_config_dev(struct pnp_dev *dev, struct pnp_resource_table * res, int mode) | 407 | int pnp_manual_config_dev(struct pnp_dev *dev, struct pnp_resource_table *res, |
408 | int mode) | ||
396 | { | 409 | { |
397 | int i; | 410 | int i; |
398 | struct pnp_resource_table * bak; | 411 | struct pnp_resource_table *bak; |
412 | |||
399 | if (!dev || !res) | 413 | if (!dev || !res) |
400 | return -EINVAL; | 414 | return -EINVAL; |
401 | if (!pnp_can_configure(dev)) | 415 | if (!pnp_can_configure(dev)) |
@@ -409,19 +423,19 @@ int pnp_manual_config_dev(struct pnp_dev *dev, struct pnp_resource_table * res, | |||
409 | dev->res = *res; | 423 | dev->res = *res; |
410 | if (!(mode & PNP_CONFIG_FORCE)) { | 424 | if (!(mode & PNP_CONFIG_FORCE)) { |
411 | for (i = 0; i < PNP_MAX_PORT; i++) { | 425 | for (i = 0; i < PNP_MAX_PORT; i++) { |
412 | if(!pnp_check_port(dev,i)) | 426 | if (!pnp_check_port(dev, i)) |
413 | goto fail; | 427 | goto fail; |
414 | } | 428 | } |
415 | for (i = 0; i < PNP_MAX_MEM; i++) { | 429 | for (i = 0; i < PNP_MAX_MEM; i++) { |
416 | if(!pnp_check_mem(dev,i)) | 430 | if (!pnp_check_mem(dev, i)) |
417 | goto fail; | 431 | goto fail; |
418 | } | 432 | } |
419 | for (i = 0; i < PNP_MAX_IRQ; i++) { | 433 | for (i = 0; i < PNP_MAX_IRQ; i++) { |
420 | if(!pnp_check_irq(dev,i)) | 434 | if (!pnp_check_irq(dev, i)) |
421 | goto fail; | 435 | goto fail; |
422 | } | 436 | } |
423 | for (i = 0; i < PNP_MAX_DMA; i++) { | 437 | for (i = 0; i < PNP_MAX_DMA; i++) { |
424 | if(!pnp_check_dma(dev,i)) | 438 | if (!pnp_check_dma(dev, i)) |
425 | goto fail; | 439 | goto fail; |
426 | } | 440 | } |
427 | } | 441 | } |
@@ -430,7 +444,7 @@ int pnp_manual_config_dev(struct pnp_dev *dev, struct pnp_resource_table * res, | |||
430 | kfree(bak); | 444 | kfree(bak); |
431 | return 0; | 445 | return 0; |
432 | 446 | ||
433 | fail: | 447 | fail: |
434 | dev->res = *bak; | 448 | dev->res = *bak; |
435 | up(&pnp_res_mutex); | 449 | up(&pnp_res_mutex); |
436 | kfree(bak); | 450 | kfree(bak); |
@@ -440,18 +454,18 @@ fail: | |||
440 | /** | 454 | /** |
441 | * pnp_auto_config_dev - automatically assigns resources to a device | 455 | * pnp_auto_config_dev - automatically assigns resources to a device |
442 | * @dev: pointer to the desired device | 456 | * @dev: pointer to the desired device |
443 | * | ||
444 | */ | 457 | */ |
445 | int pnp_auto_config_dev(struct pnp_dev *dev) | 458 | int pnp_auto_config_dev(struct pnp_dev *dev) |
446 | { | 459 | { |
447 | struct pnp_option *dep; | 460 | struct pnp_option *dep; |
448 | int i = 1; | 461 | int i = 1; |
449 | 462 | ||
450 | if(!dev) | 463 | if (!dev) |
451 | return -EINVAL; | 464 | return -EINVAL; |
452 | 465 | ||
453 | if(!pnp_can_configure(dev)) { | 466 | if (!pnp_can_configure(dev)) { |
454 | pnp_dbg("Device %s does not support resource configuration.", dev->dev.bus_id); | 467 | pnp_dbg("Device %s does not support resource configuration.", |
468 | dev->dev.bus_id); | ||
455 | return -ENODEV; | 469 | return -ENODEV; |
456 | } | 470 | } |
457 | 471 | ||
@@ -476,23 +490,22 @@ int pnp_auto_config_dev(struct pnp_dev *dev) | |||
476 | * pnp_start_dev - low-level start of the PnP device | 490 | * pnp_start_dev - low-level start of the PnP device |
477 | * @dev: pointer to the desired device | 491 | * @dev: pointer to the desired device |
478 | * | 492 | * |
479 | * assumes that resources have alread been allocated | 493 | * assumes that resources have already been allocated |
480 | */ | 494 | */ |
481 | |||
482 | int pnp_start_dev(struct pnp_dev *dev) | 495 | int pnp_start_dev(struct pnp_dev *dev) |
483 | { | 496 | { |
484 | if (!pnp_can_write(dev)) { | 497 | if (!pnp_can_write(dev)) { |
485 | pnp_dbg("Device %s does not support activation.", dev->dev.bus_id); | 498 | pnp_dbg("Device %s does not support activation.", |
499 | dev->dev.bus_id); | ||
486 | return -EINVAL; | 500 | return -EINVAL; |
487 | } | 501 | } |
488 | 502 | ||
489 | if (dev->protocol->set(dev, &dev->res)<0) { | 503 | if (dev->protocol->set(dev, &dev->res) < 0) { |
490 | pnp_err("Failed to activate device %s.", dev->dev.bus_id); | 504 | pnp_err("Failed to activate device %s.", dev->dev.bus_id); |
491 | return -EIO; | 505 | return -EIO; |
492 | } | 506 | } |
493 | 507 | ||
494 | pnp_info("Device %s activated.", dev->dev.bus_id); | 508 | pnp_info("Device %s activated.", dev->dev.bus_id); |
495 | |||
496 | return 0; | 509 | return 0; |
497 | } | 510 | } |
498 | 511 | ||
@@ -502,20 +515,19 @@ int pnp_start_dev(struct pnp_dev *dev) | |||
502 | * | 515 | * |
503 | * does not free resources | 516 | * does not free resources |
504 | */ | 517 | */ |
505 | |||
506 | int pnp_stop_dev(struct pnp_dev *dev) | 518 | int pnp_stop_dev(struct pnp_dev *dev) |
507 | { | 519 | { |
508 | if (!pnp_can_disable(dev)) { | 520 | if (!pnp_can_disable(dev)) { |
509 | pnp_dbg("Device %s does not support disabling.", dev->dev.bus_id); | 521 | pnp_dbg("Device %s does not support disabling.", |
522 | dev->dev.bus_id); | ||
510 | return -EINVAL; | 523 | return -EINVAL; |
511 | } | 524 | } |
512 | if (dev->protocol->disable(dev)<0) { | 525 | if (dev->protocol->disable(dev) < 0) { |
513 | pnp_err("Failed to disable device %s.", dev->dev.bus_id); | 526 | pnp_err("Failed to disable device %s.", dev->dev.bus_id); |
514 | return -EIO; | 527 | return -EIO; |
515 | } | 528 | } |
516 | 529 | ||
517 | pnp_info("Device %s disabled.", dev->dev.bus_id); | 530 | pnp_info("Device %s disabled.", dev->dev.bus_id); |
518 | |||
519 | return 0; | 531 | return 0; |
520 | } | 532 | } |
521 | 533 | ||
@@ -531,9 +543,8 @@ int pnp_activate_dev(struct pnp_dev *dev) | |||
531 | 543 | ||
532 | if (!dev) | 544 | if (!dev) |
533 | return -EINVAL; | 545 | return -EINVAL; |
534 | if (dev->active) { | 546 | if (dev->active) |
535 | return 0; /* the device is already active */ | 547 | return 0; /* the device is already active */ |
536 | } | ||
537 | 548 | ||
538 | /* ensure resources are allocated */ | 549 | /* ensure resources are allocated */ |
539 | if (pnp_auto_config_dev(dev)) | 550 | if (pnp_auto_config_dev(dev)) |
@@ -544,7 +555,6 @@ int pnp_activate_dev(struct pnp_dev *dev) | |||
544 | return error; | 555 | return error; |
545 | 556 | ||
546 | dev->active = 1; | 557 | dev->active = 1; |
547 | |||
548 | return 1; | 558 | return 1; |
549 | } | 559 | } |
550 | 560 | ||
@@ -558,11 +568,10 @@ int pnp_disable_dev(struct pnp_dev *dev) | |||
558 | { | 568 | { |
559 | int error; | 569 | int error; |
560 | 570 | ||
561 | if (!dev) | 571 | if (!dev) |
562 | return -EINVAL; | 572 | return -EINVAL; |
563 | if (!dev->active) { | 573 | if (!dev->active) |
564 | return 0; /* the device is already disabled */ | 574 | return 0; /* the device is already disabled */ |
565 | } | ||
566 | 575 | ||
567 | error = pnp_stop_dev(dev); | 576 | error = pnp_stop_dev(dev); |
568 | if (error) | 577 | if (error) |
@@ -583,10 +592,9 @@ int pnp_disable_dev(struct pnp_dev *dev) | |||
583 | * @resource: pointer to resource to be changed | 592 | * @resource: pointer to resource to be changed |
584 | * @start: start of region | 593 | * @start: start of region |
585 | * @size: size of region | 594 | * @size: size of region |
586 | * | ||
587 | */ | 595 | */ |
588 | void pnp_resource_change(struct resource *resource, resource_size_t start, | 596 | void pnp_resource_change(struct resource *resource, resource_size_t start, |
589 | resource_size_t size) | 597 | resource_size_t size) |
590 | { | 598 | { |
591 | if (resource == NULL) | 599 | if (resource == NULL) |
592 | return; | 600 | return; |
@@ -595,11 +603,7 @@ void pnp_resource_change(struct resource *resource, resource_size_t start, | |||
595 | resource->end = start + size - 1; | 603 | resource->end = start + size - 1; |
596 | } | 604 | } |
597 | 605 | ||
598 | |||
599 | EXPORT_SYMBOL(pnp_manual_config_dev); | 606 | EXPORT_SYMBOL(pnp_manual_config_dev); |
600 | #if 0 | ||
601 | EXPORT_SYMBOL(pnp_auto_config_dev); | ||
602 | #endif | ||
603 | EXPORT_SYMBOL(pnp_start_dev); | 607 | EXPORT_SYMBOL(pnp_start_dev); |
604 | EXPORT_SYMBOL(pnp_stop_dev); | 608 | EXPORT_SYMBOL(pnp_stop_dev); |
605 | EXPORT_SYMBOL(pnp_activate_dev); | 609 | EXPORT_SYMBOL(pnp_activate_dev); |
diff --git a/drivers/pnp/pnpacpi/core.c b/drivers/pnp/pnpacpi/core.c index a00548799e98..6a2a3c2f4d5e 100644 --- a/drivers/pnp/pnpacpi/core.c +++ b/drivers/pnp/pnpacpi/core.c | |||
@@ -21,7 +21,10 @@ | |||
21 | 21 | ||
22 | #include <linux/acpi.h> | 22 | #include <linux/acpi.h> |
23 | #include <linux/pnp.h> | 23 | #include <linux/pnp.h> |
24 | #include <linux/mod_devicetable.h> | ||
24 | #include <acpi/acpi_bus.h> | 25 | #include <acpi/acpi_bus.h> |
26 | #include <acpi/actypes.h> | ||
27 | |||
25 | #include "pnpacpi.h" | 28 | #include "pnpacpi.h" |
26 | 29 | ||
27 | static int num = 0; | 30 | static int num = 0; |
@@ -31,17 +34,19 @@ static int num = 0; | |||
31 | * used by the kernel (PCI root, ...), as it is harmless and there were | 34 | * used by the kernel (PCI root, ...), as it is harmless and there were |
32 | * already present in pnpbios. But there is an exception for devices that | 35 | * already present in pnpbios. But there is an exception for devices that |
33 | * have irqs (PIC, Timer) because we call acpi_register_gsi. | 36 | * have irqs (PIC, Timer) because we call acpi_register_gsi. |
34 | * Finaly only devices that have a CRS method need to be in this list. | 37 | * Finally, only devices that have a CRS method need to be in this list. |
35 | */ | 38 | */ |
36 | static char __initdata excluded_id_list[] = | 39 | static struct __initdata acpi_device_id excluded_id_list[] = { |
37 | "PNP0C09," /* EC */ | 40 | {"PNP0C09", 0}, /* EC */ |
38 | "PNP0C0F," /* Link device */ | 41 | {"PNP0C0F", 0}, /* Link device */ |
39 | "PNP0000," /* PIC */ | 42 | {"PNP0000", 0}, /* PIC */ |
40 | "PNP0100," /* Timer */ | 43 | {"PNP0100", 0}, /* Timer */ |
41 | ; | 44 | {"", 0}, |
45 | }; | ||
46 | |||
42 | static inline int is_exclusive_device(struct acpi_device *dev) | 47 | static inline int is_exclusive_device(struct acpi_device *dev) |
43 | { | 48 | { |
44 | return (!acpi_match_ids(dev, excluded_id_list)); | 49 | return (!acpi_match_device_ids(dev, excluded_id_list)); |
45 | } | 50 | } |
46 | 51 | ||
47 | /* | 52 | /* |
@@ -79,15 +84,18 @@ static void __init pnpidacpi_to_pnpid(char *id, char *str) | |||
79 | str[7] = '\0'; | 84 | str[7] = '\0'; |
80 | } | 85 | } |
81 | 86 | ||
82 | static int pnpacpi_get_resources(struct pnp_dev * dev, struct pnp_resource_table * res) | 87 | static int pnpacpi_get_resources(struct pnp_dev *dev, |
88 | struct pnp_resource_table *res) | ||
83 | { | 89 | { |
84 | acpi_status status; | 90 | acpi_status status; |
85 | status = pnpacpi_parse_allocated_resource((acpi_handle)dev->data, | 91 | |
86 | &dev->res); | 92 | status = pnpacpi_parse_allocated_resource((acpi_handle) dev->data, |
93 | &dev->res); | ||
87 | return ACPI_FAILURE(status) ? -ENODEV : 0; | 94 | return ACPI_FAILURE(status) ? -ENODEV : 0; |
88 | } | 95 | } |
89 | 96 | ||
90 | static int pnpacpi_set_resources(struct pnp_dev * dev, struct pnp_resource_table * res) | 97 | static int pnpacpi_set_resources(struct pnp_dev *dev, |
98 | struct pnp_resource_table *res) | ||
91 | { | 99 | { |
92 | acpi_handle handle = dev->data; | 100 | acpi_handle handle = dev->data; |
93 | struct acpi_buffer buffer; | 101 | struct acpi_buffer buffer; |
@@ -114,16 +122,32 @@ static int pnpacpi_disable_resources(struct pnp_dev *dev) | |||
114 | acpi_status status; | 122 | acpi_status status; |
115 | 123 | ||
116 | /* acpi_unregister_gsi(pnp_irq(dev, 0)); */ | 124 | /* acpi_unregister_gsi(pnp_irq(dev, 0)); */ |
117 | status = acpi_evaluate_object((acpi_handle)dev->data, | 125 | status = acpi_evaluate_object((acpi_handle) dev->data, |
118 | "_DIS", NULL, NULL); | 126 | "_DIS", NULL, NULL); |
119 | return ACPI_FAILURE(status) ? -ENODEV : 0; | 127 | return ACPI_FAILURE(status) ? -ENODEV : 0; |
120 | } | 128 | } |
121 | 129 | ||
130 | static int pnpacpi_suspend(struct pnp_dev *dev, pm_message_t state) | ||
131 | { | ||
132 | return acpi_bus_set_power((acpi_handle) dev->data, | ||
133 | acpi_pm_device_sleep_state(&dev->dev, | ||
134 | device_may_wakeup | ||
135 | (&dev->dev), | ||
136 | NULL)); | ||
137 | } | ||
138 | |||
139 | static int pnpacpi_resume(struct pnp_dev *dev) | ||
140 | { | ||
141 | return acpi_bus_set_power((acpi_handle) dev->data, ACPI_STATE_D0); | ||
142 | } | ||
143 | |||
122 | static struct pnp_protocol pnpacpi_protocol = { | 144 | static struct pnp_protocol pnpacpi_protocol = { |
123 | .name = "Plug and Play ACPI", | 145 | .name = "Plug and Play ACPI", |
124 | .get = pnpacpi_get_resources, | 146 | .get = pnpacpi_get_resources, |
125 | .set = pnpacpi_set_resources, | 147 | .set = pnpacpi_set_resources, |
126 | .disable = pnpacpi_disable_resources, | 148 | .disable = pnpacpi_disable_resources, |
149 | .suspend = pnpacpi_suspend, | ||
150 | .resume = pnpacpi_resume, | ||
127 | }; | 151 | }; |
128 | 152 | ||
129 | static int __init pnpacpi_add_device(struct acpi_device *device) | 153 | static int __init pnpacpi_add_device(struct acpi_device *device) |
@@ -135,17 +159,17 @@ static int __init pnpacpi_add_device(struct acpi_device *device) | |||
135 | 159 | ||
136 | status = acpi_get_handle(device->handle, "_CRS", &temp); | 160 | status = acpi_get_handle(device->handle, "_CRS", &temp); |
137 | if (ACPI_FAILURE(status) || !ispnpidacpi(acpi_device_hid(device)) || | 161 | if (ACPI_FAILURE(status) || !ispnpidacpi(acpi_device_hid(device)) || |
138 | is_exclusive_device(device)) | 162 | is_exclusive_device(device)) |
139 | return 0; | 163 | return 0; |
140 | 164 | ||
141 | pnp_dbg("ACPI device : hid %s", acpi_device_hid(device)); | 165 | pnp_dbg("ACPI device : hid %s", acpi_device_hid(device)); |
142 | dev = kzalloc(sizeof(struct pnp_dev), GFP_KERNEL); | 166 | dev = kzalloc(sizeof(struct pnp_dev), GFP_KERNEL); |
143 | if (!dev) { | 167 | if (!dev) { |
144 | pnp_err("Out of memory"); | 168 | pnp_err("Out of memory"); |
145 | return -ENOMEM; | 169 | return -ENOMEM; |
146 | } | 170 | } |
147 | dev->data = device->handle; | 171 | dev->data = device->handle; |
148 | /* .enabled means if the device can decode the resources */ | 172 | /* .enabled means the device can decode the resources */ |
149 | dev->active = device->status.enabled; | 173 | dev->active = device->status.enabled; |
150 | status = acpi_get_handle(device->handle, "_SRS", &temp); | 174 | status = acpi_get_handle(device->handle, "_SRS", &temp); |
151 | if (ACPI_SUCCESS(status)) | 175 | if (ACPI_SUCCESS(status)) |
@@ -175,20 +199,23 @@ static int __init pnpacpi_add_device(struct acpi_device *device) | |||
175 | pnpidacpi_to_pnpid(acpi_device_hid(device), dev_id->id); | 199 | pnpidacpi_to_pnpid(acpi_device_hid(device), dev_id->id); |
176 | pnp_add_id(dev_id, dev); | 200 | pnp_add_id(dev_id, dev); |
177 | 201 | ||
178 | if(dev->active) { | 202 | if (dev->active) { |
179 | /* parse allocated resource */ | 203 | /* parse allocated resource */ |
180 | status = pnpacpi_parse_allocated_resource(device->handle, &dev->res); | 204 | status = pnpacpi_parse_allocated_resource(device->handle, |
205 | &dev->res); | ||
181 | if (ACPI_FAILURE(status) && (status != AE_NOT_FOUND)) { | 206 | if (ACPI_FAILURE(status) && (status != AE_NOT_FOUND)) { |
182 | pnp_err("PnPACPI: METHOD_NAME__CRS failure for %s", dev_id->id); | 207 | pnp_err("PnPACPI: METHOD_NAME__CRS failure for %s", |
208 | dev_id->id); | ||
183 | goto err1; | 209 | goto err1; |
184 | } | 210 | } |
185 | } | 211 | } |
186 | 212 | ||
187 | if(dev->capabilities & PNP_CONFIGURABLE) { | 213 | if (dev->capabilities & PNP_CONFIGURABLE) { |
188 | status = pnpacpi_parse_resource_option_data(device->handle, | 214 | status = pnpacpi_parse_resource_option_data(device->handle, |
189 | dev); | 215 | dev); |
190 | if (ACPI_FAILURE(status) && (status != AE_NOT_FOUND)) { | 216 | if (ACPI_FAILURE(status) && (status != AE_NOT_FOUND)) { |
191 | pnp_err("PnPACPI: METHOD_NAME__PRS failure for %s", dev_id->id); | 217 | pnp_err("PnPACPI: METHOD_NAME__PRS failure for %s", |
218 | dev_id->id); | ||
192 | goto err1; | 219 | goto err1; |
193 | } | 220 | } |
194 | } | 221 | } |
@@ -214,18 +241,19 @@ static int __init pnpacpi_add_device(struct acpi_device *device) | |||
214 | if (!dev->active) | 241 | if (!dev->active) |
215 | pnp_init_resource_table(&dev->res); | 242 | pnp_init_resource_table(&dev->res); |
216 | pnp_add_device(dev); | 243 | pnp_add_device(dev); |
217 | num ++; | 244 | num++; |
218 | 245 | ||
219 | return AE_OK; | 246 | return AE_OK; |
220 | err1: | 247 | err1: |
221 | kfree(dev_id); | 248 | kfree(dev_id); |
222 | err: | 249 | err: |
223 | kfree(dev); | 250 | kfree(dev); |
224 | return -EINVAL; | 251 | return -EINVAL; |
225 | } | 252 | } |
226 | 253 | ||
227 | static acpi_status __init pnpacpi_add_device_handler(acpi_handle handle, | 254 | static acpi_status __init pnpacpi_add_device_handler(acpi_handle handle, |
228 | u32 lvl, void *context, void **rv) | 255 | u32 lvl, void *context, |
256 | void **rv) | ||
229 | { | 257 | { |
230 | struct acpi_device *device; | 258 | struct acpi_device *device; |
231 | 259 | ||
@@ -238,23 +266,22 @@ static acpi_status __init pnpacpi_add_device_handler(acpi_handle handle, | |||
238 | 266 | ||
239 | static int __init acpi_pnp_match(struct device *dev, void *_pnp) | 267 | static int __init acpi_pnp_match(struct device *dev, void *_pnp) |
240 | { | 268 | { |
241 | struct acpi_device *acpi = to_acpi_device(dev); | 269 | struct acpi_device *acpi = to_acpi_device(dev); |
242 | struct pnp_dev *pnp = _pnp; | 270 | struct pnp_dev *pnp = _pnp; |
243 | 271 | ||
244 | /* true means it matched */ | 272 | /* true means it matched */ |
245 | return acpi->flags.hardware_id | 273 | return acpi->flags.hardware_id |
246 | && !acpi_get_physical_device(acpi->handle) | 274 | && !acpi_get_physical_device(acpi->handle) |
247 | && compare_pnp_id(pnp->id, acpi->pnp.hardware_id); | 275 | && compare_pnp_id(pnp->id, acpi->pnp.hardware_id); |
248 | } | 276 | } |
249 | 277 | ||
250 | static int __init acpi_pnp_find_device(struct device *dev, acpi_handle *handle) | 278 | static int __init acpi_pnp_find_device(struct device *dev, acpi_handle * handle) |
251 | { | 279 | { |
252 | struct device *adev; | 280 | struct device *adev; |
253 | struct acpi_device *acpi; | 281 | struct acpi_device *acpi; |
254 | 282 | ||
255 | adev = bus_find_device(&acpi_bus_type, NULL, | 283 | adev = bus_find_device(&acpi_bus_type, NULL, |
256 | to_pnp_dev(dev), | 284 | to_pnp_dev(dev), acpi_pnp_match); |
257 | acpi_pnp_match); | ||
258 | if (!adev) | 285 | if (!adev) |
259 | return -ENODEV; | 286 | return -ENODEV; |
260 | 287 | ||
@@ -268,7 +295,7 @@ static int __init acpi_pnp_find_device(struct device *dev, acpi_handle *handle) | |||
268 | * pnpdev->dev.archdata.acpi_handle point to its ACPI sibling. | 295 | * pnpdev->dev.archdata.acpi_handle point to its ACPI sibling. |
269 | */ | 296 | */ |
270 | static struct acpi_bus_type __initdata acpi_pnp_bus = { | 297 | static struct acpi_bus_type __initdata acpi_pnp_bus = { |
271 | .bus = &pnp_bus_type, | 298 | .bus = &pnp_bus_type, |
272 | .find_device = acpi_pnp_find_device, | 299 | .find_device = acpi_pnp_find_device, |
273 | }; | 300 | }; |
274 | 301 | ||
@@ -288,6 +315,7 @@ static int __init pnpacpi_init(void) | |||
288 | pnp_platform_devices = 1; | 315 | pnp_platform_devices = 1; |
289 | return 0; | 316 | return 0; |
290 | } | 317 | } |
318 | |||
291 | subsys_initcall(pnpacpi_init); | 319 | subsys_initcall(pnpacpi_init); |
292 | 320 | ||
293 | static int __init pnpacpi_setup(char *str) | 321 | static int __init pnpacpi_setup(char *str) |
@@ -298,8 +326,5 @@ static int __init pnpacpi_setup(char *str) | |||
298 | pnpacpi_disabled = 1; | 326 | pnpacpi_disabled = 1; |
299 | return 1; | 327 | return 1; |
300 | } | 328 | } |
301 | __setup("pnpacpi=", pnpacpi_setup); | ||
302 | 329 | ||
303 | #if 0 | 330 | __setup("pnpacpi=", pnpacpi_setup); |
304 | EXPORT_SYMBOL(pnpacpi_protocol); | ||
305 | #endif | ||
diff --git a/drivers/pnp/pnpacpi/rsparser.c b/drivers/pnp/pnpacpi/rsparser.c index 118ac9779b3c..ce5027feb3da 100644 --- a/drivers/pnp/pnpacpi/rsparser.c +++ b/drivers/pnp/pnpacpi/rsparser.c | |||
@@ -40,8 +40,7 @@ static int irq_flags(int triggering, int polarity) | |||
40 | flag = IORESOURCE_IRQ_LOWLEVEL; | 40 | flag = IORESOURCE_IRQ_LOWLEVEL; |
41 | else | 41 | else |
42 | flag = IORESOURCE_IRQ_HIGHLEVEL; | 42 | flag = IORESOURCE_IRQ_HIGHLEVEL; |
43 | } | 43 | } else { |
44 | else { | ||
45 | if (polarity == ACPI_ACTIVE_LOW) | 44 | if (polarity == ACPI_ACTIVE_LOW) |
46 | flag = IORESOURCE_IRQ_LOWEDGE; | 45 | flag = IORESOURCE_IRQ_LOWEDGE; |
47 | else | 46 | else |
@@ -72,9 +71,9 @@ static void decode_irq_flags(int flag, int *triggering, int *polarity) | |||
72 | } | 71 | } |
73 | } | 72 | } |
74 | 73 | ||
75 | static void | 74 | static void pnpacpi_parse_allocated_irqresource(struct pnp_resource_table *res, |
76 | pnpacpi_parse_allocated_irqresource(struct pnp_resource_table *res, u32 gsi, | 75 | u32 gsi, int triggering, |
77 | int triggering, int polarity, int shareable) | 76 | int polarity, int shareable) |
78 | { | 77 | { |
79 | int i = 0; | 78 | int i = 0; |
80 | int irq; | 79 | int irq; |
@@ -83,12 +82,12 @@ pnpacpi_parse_allocated_irqresource(struct pnp_resource_table *res, u32 gsi, | |||
83 | return; | 82 | return; |
84 | 83 | ||
85 | while (!(res->irq_resource[i].flags & IORESOURCE_UNSET) && | 84 | while (!(res->irq_resource[i].flags & IORESOURCE_UNSET) && |
86 | i < PNP_MAX_IRQ) | 85 | i < PNP_MAX_IRQ) |
87 | i++; | 86 | i++; |
88 | if (i >= PNP_MAX_IRQ) | 87 | if (i >= PNP_MAX_IRQ) |
89 | return; | 88 | return; |
90 | 89 | ||
91 | res->irq_resource[i].flags = IORESOURCE_IRQ; // Also clears _UNSET flag | 90 | res->irq_resource[i].flags = IORESOURCE_IRQ; // Also clears _UNSET flag |
92 | res->irq_resource[i].flags |= irq_flags(triggering, polarity); | 91 | res->irq_resource[i].flags |= irq_flags(triggering, polarity); |
93 | irq = acpi_register_gsi(gsi, triggering, polarity); | 92 | irq = acpi_register_gsi(gsi, triggering, polarity); |
94 | if (irq < 0) { | 93 | if (irq < 0) { |
@@ -147,17 +146,19 @@ static int dma_flags(int type, int bus_master, int transfer) | |||
147 | return flags; | 146 | return flags; |
148 | } | 147 | } |
149 | 148 | ||
150 | static void | 149 | static void pnpacpi_parse_allocated_dmaresource(struct pnp_resource_table *res, |
151 | pnpacpi_parse_allocated_dmaresource(struct pnp_resource_table *res, u32 dma, | 150 | u32 dma, int type, |
152 | int type, int bus_master, int transfer) | 151 | int bus_master, int transfer) |
153 | { | 152 | { |
154 | int i = 0; | 153 | int i = 0; |
154 | |||
155 | while (i < PNP_MAX_DMA && | 155 | while (i < PNP_MAX_DMA && |
156 | !(res->dma_resource[i].flags & IORESOURCE_UNSET)) | 156 | !(res->dma_resource[i].flags & IORESOURCE_UNSET)) |
157 | i++; | 157 | i++; |
158 | if (i < PNP_MAX_DMA) { | 158 | if (i < PNP_MAX_DMA) { |
159 | res->dma_resource[i].flags = IORESOURCE_DMA; // Also clears _UNSET flag | 159 | res->dma_resource[i].flags = IORESOURCE_DMA; // Also clears _UNSET flag |
160 | res->dma_resource[i].flags |= dma_flags(type, bus_master, transfer); | 160 | res->dma_resource[i].flags |= |
161 | dma_flags(type, bus_master, transfer); | ||
161 | if (dma == -1) { | 162 | if (dma == -1) { |
162 | res->dma_resource[i].flags |= IORESOURCE_DISABLED; | 163 | res->dma_resource[i].flags |= IORESOURCE_DISABLED; |
163 | return; | 164 | return; |
@@ -167,19 +168,19 @@ pnpacpi_parse_allocated_dmaresource(struct pnp_resource_table *res, u32 dma, | |||
167 | } | 168 | } |
168 | } | 169 | } |
169 | 170 | ||
170 | static void | 171 | static void pnpacpi_parse_allocated_ioresource(struct pnp_resource_table *res, |
171 | pnpacpi_parse_allocated_ioresource(struct pnp_resource_table *res, | 172 | u64 io, u64 len, int io_decode) |
172 | u64 io, u64 len, int io_decode) | ||
173 | { | 173 | { |
174 | int i = 0; | 174 | int i = 0; |
175 | |||
175 | while (!(res->port_resource[i].flags & IORESOURCE_UNSET) && | 176 | while (!(res->port_resource[i].flags & IORESOURCE_UNSET) && |
176 | i < PNP_MAX_PORT) | 177 | i < PNP_MAX_PORT) |
177 | i++; | 178 | i++; |
178 | if (i < PNP_MAX_PORT) { | 179 | if (i < PNP_MAX_PORT) { |
179 | res->port_resource[i].flags = IORESOURCE_IO; // Also clears _UNSET flag | 180 | res->port_resource[i].flags = IORESOURCE_IO; // Also clears _UNSET flag |
180 | if (io_decode == ACPI_DECODE_16) | 181 | if (io_decode == ACPI_DECODE_16) |
181 | res->port_resource[i].flags |= PNP_PORT_FLAG_16BITADDR; | 182 | res->port_resource[i].flags |= PNP_PORT_FLAG_16BITADDR; |
182 | if (len <= 0 || (io + len -1) >= 0x10003) { | 183 | if (len <= 0 || (io + len - 1) >= 0x10003) { |
183 | res->port_resource[i].flags |= IORESOURCE_DISABLED; | 184 | res->port_resource[i].flags |= IORESOURCE_DISABLED; |
184 | return; | 185 | return; |
185 | } | 186 | } |
@@ -188,21 +189,22 @@ pnpacpi_parse_allocated_ioresource(struct pnp_resource_table *res, | |||
188 | } | 189 | } |
189 | } | 190 | } |
190 | 191 | ||
191 | static void | 192 | static void pnpacpi_parse_allocated_memresource(struct pnp_resource_table *res, |
192 | pnpacpi_parse_allocated_memresource(struct pnp_resource_table *res, | 193 | u64 mem, u64 len, |
193 | u64 mem, u64 len, int write_protect) | 194 | int write_protect) |
194 | { | 195 | { |
195 | int i = 0; | 196 | int i = 0; |
197 | |||
196 | while (!(res->mem_resource[i].flags & IORESOURCE_UNSET) && | 198 | while (!(res->mem_resource[i].flags & IORESOURCE_UNSET) && |
197 | (i < PNP_MAX_MEM)) | 199 | (i < PNP_MAX_MEM)) |
198 | i++; | 200 | i++; |
199 | if (i < PNP_MAX_MEM) { | 201 | if (i < PNP_MAX_MEM) { |
200 | res->mem_resource[i].flags = IORESOURCE_MEM; // Also clears _UNSET flag | 202 | res->mem_resource[i].flags = IORESOURCE_MEM; // Also clears _UNSET flag |
201 | if (len <= 0) { | 203 | if (len <= 0) { |
202 | res->mem_resource[i].flags |= IORESOURCE_DISABLED; | 204 | res->mem_resource[i].flags |= IORESOURCE_DISABLED; |
203 | return; | 205 | return; |
204 | } | 206 | } |
205 | if(write_protect == ACPI_READ_WRITE_MEMORY) | 207 | if (write_protect == ACPI_READ_WRITE_MEMORY) |
206 | res->mem_resource[i].flags |= IORESOURCE_MEM_WRITEABLE; | 208 | res->mem_resource[i].flags |= IORESOURCE_MEM_WRITEABLE; |
207 | 209 | ||
208 | res->mem_resource[i].start = mem; | 210 | res->mem_resource[i].start = mem; |
@@ -210,9 +212,8 @@ pnpacpi_parse_allocated_memresource(struct pnp_resource_table *res, | |||
210 | } | 212 | } |
211 | } | 213 | } |
212 | 214 | ||
213 | static void | 215 | static void pnpacpi_parse_allocated_address_space(struct pnp_resource_table *res_table, |
214 | pnpacpi_parse_allocated_address_space(struct pnp_resource_table *res_table, | 216 | struct acpi_resource *res) |
215 | struct acpi_resource *res) | ||
216 | { | 217 | { |
217 | struct acpi_resource_address64 addr, *p = &addr; | 218 | struct acpi_resource_address64 addr, *p = &addr; |
218 | acpi_status status; | 219 | acpi_status status; |
@@ -220,7 +221,7 @@ pnpacpi_parse_allocated_address_space(struct pnp_resource_table *res_table, | |||
220 | status = acpi_resource_to_address64(res, p); | 221 | status = acpi_resource_to_address64(res, p); |
221 | if (!ACPI_SUCCESS(status)) { | 222 | if (!ACPI_SUCCESS(status)) { |
222 | pnp_warn("PnPACPI: failed to convert resource type %d", | 223 | pnp_warn("PnPACPI: failed to convert resource type %d", |
223 | res->type); | 224 | res->type); |
224 | return; | 225 | return; |
225 | } | 226 | } |
226 | 227 | ||
@@ -229,17 +230,20 @@ pnpacpi_parse_allocated_address_space(struct pnp_resource_table *res_table, | |||
229 | 230 | ||
230 | if (p->resource_type == ACPI_MEMORY_RANGE) | 231 | if (p->resource_type == ACPI_MEMORY_RANGE) |
231 | pnpacpi_parse_allocated_memresource(res_table, | 232 | pnpacpi_parse_allocated_memresource(res_table, |
232 | p->minimum, p->address_length, p->info.mem.write_protect); | 233 | p->minimum, p->address_length, |
234 | p->info.mem.write_protect); | ||
233 | else if (p->resource_type == ACPI_IO_RANGE) | 235 | else if (p->resource_type == ACPI_IO_RANGE) |
234 | pnpacpi_parse_allocated_ioresource(res_table, | 236 | pnpacpi_parse_allocated_ioresource(res_table, |
235 | p->minimum, p->address_length, | 237 | p->minimum, p->address_length, |
236 | p->granularity == 0xfff ? ACPI_DECODE_10 : ACPI_DECODE_16); | 238 | p->granularity == 0xfff ? ACPI_DECODE_10 : |
239 | ACPI_DECODE_16); | ||
237 | } | 240 | } |
238 | 241 | ||
239 | static acpi_status pnpacpi_allocated_resource(struct acpi_resource *res, | 242 | static acpi_status pnpacpi_allocated_resource(struct acpi_resource *res, |
240 | void *data) | 243 | void *data) |
241 | { | 244 | { |
242 | struct pnp_resource_table *res_table = (struct pnp_resource_table *)data; | 245 | struct pnp_resource_table *res_table = |
246 | (struct pnp_resource_table *)data; | ||
243 | int i; | 247 | int i; |
244 | 248 | ||
245 | switch (res->type) { | 249 | switch (res->type) { |
@@ -260,17 +264,17 @@ static acpi_status pnpacpi_allocated_resource(struct acpi_resource *res, | |||
260 | case ACPI_RESOURCE_TYPE_DMA: | 264 | case ACPI_RESOURCE_TYPE_DMA: |
261 | if (res->data.dma.channel_count > 0) | 265 | if (res->data.dma.channel_count > 0) |
262 | pnpacpi_parse_allocated_dmaresource(res_table, | 266 | pnpacpi_parse_allocated_dmaresource(res_table, |
263 | res->data.dma.channels[0], | 267 | res->data.dma.channels[0], |
264 | res->data.dma.type, | 268 | res->data.dma.type, |
265 | res->data.dma.bus_master, | 269 | res->data.dma.bus_master, |
266 | res->data.dma.transfer); | 270 | res->data.dma.transfer); |
267 | break; | 271 | break; |
268 | 272 | ||
269 | case ACPI_RESOURCE_TYPE_IO: | 273 | case ACPI_RESOURCE_TYPE_IO: |
270 | pnpacpi_parse_allocated_ioresource(res_table, | 274 | pnpacpi_parse_allocated_ioresource(res_table, |
271 | res->data.io.minimum, | 275 | res->data.io.minimum, |
272 | res->data.io.address_length, | 276 | res->data.io.address_length, |
273 | res->data.io.io_decode); | 277 | res->data.io.io_decode); |
274 | break; | 278 | break; |
275 | 279 | ||
276 | case ACPI_RESOURCE_TYPE_START_DEPENDENT: | 280 | case ACPI_RESOURCE_TYPE_START_DEPENDENT: |
@@ -279,9 +283,9 @@ static acpi_status pnpacpi_allocated_resource(struct acpi_resource *res, | |||
279 | 283 | ||
280 | case ACPI_RESOURCE_TYPE_FIXED_IO: | 284 | case ACPI_RESOURCE_TYPE_FIXED_IO: |
281 | pnpacpi_parse_allocated_ioresource(res_table, | 285 | pnpacpi_parse_allocated_ioresource(res_table, |
282 | res->data.fixed_io.address, | 286 | res->data.fixed_io.address, |
283 | res->data.fixed_io.address_length, | 287 | res->data.fixed_io.address_length, |
284 | ACPI_DECODE_10); | 288 | ACPI_DECODE_10); |
285 | break; | 289 | break; |
286 | 290 | ||
287 | case ACPI_RESOURCE_TYPE_VENDOR: | 291 | case ACPI_RESOURCE_TYPE_VENDOR: |
@@ -292,21 +296,21 @@ static acpi_status pnpacpi_allocated_resource(struct acpi_resource *res, | |||
292 | 296 | ||
293 | case ACPI_RESOURCE_TYPE_MEMORY24: | 297 | case ACPI_RESOURCE_TYPE_MEMORY24: |
294 | pnpacpi_parse_allocated_memresource(res_table, | 298 | pnpacpi_parse_allocated_memresource(res_table, |
295 | res->data.memory24.minimum, | 299 | res->data.memory24.minimum, |
296 | res->data.memory24.address_length, | 300 | res->data.memory24.address_length, |
297 | res->data.memory24.write_protect); | 301 | res->data.memory24.write_protect); |
298 | break; | 302 | break; |
299 | case ACPI_RESOURCE_TYPE_MEMORY32: | 303 | case ACPI_RESOURCE_TYPE_MEMORY32: |
300 | pnpacpi_parse_allocated_memresource(res_table, | 304 | pnpacpi_parse_allocated_memresource(res_table, |
301 | res->data.memory32.minimum, | 305 | res->data.memory32.minimum, |
302 | res->data.memory32.address_length, | 306 | res->data.memory32.address_length, |
303 | res->data.memory32.write_protect); | 307 | res->data.memory32.write_protect); |
304 | break; | 308 | break; |
305 | case ACPI_RESOURCE_TYPE_FIXED_MEMORY32: | 309 | case ACPI_RESOURCE_TYPE_FIXED_MEMORY32: |
306 | pnpacpi_parse_allocated_memresource(res_table, | 310 | pnpacpi_parse_allocated_memresource(res_table, |
307 | res->data.fixed_memory32.address, | 311 | res->data.fixed_memory32.address, |
308 | res->data.fixed_memory32.address_length, | 312 | res->data.fixed_memory32.address_length, |
309 | res->data.fixed_memory32.write_protect); | 313 | res->data.fixed_memory32.write_protect); |
310 | break; | 314 | break; |
311 | case ACPI_RESOURCE_TYPE_ADDRESS16: | 315 | case ACPI_RESOURCE_TYPE_ADDRESS16: |
312 | case ACPI_RESOURCE_TYPE_ADDRESS32: | 316 | case ACPI_RESOURCE_TYPE_ADDRESS32: |
@@ -343,18 +347,21 @@ static acpi_status pnpacpi_allocated_resource(struct acpi_resource *res, | |||
343 | return AE_OK; | 347 | return AE_OK; |
344 | } | 348 | } |
345 | 349 | ||
346 | acpi_status pnpacpi_parse_allocated_resource(acpi_handle handle, struct pnp_resource_table *res) | 350 | acpi_status pnpacpi_parse_allocated_resource(acpi_handle handle, |
351 | struct pnp_resource_table * res) | ||
347 | { | 352 | { |
348 | /* Blank the resource table values */ | 353 | /* Blank the resource table values */ |
349 | pnp_init_resource_table(res); | 354 | pnp_init_resource_table(res); |
350 | 355 | ||
351 | return acpi_walk_resources(handle, METHOD_NAME__CRS, pnpacpi_allocated_resource, res); | 356 | return acpi_walk_resources(handle, METHOD_NAME__CRS, |
357 | pnpacpi_allocated_resource, res); | ||
352 | } | 358 | } |
353 | 359 | ||
354 | static void pnpacpi_parse_dma_option(struct pnp_option *option, struct acpi_resource_dma *p) | 360 | static void pnpacpi_parse_dma_option(struct pnp_option *option, |
361 | struct acpi_resource_dma *p) | ||
355 | { | 362 | { |
356 | int i; | 363 | int i; |
357 | struct pnp_dma * dma; | 364 | struct pnp_dma *dma; |
358 | 365 | ||
359 | if (p->channel_count == 0) | 366 | if (p->channel_count == 0) |
360 | return; | 367 | return; |
@@ -362,18 +369,16 @@ static void pnpacpi_parse_dma_option(struct pnp_option *option, struct acpi_reso | |||
362 | if (!dma) | 369 | if (!dma) |
363 | return; | 370 | return; |
364 | 371 | ||
365 | for(i = 0; i < p->channel_count; i++) | 372 | for (i = 0; i < p->channel_count; i++) |
366 | dma->map |= 1 << p->channels[i]; | 373 | dma->map |= 1 << p->channels[i]; |
367 | 374 | ||
368 | dma->flags = dma_flags(p->type, p->bus_master, p->transfer); | 375 | dma->flags = dma_flags(p->type, p->bus_master, p->transfer); |
369 | 376 | ||
370 | pnp_register_dma_resource(option, dma); | 377 | pnp_register_dma_resource(option, dma); |
371 | return; | ||
372 | } | 378 | } |
373 | 379 | ||
374 | |||
375 | static void pnpacpi_parse_irq_option(struct pnp_option *option, | 380 | static void pnpacpi_parse_irq_option(struct pnp_option *option, |
376 | struct acpi_resource_irq *p) | 381 | struct acpi_resource_irq *p) |
377 | { | 382 | { |
378 | int i; | 383 | int i; |
379 | struct pnp_irq *irq; | 384 | struct pnp_irq *irq; |
@@ -384,17 +389,16 @@ static void pnpacpi_parse_irq_option(struct pnp_option *option, | |||
384 | if (!irq) | 389 | if (!irq) |
385 | return; | 390 | return; |
386 | 391 | ||
387 | for(i = 0; i < p->interrupt_count; i++) | 392 | for (i = 0; i < p->interrupt_count; i++) |
388 | if (p->interrupts[i]) | 393 | if (p->interrupts[i]) |
389 | __set_bit(p->interrupts[i], irq->map); | 394 | __set_bit(p->interrupts[i], irq->map); |
390 | irq->flags = irq_flags(p->triggering, p->polarity); | 395 | irq->flags = irq_flags(p->triggering, p->polarity); |
391 | 396 | ||
392 | pnp_register_irq_resource(option, irq); | 397 | pnp_register_irq_resource(option, irq); |
393 | return; | ||
394 | } | 398 | } |
395 | 399 | ||
396 | static void pnpacpi_parse_ext_irq_option(struct pnp_option *option, | 400 | static void pnpacpi_parse_ext_irq_option(struct pnp_option *option, |
397 | struct acpi_resource_extended_irq *p) | 401 | struct acpi_resource_extended_irq *p) |
398 | { | 402 | { |
399 | int i; | 403 | int i; |
400 | struct pnp_irq *irq; | 404 | struct pnp_irq *irq; |
@@ -405,18 +409,16 @@ static void pnpacpi_parse_ext_irq_option(struct pnp_option *option, | |||
405 | if (!irq) | 409 | if (!irq) |
406 | return; | 410 | return; |
407 | 411 | ||
408 | for(i = 0; i < p->interrupt_count; i++) | 412 | for (i = 0; i < p->interrupt_count; i++) |
409 | if (p->interrupts[i]) | 413 | if (p->interrupts[i]) |
410 | __set_bit(p->interrupts[i], irq->map); | 414 | __set_bit(p->interrupts[i], irq->map); |
411 | irq->flags = irq_flags(p->triggering, p->polarity); | 415 | irq->flags = irq_flags(p->triggering, p->polarity); |
412 | 416 | ||
413 | pnp_register_irq_resource(option, irq); | 417 | pnp_register_irq_resource(option, irq); |
414 | return; | ||
415 | } | 418 | } |
416 | 419 | ||
417 | static void | 420 | static void pnpacpi_parse_port_option(struct pnp_option *option, |
418 | pnpacpi_parse_port_option(struct pnp_option *option, | 421 | struct acpi_resource_io *io) |
419 | struct acpi_resource_io *io) | ||
420 | { | 422 | { |
421 | struct pnp_port *port; | 423 | struct pnp_port *port; |
422 | 424 | ||
@@ -430,14 +432,12 @@ pnpacpi_parse_port_option(struct pnp_option *option, | |||
430 | port->align = io->alignment; | 432 | port->align = io->alignment; |
431 | port->size = io->address_length; | 433 | port->size = io->address_length; |
432 | port->flags = ACPI_DECODE_16 == io->io_decode ? | 434 | port->flags = ACPI_DECODE_16 == io->io_decode ? |
433 | PNP_PORT_FLAG_16BITADDR : 0; | 435 | PNP_PORT_FLAG_16BITADDR : 0; |
434 | pnp_register_port_resource(option, port); | 436 | pnp_register_port_resource(option, port); |
435 | return; | ||
436 | } | 437 | } |
437 | 438 | ||
438 | static void | 439 | static void pnpacpi_parse_fixed_port_option(struct pnp_option *option, |
439 | pnpacpi_parse_fixed_port_option(struct pnp_option *option, | 440 | struct acpi_resource_fixed_io *io) |
440 | struct acpi_resource_fixed_io *io) | ||
441 | { | 441 | { |
442 | struct pnp_port *port; | 442 | struct pnp_port *port; |
443 | 443 | ||
@@ -451,12 +451,10 @@ pnpacpi_parse_fixed_port_option(struct pnp_option *option, | |||
451 | port->align = 0; | 451 | port->align = 0; |
452 | port->flags = PNP_PORT_FLAG_FIXED; | 452 | port->flags = PNP_PORT_FLAG_FIXED; |
453 | pnp_register_port_resource(option, port); | 453 | pnp_register_port_resource(option, port); |
454 | return; | ||
455 | } | 454 | } |
456 | 455 | ||
457 | static void | 456 | static void pnpacpi_parse_mem24_option(struct pnp_option *option, |
458 | pnpacpi_parse_mem24_option(struct pnp_option *option, | 457 | struct acpi_resource_memory24 *p) |
459 | struct acpi_resource_memory24 *p) | ||
460 | { | 458 | { |
461 | struct pnp_mem *mem; | 459 | struct pnp_mem *mem; |
462 | 460 | ||
@@ -471,15 +469,13 @@ pnpacpi_parse_mem24_option(struct pnp_option *option, | |||
471 | mem->size = p->address_length; | 469 | mem->size = p->address_length; |
472 | 470 | ||
473 | mem->flags = (ACPI_READ_WRITE_MEMORY == p->write_protect) ? | 471 | mem->flags = (ACPI_READ_WRITE_MEMORY == p->write_protect) ? |
474 | IORESOURCE_MEM_WRITEABLE : 0; | 472 | IORESOURCE_MEM_WRITEABLE : 0; |
475 | 473 | ||
476 | pnp_register_mem_resource(option, mem); | 474 | pnp_register_mem_resource(option, mem); |
477 | return; | ||
478 | } | 475 | } |
479 | 476 | ||
480 | static void | 477 | static void pnpacpi_parse_mem32_option(struct pnp_option *option, |
481 | pnpacpi_parse_mem32_option(struct pnp_option *option, | 478 | struct acpi_resource_memory32 *p) |
482 | struct acpi_resource_memory32 *p) | ||
483 | { | 479 | { |
484 | struct pnp_mem *mem; | 480 | struct pnp_mem *mem; |
485 | 481 | ||
@@ -494,15 +490,13 @@ pnpacpi_parse_mem32_option(struct pnp_option *option, | |||
494 | mem->size = p->address_length; | 490 | mem->size = p->address_length; |
495 | 491 | ||
496 | mem->flags = (ACPI_READ_WRITE_MEMORY == p->write_protect) ? | 492 | mem->flags = (ACPI_READ_WRITE_MEMORY == p->write_protect) ? |
497 | IORESOURCE_MEM_WRITEABLE : 0; | 493 | IORESOURCE_MEM_WRITEABLE : 0; |
498 | 494 | ||
499 | pnp_register_mem_resource(option, mem); | 495 | pnp_register_mem_resource(option, mem); |
500 | return; | ||
501 | } | 496 | } |
502 | 497 | ||
503 | static void | 498 | static void pnpacpi_parse_fixed_mem32_option(struct pnp_option *option, |
504 | pnpacpi_parse_fixed_mem32_option(struct pnp_option *option, | 499 | struct acpi_resource_fixed_memory32 *p) |
505 | struct acpi_resource_fixed_memory32 *p) | ||
506 | { | 500 | { |
507 | struct pnp_mem *mem; | 501 | struct pnp_mem *mem; |
508 | 502 | ||
@@ -516,14 +510,13 @@ pnpacpi_parse_fixed_mem32_option(struct pnp_option *option, | |||
516 | mem->align = 0; | 510 | mem->align = 0; |
517 | 511 | ||
518 | mem->flags = (ACPI_READ_WRITE_MEMORY == p->write_protect) ? | 512 | mem->flags = (ACPI_READ_WRITE_MEMORY == p->write_protect) ? |
519 | IORESOURCE_MEM_WRITEABLE : 0; | 513 | IORESOURCE_MEM_WRITEABLE : 0; |
520 | 514 | ||
521 | pnp_register_mem_resource(option, mem); | 515 | pnp_register_mem_resource(option, mem); |
522 | return; | ||
523 | } | 516 | } |
524 | 517 | ||
525 | static void | 518 | static void pnpacpi_parse_address_option(struct pnp_option *option, |
526 | pnpacpi_parse_address_option(struct pnp_option *option, struct acpi_resource *r) | 519 | struct acpi_resource *r) |
527 | { | 520 | { |
528 | struct acpi_resource_address64 addr, *p = &addr; | 521 | struct acpi_resource_address64 addr, *p = &addr; |
529 | acpi_status status; | 522 | acpi_status status; |
@@ -532,7 +525,8 @@ pnpacpi_parse_address_option(struct pnp_option *option, struct acpi_resource *r) | |||
532 | 525 | ||
533 | status = acpi_resource_to_address64(r, p); | 526 | status = acpi_resource_to_address64(r, p); |
534 | if (!ACPI_SUCCESS(status)) { | 527 | if (!ACPI_SUCCESS(status)) { |
535 | pnp_warn("PnPACPI: failed to convert resource type %d", r->type); | 528 | pnp_warn("PnPACPI: failed to convert resource type %d", |
529 | r->type); | ||
536 | return; | 530 | return; |
537 | } | 531 | } |
538 | 532 | ||
@@ -547,7 +541,8 @@ pnpacpi_parse_address_option(struct pnp_option *option, struct acpi_resource *r) | |||
547 | mem->size = p->address_length; | 541 | mem->size = p->address_length; |
548 | mem->align = 0; | 542 | mem->align = 0; |
549 | mem->flags = (p->info.mem.write_protect == | 543 | mem->flags = (p->info.mem.write_protect == |
550 | ACPI_READ_WRITE_MEMORY) ? IORESOURCE_MEM_WRITEABLE : 0; | 544 | ACPI_READ_WRITE_MEMORY) ? IORESOURCE_MEM_WRITEABLE |
545 | : 0; | ||
551 | pnp_register_mem_resource(option, mem); | 546 | pnp_register_mem_resource(option, mem); |
552 | } else if (p->resource_type == ACPI_IO_RANGE) { | 547 | } else if (p->resource_type == ACPI_IO_RANGE) { |
553 | port = kzalloc(sizeof(struct pnp_port), GFP_KERNEL); | 548 | port = kzalloc(sizeof(struct pnp_port), GFP_KERNEL); |
@@ -568,109 +563,108 @@ struct acpipnp_parse_option_s { | |||
568 | }; | 563 | }; |
569 | 564 | ||
570 | static acpi_status pnpacpi_option_resource(struct acpi_resource *res, | 565 | static acpi_status pnpacpi_option_resource(struct acpi_resource *res, |
571 | void *data) | 566 | void *data) |
572 | { | 567 | { |
573 | int priority = 0; | 568 | int priority = 0; |
574 | struct acpipnp_parse_option_s *parse_data = (struct acpipnp_parse_option_s *)data; | 569 | struct acpipnp_parse_option_s *parse_data = |
570 | (struct acpipnp_parse_option_s *)data; | ||
575 | struct pnp_dev *dev = parse_data->dev; | 571 | struct pnp_dev *dev = parse_data->dev; |
576 | struct pnp_option *option = parse_data->option; | 572 | struct pnp_option *option = parse_data->option; |
577 | 573 | ||
578 | switch (res->type) { | 574 | switch (res->type) { |
579 | case ACPI_RESOURCE_TYPE_IRQ: | 575 | case ACPI_RESOURCE_TYPE_IRQ: |
580 | pnpacpi_parse_irq_option(option, &res->data.irq); | 576 | pnpacpi_parse_irq_option(option, &res->data.irq); |
581 | break; | 577 | break; |
582 | 578 | ||
583 | case ACPI_RESOURCE_TYPE_DMA: | 579 | case ACPI_RESOURCE_TYPE_DMA: |
584 | pnpacpi_parse_dma_option(option, &res->data.dma); | 580 | pnpacpi_parse_dma_option(option, &res->data.dma); |
585 | break; | 581 | break; |
586 | 582 | ||
587 | case ACPI_RESOURCE_TYPE_START_DEPENDENT: | 583 | case ACPI_RESOURCE_TYPE_START_DEPENDENT: |
588 | switch (res->data.start_dpf.compatibility_priority) { | 584 | switch (res->data.start_dpf.compatibility_priority) { |
589 | case ACPI_GOOD_CONFIGURATION: | 585 | case ACPI_GOOD_CONFIGURATION: |
590 | priority = PNP_RES_PRIORITY_PREFERRED; | 586 | priority = PNP_RES_PRIORITY_PREFERRED; |
591 | break; | ||
592 | |||
593 | case ACPI_ACCEPTABLE_CONFIGURATION: | ||
594 | priority = PNP_RES_PRIORITY_ACCEPTABLE; | ||
595 | break; | ||
596 | |||
597 | case ACPI_SUB_OPTIMAL_CONFIGURATION: | ||
598 | priority = PNP_RES_PRIORITY_FUNCTIONAL; | ||
599 | break; | ||
600 | default: | ||
601 | priority = PNP_RES_PRIORITY_INVALID; | ||
602 | break; | ||
603 | } | ||
604 | /* TBD: Considering performace/robustness bits */ | ||
605 | option = pnp_register_dependent_option(dev, priority); | ||
606 | if (!option) | ||
607 | return AE_ERROR; | ||
608 | parse_data->option = option; | ||
609 | break; | 587 | break; |
610 | 588 | ||
611 | case ACPI_RESOURCE_TYPE_END_DEPENDENT: | 589 | case ACPI_ACCEPTABLE_CONFIGURATION: |
612 | /*only one EndDependentFn is allowed*/ | 590 | priority = PNP_RES_PRIORITY_ACCEPTABLE; |
613 | if (!parse_data->option_independent) { | ||
614 | pnp_warn("PnPACPI: more than one EndDependentFn"); | ||
615 | return AE_ERROR; | ||
616 | } | ||
617 | parse_data->option = parse_data->option_independent; | ||
618 | parse_data->option_independent = NULL; | ||
619 | break; | 591 | break; |
620 | 592 | ||
621 | case ACPI_RESOURCE_TYPE_IO: | 593 | case ACPI_SUB_OPTIMAL_CONFIGURATION: |
622 | pnpacpi_parse_port_option(option, &res->data.io); | 594 | priority = PNP_RES_PRIORITY_FUNCTIONAL; |
623 | break; | 595 | break; |
624 | 596 | default: | |
625 | case ACPI_RESOURCE_TYPE_FIXED_IO: | 597 | priority = PNP_RES_PRIORITY_INVALID; |
626 | pnpacpi_parse_fixed_port_option(option, | ||
627 | &res->data.fixed_io); | ||
628 | break; | 598 | break; |
599 | } | ||
600 | /* TBD: Consider performance/robustness bits */ | ||
601 | option = pnp_register_dependent_option(dev, priority); | ||
602 | if (!option) | ||
603 | return AE_ERROR; | ||
604 | parse_data->option = option; | ||
605 | break; | ||
629 | 606 | ||
630 | case ACPI_RESOURCE_TYPE_VENDOR: | 607 | case ACPI_RESOURCE_TYPE_END_DEPENDENT: |
631 | case ACPI_RESOURCE_TYPE_END_TAG: | 608 | /*only one EndDependentFn is allowed */ |
632 | break; | 609 | if (!parse_data->option_independent) { |
610 | pnp_warn("PnPACPI: more than one EndDependentFn"); | ||
611 | return AE_ERROR; | ||
612 | } | ||
613 | parse_data->option = parse_data->option_independent; | ||
614 | parse_data->option_independent = NULL; | ||
615 | break; | ||
633 | 616 | ||
634 | case ACPI_RESOURCE_TYPE_MEMORY24: | 617 | case ACPI_RESOURCE_TYPE_IO: |
635 | pnpacpi_parse_mem24_option(option, &res->data.memory24); | 618 | pnpacpi_parse_port_option(option, &res->data.io); |
636 | break; | 619 | break; |
637 | 620 | ||
638 | case ACPI_RESOURCE_TYPE_MEMORY32: | 621 | case ACPI_RESOURCE_TYPE_FIXED_IO: |
639 | pnpacpi_parse_mem32_option(option, &res->data.memory32); | 622 | pnpacpi_parse_fixed_port_option(option, &res->data.fixed_io); |
640 | break; | 623 | break; |
641 | 624 | ||
642 | case ACPI_RESOURCE_TYPE_FIXED_MEMORY32: | 625 | case ACPI_RESOURCE_TYPE_VENDOR: |
643 | pnpacpi_parse_fixed_mem32_option(option, | 626 | case ACPI_RESOURCE_TYPE_END_TAG: |
644 | &res->data.fixed_memory32); | 627 | break; |
645 | break; | ||
646 | 628 | ||
647 | case ACPI_RESOURCE_TYPE_ADDRESS16: | 629 | case ACPI_RESOURCE_TYPE_MEMORY24: |
648 | case ACPI_RESOURCE_TYPE_ADDRESS32: | 630 | pnpacpi_parse_mem24_option(option, &res->data.memory24); |
649 | case ACPI_RESOURCE_TYPE_ADDRESS64: | 631 | break; |
650 | pnpacpi_parse_address_option(option, res); | ||
651 | break; | ||
652 | 632 | ||
653 | case ACPI_RESOURCE_TYPE_EXTENDED_ADDRESS64: | 633 | case ACPI_RESOURCE_TYPE_MEMORY32: |
654 | break; | 634 | pnpacpi_parse_mem32_option(option, &res->data.memory32); |
635 | break; | ||
655 | 636 | ||
656 | case ACPI_RESOURCE_TYPE_EXTENDED_IRQ: | 637 | case ACPI_RESOURCE_TYPE_FIXED_MEMORY32: |
657 | pnpacpi_parse_ext_irq_option(option, | 638 | pnpacpi_parse_fixed_mem32_option(option, |
658 | &res->data.extended_irq); | 639 | &res->data.fixed_memory32); |
659 | break; | 640 | break; |
660 | 641 | ||
661 | case ACPI_RESOURCE_TYPE_GENERIC_REGISTER: | 642 | case ACPI_RESOURCE_TYPE_ADDRESS16: |
662 | break; | 643 | case ACPI_RESOURCE_TYPE_ADDRESS32: |
644 | case ACPI_RESOURCE_TYPE_ADDRESS64: | ||
645 | pnpacpi_parse_address_option(option, res); | ||
646 | break; | ||
663 | 647 | ||
664 | default: | 648 | case ACPI_RESOURCE_TYPE_EXTENDED_ADDRESS64: |
665 | pnp_warn("PnPACPI: unknown resource type %d", res->type); | 649 | break; |
666 | return AE_ERROR; | 650 | |
651 | case ACPI_RESOURCE_TYPE_EXTENDED_IRQ: | ||
652 | pnpacpi_parse_ext_irq_option(option, &res->data.extended_irq); | ||
653 | break; | ||
654 | |||
655 | case ACPI_RESOURCE_TYPE_GENERIC_REGISTER: | ||
656 | break; | ||
657 | |||
658 | default: | ||
659 | pnp_warn("PnPACPI: unknown resource type %d", res->type); | ||
660 | return AE_ERROR; | ||
667 | } | 661 | } |
668 | 662 | ||
669 | return AE_OK; | 663 | return AE_OK; |
670 | } | 664 | } |
671 | 665 | ||
672 | acpi_status pnpacpi_parse_resource_option_data(acpi_handle handle, | 666 | acpi_status pnpacpi_parse_resource_option_data(acpi_handle handle, |
673 | struct pnp_dev *dev) | 667 | struct pnp_dev * dev) |
674 | { | 668 | { |
675 | acpi_status status; | 669 | acpi_status status; |
676 | struct acpipnp_parse_option_s parse_data; | 670 | struct acpipnp_parse_option_s parse_data; |
@@ -681,7 +675,7 @@ acpi_status pnpacpi_parse_resource_option_data(acpi_handle handle, | |||
681 | parse_data.option_independent = parse_data.option; | 675 | parse_data.option_independent = parse_data.option; |
682 | parse_data.dev = dev; | 676 | parse_data.dev = dev; |
683 | status = acpi_walk_resources(handle, METHOD_NAME__PRS, | 677 | status = acpi_walk_resources(handle, METHOD_NAME__PRS, |
684 | pnpacpi_option_resource, &parse_data); | 678 | pnpacpi_option_resource, &parse_data); |
685 | 679 | ||
686 | return status; | 680 | return status; |
687 | } | 681 | } |
@@ -709,7 +703,7 @@ static int pnpacpi_supported_resource(struct acpi_resource *res) | |||
709 | * Set resource | 703 | * Set resource |
710 | */ | 704 | */ |
711 | static acpi_status pnpacpi_count_resources(struct acpi_resource *res, | 705 | static acpi_status pnpacpi_count_resources(struct acpi_resource *res, |
712 | void *data) | 706 | void *data) |
713 | { | 707 | { |
714 | int *res_cnt = (int *)data; | 708 | int *res_cnt = (int *)data; |
715 | 709 | ||
@@ -732,14 +726,14 @@ static acpi_status pnpacpi_type_resources(struct acpi_resource *res, void *data) | |||
732 | } | 726 | } |
733 | 727 | ||
734 | int pnpacpi_build_resource_template(acpi_handle handle, | 728 | int pnpacpi_build_resource_template(acpi_handle handle, |
735 | struct acpi_buffer *buffer) | 729 | struct acpi_buffer *buffer) |
736 | { | 730 | { |
737 | struct acpi_resource *resource; | 731 | struct acpi_resource *resource; |
738 | int res_cnt = 0; | 732 | int res_cnt = 0; |
739 | acpi_status status; | 733 | acpi_status status; |
740 | 734 | ||
741 | status = acpi_walk_resources(handle, METHOD_NAME__CRS, | 735 | status = acpi_walk_resources(handle, METHOD_NAME__CRS, |
742 | pnpacpi_count_resources, &res_cnt); | 736 | pnpacpi_count_resources, &res_cnt); |
743 | if (ACPI_FAILURE(status)) { | 737 | if (ACPI_FAILURE(status)) { |
744 | pnp_err("Evaluate _CRS failed"); | 738 | pnp_err("Evaluate _CRS failed"); |
745 | return -EINVAL; | 739 | return -EINVAL; |
@@ -753,7 +747,7 @@ int pnpacpi_build_resource_template(acpi_handle handle, | |||
753 | pnp_dbg("Res cnt %d", res_cnt); | 747 | pnp_dbg("Res cnt %d", res_cnt); |
754 | resource = (struct acpi_resource *)buffer->pointer; | 748 | resource = (struct acpi_resource *)buffer->pointer; |
755 | status = acpi_walk_resources(handle, METHOD_NAME__CRS, | 749 | status = acpi_walk_resources(handle, METHOD_NAME__CRS, |
756 | pnpacpi_type_resources, &resource); | 750 | pnpacpi_type_resources, &resource); |
757 | if (ACPI_FAILURE(status)) { | 751 | if (ACPI_FAILURE(status)) { |
758 | kfree(buffer->pointer); | 752 | kfree(buffer->pointer); |
759 | pnp_err("Evaluate _CRS failed"); | 753 | pnp_err("Evaluate _CRS failed"); |
@@ -766,7 +760,7 @@ int pnpacpi_build_resource_template(acpi_handle handle, | |||
766 | } | 760 | } |
767 | 761 | ||
768 | static void pnpacpi_encode_irq(struct acpi_resource *resource, | 762 | static void pnpacpi_encode_irq(struct acpi_resource *resource, |
769 | struct resource *p) | 763 | struct resource *p) |
770 | { | 764 | { |
771 | int triggering, polarity; | 765 | int triggering, polarity; |
772 | 766 | ||
@@ -782,7 +776,7 @@ static void pnpacpi_encode_irq(struct acpi_resource *resource, | |||
782 | } | 776 | } |
783 | 777 | ||
784 | static void pnpacpi_encode_ext_irq(struct acpi_resource *resource, | 778 | static void pnpacpi_encode_ext_irq(struct acpi_resource *resource, |
785 | struct resource *p) | 779 | struct resource *p) |
786 | { | 780 | { |
787 | int triggering, polarity; | 781 | int triggering, polarity; |
788 | 782 | ||
@@ -799,32 +793,32 @@ static void pnpacpi_encode_ext_irq(struct acpi_resource *resource, | |||
799 | } | 793 | } |
800 | 794 | ||
801 | static void pnpacpi_encode_dma(struct acpi_resource *resource, | 795 | static void pnpacpi_encode_dma(struct acpi_resource *resource, |
802 | struct resource *p) | 796 | struct resource *p) |
803 | { | 797 | { |
804 | /* Note: pnp_assign_dma will copy pnp_dma->flags into p->flags */ | 798 | /* Note: pnp_assign_dma will copy pnp_dma->flags into p->flags */ |
805 | switch (p->flags & IORESOURCE_DMA_SPEED_MASK) { | 799 | switch (p->flags & IORESOURCE_DMA_SPEED_MASK) { |
806 | case IORESOURCE_DMA_TYPEA: | 800 | case IORESOURCE_DMA_TYPEA: |
807 | resource->data.dma.type = ACPI_TYPE_A; | 801 | resource->data.dma.type = ACPI_TYPE_A; |
808 | break; | 802 | break; |
809 | case IORESOURCE_DMA_TYPEB: | 803 | case IORESOURCE_DMA_TYPEB: |
810 | resource->data.dma.type = ACPI_TYPE_B; | 804 | resource->data.dma.type = ACPI_TYPE_B; |
811 | break; | 805 | break; |
812 | case IORESOURCE_DMA_TYPEF: | 806 | case IORESOURCE_DMA_TYPEF: |
813 | resource->data.dma.type = ACPI_TYPE_F; | 807 | resource->data.dma.type = ACPI_TYPE_F; |
814 | break; | 808 | break; |
815 | default: | 809 | default: |
816 | resource->data.dma.type = ACPI_COMPATIBILITY; | 810 | resource->data.dma.type = ACPI_COMPATIBILITY; |
817 | } | 811 | } |
818 | 812 | ||
819 | switch (p->flags & IORESOURCE_DMA_TYPE_MASK) { | 813 | switch (p->flags & IORESOURCE_DMA_TYPE_MASK) { |
820 | case IORESOURCE_DMA_8BIT: | 814 | case IORESOURCE_DMA_8BIT: |
821 | resource->data.dma.transfer = ACPI_TRANSFER_8; | 815 | resource->data.dma.transfer = ACPI_TRANSFER_8; |
822 | break; | 816 | break; |
823 | case IORESOURCE_DMA_8AND16BIT: | 817 | case IORESOURCE_DMA_8AND16BIT: |
824 | resource->data.dma.transfer = ACPI_TRANSFER_8_16; | 818 | resource->data.dma.transfer = ACPI_TRANSFER_8_16; |
825 | break; | 819 | break; |
826 | default: | 820 | default: |
827 | resource->data.dma.transfer = ACPI_TRANSFER_16; | 821 | resource->data.dma.transfer = ACPI_TRANSFER_16; |
828 | } | 822 | } |
829 | 823 | ||
830 | resource->data.dma.bus_master = !!(p->flags & IORESOURCE_DMA_MASTER); | 824 | resource->data.dma.bus_master = !!(p->flags & IORESOURCE_DMA_MASTER); |
@@ -833,31 +827,31 @@ static void pnpacpi_encode_dma(struct acpi_resource *resource, | |||
833 | } | 827 | } |
834 | 828 | ||
835 | static void pnpacpi_encode_io(struct acpi_resource *resource, | 829 | static void pnpacpi_encode_io(struct acpi_resource *resource, |
836 | struct resource *p) | 830 | struct resource *p) |
837 | { | 831 | { |
838 | /* Note: pnp_assign_port will copy pnp_port->flags into p->flags */ | 832 | /* Note: pnp_assign_port will copy pnp_port->flags into p->flags */ |
839 | resource->data.io.io_decode = (p->flags & PNP_PORT_FLAG_16BITADDR)? | 833 | resource->data.io.io_decode = (p->flags & PNP_PORT_FLAG_16BITADDR) ? |
840 | ACPI_DECODE_16 : ACPI_DECODE_10; | 834 | ACPI_DECODE_16 : ACPI_DECODE_10; |
841 | resource->data.io.minimum = p->start; | 835 | resource->data.io.minimum = p->start; |
842 | resource->data.io.maximum = p->end; | 836 | resource->data.io.maximum = p->end; |
843 | resource->data.io.alignment = 0; /* Correct? */ | 837 | resource->data.io.alignment = 0; /* Correct? */ |
844 | resource->data.io.address_length = p->end - p->start + 1; | 838 | resource->data.io.address_length = p->end - p->start + 1; |
845 | } | 839 | } |
846 | 840 | ||
847 | static void pnpacpi_encode_fixed_io(struct acpi_resource *resource, | 841 | static void pnpacpi_encode_fixed_io(struct acpi_resource *resource, |
848 | struct resource *p) | 842 | struct resource *p) |
849 | { | 843 | { |
850 | resource->data.fixed_io.address = p->start; | 844 | resource->data.fixed_io.address = p->start; |
851 | resource->data.fixed_io.address_length = p->end - p->start + 1; | 845 | resource->data.fixed_io.address_length = p->end - p->start + 1; |
852 | } | 846 | } |
853 | 847 | ||
854 | static void pnpacpi_encode_mem24(struct acpi_resource *resource, | 848 | static void pnpacpi_encode_mem24(struct acpi_resource *resource, |
855 | struct resource *p) | 849 | struct resource *p) |
856 | { | 850 | { |
857 | /* Note: pnp_assign_mem will copy pnp_mem->flags into p->flags */ | 851 | /* Note: pnp_assign_mem will copy pnp_mem->flags into p->flags */ |
858 | resource->data.memory24.write_protect = | 852 | resource->data.memory24.write_protect = |
859 | (p->flags & IORESOURCE_MEM_WRITEABLE) ? | 853 | (p->flags & IORESOURCE_MEM_WRITEABLE) ? |
860 | ACPI_READ_WRITE_MEMORY : ACPI_READ_ONLY_MEMORY; | 854 | ACPI_READ_WRITE_MEMORY : ACPI_READ_ONLY_MEMORY; |
861 | resource->data.memory24.minimum = p->start; | 855 | resource->data.memory24.minimum = p->start; |
862 | resource->data.memory24.maximum = p->end; | 856 | resource->data.memory24.maximum = p->end; |
863 | resource->data.memory24.alignment = 0; | 857 | resource->data.memory24.alignment = 0; |
@@ -865,11 +859,11 @@ static void pnpacpi_encode_mem24(struct acpi_resource *resource, | |||
865 | } | 859 | } |
866 | 860 | ||
867 | static void pnpacpi_encode_mem32(struct acpi_resource *resource, | 861 | static void pnpacpi_encode_mem32(struct acpi_resource *resource, |
868 | struct resource *p) | 862 | struct resource *p) |
869 | { | 863 | { |
870 | resource->data.memory32.write_protect = | 864 | resource->data.memory32.write_protect = |
871 | (p->flags & IORESOURCE_MEM_WRITEABLE) ? | 865 | (p->flags & IORESOURCE_MEM_WRITEABLE) ? |
872 | ACPI_READ_WRITE_MEMORY : ACPI_READ_ONLY_MEMORY; | 866 | ACPI_READ_WRITE_MEMORY : ACPI_READ_ONLY_MEMORY; |
873 | resource->data.memory32.minimum = p->start; | 867 | resource->data.memory32.minimum = p->start; |
874 | resource->data.memory32.maximum = p->end; | 868 | resource->data.memory32.maximum = p->end; |
875 | resource->data.memory32.alignment = 0; | 869 | resource->data.memory32.alignment = 0; |
@@ -877,74 +871,77 @@ static void pnpacpi_encode_mem32(struct acpi_resource *resource, | |||
877 | } | 871 | } |
878 | 872 | ||
879 | static void pnpacpi_encode_fixed_mem32(struct acpi_resource *resource, | 873 | static void pnpacpi_encode_fixed_mem32(struct acpi_resource *resource, |
880 | struct resource *p) | 874 | struct resource *p) |
881 | { | 875 | { |
882 | resource->data.fixed_memory32.write_protect = | 876 | resource->data.fixed_memory32.write_protect = |
883 | (p->flags & IORESOURCE_MEM_WRITEABLE) ? | 877 | (p->flags & IORESOURCE_MEM_WRITEABLE) ? |
884 | ACPI_READ_WRITE_MEMORY : ACPI_READ_ONLY_MEMORY; | 878 | ACPI_READ_WRITE_MEMORY : ACPI_READ_ONLY_MEMORY; |
885 | resource->data.fixed_memory32.address = p->start; | 879 | resource->data.fixed_memory32.address = p->start; |
886 | resource->data.fixed_memory32.address_length = p->end - p->start + 1; | 880 | resource->data.fixed_memory32.address_length = p->end - p->start + 1; |
887 | } | 881 | } |
888 | 882 | ||
889 | int pnpacpi_encode_resources(struct pnp_resource_table *res_table, | 883 | int pnpacpi_encode_resources(struct pnp_resource_table *res_table, |
890 | struct acpi_buffer *buffer) | 884 | struct acpi_buffer *buffer) |
891 | { | 885 | { |
892 | int i = 0; | 886 | int i = 0; |
893 | /* pnpacpi_build_resource_template allocates extra mem */ | 887 | /* pnpacpi_build_resource_template allocates extra mem */ |
894 | int res_cnt = (buffer->length - 1)/sizeof(struct acpi_resource) - 1; | 888 | int res_cnt = (buffer->length - 1) / sizeof(struct acpi_resource) - 1; |
895 | struct acpi_resource *resource = (struct acpi_resource*)buffer->pointer; | 889 | struct acpi_resource *resource = |
890 | (struct acpi_resource *)buffer->pointer; | ||
896 | int port = 0, irq = 0, dma = 0, mem = 0; | 891 | int port = 0, irq = 0, dma = 0, mem = 0; |
897 | 892 | ||
898 | pnp_dbg("res cnt %d", res_cnt); | 893 | pnp_dbg("res cnt %d", res_cnt); |
899 | while (i < res_cnt) { | 894 | while (i < res_cnt) { |
900 | switch(resource->type) { | 895 | switch (resource->type) { |
901 | case ACPI_RESOURCE_TYPE_IRQ: | 896 | case ACPI_RESOURCE_TYPE_IRQ: |
902 | pnp_dbg("Encode irq"); | 897 | pnp_dbg("Encode irq"); |
903 | pnpacpi_encode_irq(resource, | 898 | pnpacpi_encode_irq(resource, |
904 | &res_table->irq_resource[irq]); | 899 | &res_table->irq_resource[irq]); |
905 | irq++; | 900 | irq++; |
906 | break; | 901 | break; |
907 | 902 | ||
908 | case ACPI_RESOURCE_TYPE_DMA: | 903 | case ACPI_RESOURCE_TYPE_DMA: |
909 | pnp_dbg("Encode dma"); | 904 | pnp_dbg("Encode dma"); |
910 | pnpacpi_encode_dma(resource, | 905 | pnpacpi_encode_dma(resource, |
911 | &res_table->dma_resource[dma]); | 906 | &res_table->dma_resource[dma]); |
912 | dma++; | 907 | dma++; |
913 | break; | 908 | break; |
914 | case ACPI_RESOURCE_TYPE_IO: | 909 | case ACPI_RESOURCE_TYPE_IO: |
915 | pnp_dbg("Encode io"); | 910 | pnp_dbg("Encode io"); |
916 | pnpacpi_encode_io(resource, | 911 | pnpacpi_encode_io(resource, |
917 | &res_table->port_resource[port]); | 912 | &res_table->port_resource[port]); |
918 | port++; | 913 | port++; |
919 | break; | 914 | break; |
920 | case ACPI_RESOURCE_TYPE_FIXED_IO: | 915 | case ACPI_RESOURCE_TYPE_FIXED_IO: |
921 | pnp_dbg("Encode fixed io"); | 916 | pnp_dbg("Encode fixed io"); |
922 | pnpacpi_encode_fixed_io(resource, | 917 | pnpacpi_encode_fixed_io(resource, |
923 | &res_table->port_resource[port]); | 918 | &res_table-> |
919 | port_resource[port]); | ||
924 | port++; | 920 | port++; |
925 | break; | 921 | break; |
926 | case ACPI_RESOURCE_TYPE_MEMORY24: | 922 | case ACPI_RESOURCE_TYPE_MEMORY24: |
927 | pnp_dbg("Encode mem24"); | 923 | pnp_dbg("Encode mem24"); |
928 | pnpacpi_encode_mem24(resource, | 924 | pnpacpi_encode_mem24(resource, |
929 | &res_table->mem_resource[mem]); | 925 | &res_table->mem_resource[mem]); |
930 | mem++; | 926 | mem++; |
931 | break; | 927 | break; |
932 | case ACPI_RESOURCE_TYPE_MEMORY32: | 928 | case ACPI_RESOURCE_TYPE_MEMORY32: |
933 | pnp_dbg("Encode mem32"); | 929 | pnp_dbg("Encode mem32"); |
934 | pnpacpi_encode_mem32(resource, | 930 | pnpacpi_encode_mem32(resource, |
935 | &res_table->mem_resource[mem]); | 931 | &res_table->mem_resource[mem]); |
936 | mem++; | 932 | mem++; |
937 | break; | 933 | break; |
938 | case ACPI_RESOURCE_TYPE_FIXED_MEMORY32: | 934 | case ACPI_RESOURCE_TYPE_FIXED_MEMORY32: |
939 | pnp_dbg("Encode fixed mem32"); | 935 | pnp_dbg("Encode fixed mem32"); |
940 | pnpacpi_encode_fixed_mem32(resource, | 936 | pnpacpi_encode_fixed_mem32(resource, |
941 | &res_table->mem_resource[mem]); | 937 | &res_table-> |
938 | mem_resource[mem]); | ||
942 | mem++; | 939 | mem++; |
943 | break; | 940 | break; |
944 | case ACPI_RESOURCE_TYPE_EXTENDED_IRQ: | 941 | case ACPI_RESOURCE_TYPE_EXTENDED_IRQ: |
945 | pnp_dbg("Encode ext irq"); | 942 | pnp_dbg("Encode ext irq"); |
946 | pnpacpi_encode_ext_irq(resource, | 943 | pnpacpi_encode_ext_irq(resource, |
947 | &res_table->irq_resource[irq]); | 944 | &res_table->irq_resource[irq]); |
948 | irq++; | 945 | irq++; |
949 | break; | 946 | break; |
950 | case ACPI_RESOURCE_TYPE_START_DEPENDENT: | 947 | case ACPI_RESOURCE_TYPE_START_DEPENDENT: |
@@ -956,7 +953,7 @@ int pnpacpi_encode_resources(struct pnp_resource_table *res_table, | |||
956 | case ACPI_RESOURCE_TYPE_ADDRESS64: | 953 | case ACPI_RESOURCE_TYPE_ADDRESS64: |
957 | case ACPI_RESOURCE_TYPE_EXTENDED_ADDRESS64: | 954 | case ACPI_RESOURCE_TYPE_EXTENDED_ADDRESS64: |
958 | case ACPI_RESOURCE_TYPE_GENERIC_REGISTER: | 955 | case ACPI_RESOURCE_TYPE_GENERIC_REGISTER: |
959 | default: /* other type */ | 956 | default: /* other type */ |
960 | pnp_warn("unknown resource type %d", resource->type); | 957 | pnp_warn("unknown resource type %d", resource->type); |
961 | return -EINVAL; | 958 | return -EINVAL; |
962 | } | 959 | } |
diff --git a/drivers/pnp/pnpbios/bioscalls.c b/drivers/pnp/pnpbios/bioscalls.c index a1f0b0ba2bfe..5dba68fe33f5 100644 --- a/drivers/pnp/pnpbios/bioscalls.c +++ b/drivers/pnp/pnpbios/bioscalls.c | |||
@@ -1,6 +1,5 @@ | |||
1 | /* | 1 | /* |
2 | * bioscalls.c - the lowlevel layer of the PnPBIOS driver | 2 | * bioscalls.c - the lowlevel layer of the PnPBIOS driver |
3 | * | ||
4 | */ | 3 | */ |
5 | 4 | ||
6 | #include <linux/types.h> | 5 | #include <linux/types.h> |
@@ -26,11 +25,10 @@ | |||
26 | #include "pnpbios.h" | 25 | #include "pnpbios.h" |
27 | 26 | ||
28 | static struct { | 27 | static struct { |
29 | u16 offset; | 28 | u16 offset; |
30 | u16 segment; | 29 | u16 segment; |
31 | } pnp_bios_callpoint; | 30 | } pnp_bios_callpoint; |
32 | 31 | ||
33 | |||
34 | /* | 32 | /* |
35 | * These are some opcodes for a "static asmlinkage" | 33 | * These are some opcodes for a "static asmlinkage" |
36 | * As this code is *not* executed inside the linux kernel segment, but in a | 34 | * As this code is *not* executed inside the linux kernel segment, but in a |
@@ -44,8 +42,7 @@ static struct { | |||
44 | 42 | ||
45 | asmlinkage void pnp_bios_callfunc(void); | 43 | asmlinkage void pnp_bios_callfunc(void); |
46 | 44 | ||
47 | __asm__( | 45 | __asm__(".text \n" |
48 | ".text \n" | ||
49 | __ALIGN_STR "\n" | 46 | __ALIGN_STR "\n" |
50 | "pnp_bios_callfunc:\n" | 47 | "pnp_bios_callfunc:\n" |
51 | " pushl %edx \n" | 48 | " pushl %edx \n" |
@@ -55,8 +52,7 @@ __asm__( | |||
55 | " lcallw *pnp_bios_callpoint\n" | 52 | " lcallw *pnp_bios_callpoint\n" |
56 | " addl $16, %esp \n" | 53 | " addl $16, %esp \n" |
57 | " lret \n" | 54 | " lret \n" |
58 | ".previous \n" | 55 | ".previous \n"); |
59 | ); | ||
60 | 56 | ||
61 | #define Q2_SET_SEL(cpu, selname, address, size) \ | 57 | #define Q2_SET_SEL(cpu, selname, address, size) \ |
62 | do { \ | 58 | do { \ |
@@ -78,7 +74,6 @@ u32 pnp_bios_is_utter_crap = 0; | |||
78 | 74 | ||
79 | static spinlock_t pnp_bios_lock; | 75 | static spinlock_t pnp_bios_lock; |
80 | 76 | ||
81 | |||
82 | /* | 77 | /* |
83 | * Support Functions | 78 | * Support Functions |
84 | */ | 79 | */ |
@@ -97,7 +92,7 @@ static inline u16 call_pnp_bios(u16 func, u16 arg1, u16 arg2, u16 arg3, | |||
97 | * PnP BIOSes are generally not terribly re-entrant. | 92 | * PnP BIOSes are generally not terribly re-entrant. |
98 | * Also, don't rely on them to save everything correctly. | 93 | * Also, don't rely on them to save everything correctly. |
99 | */ | 94 | */ |
100 | if(pnp_bios_is_utter_crap) | 95 | if (pnp_bios_is_utter_crap) |
101 | return PNP_FUNCTION_NOT_SUPPORTED; | 96 | return PNP_FUNCTION_NOT_SUPPORTED; |
102 | 97 | ||
103 | cpu = get_cpu(); | 98 | cpu = get_cpu(); |
@@ -113,112 +108,128 @@ static inline u16 call_pnp_bios(u16 func, u16 arg1, u16 arg2, u16 arg3, | |||
113 | if (ts2_size) | 108 | if (ts2_size) |
114 | Q2_SET_SEL(smp_processor_id(), PNP_TS2, ts2_base, ts2_size); | 109 | Q2_SET_SEL(smp_processor_id(), PNP_TS2, ts2_base, ts2_size); |
115 | 110 | ||
116 | __asm__ __volatile__( | 111 | __asm__ __volatile__("pushl %%ebp\n\t" |
117 | "pushl %%ebp\n\t" | 112 | "pushl %%edi\n\t" |
118 | "pushl %%edi\n\t" | 113 | "pushl %%esi\n\t" |
119 | "pushl %%esi\n\t" | 114 | "pushl %%ds\n\t" |
120 | "pushl %%ds\n\t" | 115 | "pushl %%es\n\t" |
121 | "pushl %%es\n\t" | 116 | "pushl %%fs\n\t" |
122 | "pushl %%fs\n\t" | 117 | "pushl %%gs\n\t" |
123 | "pushl %%gs\n\t" | 118 | "pushfl\n\t" |
124 | "pushfl\n\t" | 119 | "movl %%esp, pnp_bios_fault_esp\n\t" |
125 | "movl %%esp, pnp_bios_fault_esp\n\t" | 120 | "movl $1f, pnp_bios_fault_eip\n\t" |
126 | "movl $1f, pnp_bios_fault_eip\n\t" | 121 | "lcall %5,%6\n\t" |
127 | "lcall %5,%6\n\t" | 122 | "1:popfl\n\t" |
128 | "1:popfl\n\t" | 123 | "popl %%gs\n\t" |
129 | "popl %%gs\n\t" | 124 | "popl %%fs\n\t" |
130 | "popl %%fs\n\t" | 125 | "popl %%es\n\t" |
131 | "popl %%es\n\t" | 126 | "popl %%ds\n\t" |
132 | "popl %%ds\n\t" | 127 | "popl %%esi\n\t" |
133 | "popl %%esi\n\t" | 128 | "popl %%edi\n\t" |
134 | "popl %%edi\n\t" | 129 | "popl %%ebp\n\t":"=a"(status) |
135 | "popl %%ebp\n\t" | 130 | :"0"((func) | (((u32) arg1) << 16)), |
136 | : "=a" (status) | 131 | "b"((arg2) | (((u32) arg3) << 16)), |
137 | : "0" ((func) | (((u32)arg1) << 16)), | 132 | "c"((arg4) | (((u32) arg5) << 16)), |
138 | "b" ((arg2) | (((u32)arg3) << 16)), | 133 | "d"((arg6) | (((u32) arg7) << 16)), |
139 | "c" ((arg4) | (((u32)arg5) << 16)), | 134 | "i"(PNP_CS32), "i"(0) |
140 | "d" ((arg6) | (((u32)arg7) << 16)), | 135 | :"memory"); |
141 | "i" (PNP_CS32), | ||
142 | "i" (0) | ||
143 | : "memory" | ||
144 | ); | ||
145 | spin_unlock_irqrestore(&pnp_bios_lock, flags); | 136 | spin_unlock_irqrestore(&pnp_bios_lock, flags); |
146 | 137 | ||
147 | get_cpu_gdt_table(cpu)[0x40 / 8] = save_desc_40; | 138 | get_cpu_gdt_table(cpu)[0x40 / 8] = save_desc_40; |
148 | put_cpu(); | 139 | put_cpu(); |
149 | 140 | ||
150 | /* If we get here and this is set then the PnP BIOS faulted on us. */ | 141 | /* If we get here and this is set then the PnP BIOS faulted on us. */ |
151 | if(pnp_bios_is_utter_crap) | 142 | if (pnp_bios_is_utter_crap) { |
152 | { | 143 | printk(KERN_ERR |
153 | printk(KERN_ERR "PnPBIOS: Warning! Your PnP BIOS caused a fatal error. Attempting to continue\n"); | 144 | "PnPBIOS: Warning! Your PnP BIOS caused a fatal error. Attempting to continue\n"); |
154 | printk(KERN_ERR "PnPBIOS: You may need to reboot with the \"pnpbios=off\" option to operate stably\n"); | 145 | printk(KERN_ERR |
155 | printk(KERN_ERR "PnPBIOS: Check with your vendor for an updated BIOS\n"); | 146 | "PnPBIOS: You may need to reboot with the \"pnpbios=off\" option to operate stably\n"); |
147 | printk(KERN_ERR | ||
148 | "PnPBIOS: Check with your vendor for an updated BIOS\n"); | ||
156 | } | 149 | } |
157 | 150 | ||
158 | return status; | 151 | return status; |
159 | } | 152 | } |
160 | 153 | ||
161 | void pnpbios_print_status(const char * module, u16 status) | 154 | void pnpbios_print_status(const char *module, u16 status) |
162 | { | 155 | { |
163 | switch(status) { | 156 | switch (status) { |
164 | case PNP_SUCCESS: | 157 | case PNP_SUCCESS: |
165 | printk(KERN_ERR "PnPBIOS: %s: function successful\n", module); | 158 | printk(KERN_ERR "PnPBIOS: %s: function successful\n", module); |
166 | break; | 159 | break; |
167 | case PNP_NOT_SET_STATICALLY: | 160 | case PNP_NOT_SET_STATICALLY: |
168 | printk(KERN_ERR "PnPBIOS: %s: unable to set static resources\n", module); | 161 | printk(KERN_ERR "PnPBIOS: %s: unable to set static resources\n", |
162 | module); | ||
169 | break; | 163 | break; |
170 | case PNP_UNKNOWN_FUNCTION: | 164 | case PNP_UNKNOWN_FUNCTION: |
171 | printk(KERN_ERR "PnPBIOS: %s: invalid function number passed\n", module); | 165 | printk(KERN_ERR "PnPBIOS: %s: invalid function number passed\n", |
166 | module); | ||
172 | break; | 167 | break; |
173 | case PNP_FUNCTION_NOT_SUPPORTED: | 168 | case PNP_FUNCTION_NOT_SUPPORTED: |
174 | printk(KERN_ERR "PnPBIOS: %s: function not supported on this system\n", module); | 169 | printk(KERN_ERR |
170 | "PnPBIOS: %s: function not supported on this system\n", | ||
171 | module); | ||
175 | break; | 172 | break; |
176 | case PNP_INVALID_HANDLE: | 173 | case PNP_INVALID_HANDLE: |
177 | printk(KERN_ERR "PnPBIOS: %s: invalid handle\n", module); | 174 | printk(KERN_ERR "PnPBIOS: %s: invalid handle\n", module); |
178 | break; | 175 | break; |
179 | case PNP_BAD_PARAMETER: | 176 | case PNP_BAD_PARAMETER: |
180 | printk(KERN_ERR "PnPBIOS: %s: invalid parameters were passed\n", module); | 177 | printk(KERN_ERR "PnPBIOS: %s: invalid parameters were passed\n", |
178 | module); | ||
181 | break; | 179 | break; |
182 | case PNP_SET_FAILED: | 180 | case PNP_SET_FAILED: |
183 | printk(KERN_ERR "PnPBIOS: %s: unable to set resources\n", module); | 181 | printk(KERN_ERR "PnPBIOS: %s: unable to set resources\n", |
182 | module); | ||
184 | break; | 183 | break; |
185 | case PNP_EVENTS_NOT_PENDING: | 184 | case PNP_EVENTS_NOT_PENDING: |
186 | printk(KERN_ERR "PnPBIOS: %s: no events are pending\n", module); | 185 | printk(KERN_ERR "PnPBIOS: %s: no events are pending\n", module); |
187 | break; | 186 | break; |
188 | case PNP_SYSTEM_NOT_DOCKED: | 187 | case PNP_SYSTEM_NOT_DOCKED: |
189 | printk(KERN_ERR "PnPBIOS: %s: the system is not docked\n", module); | 188 | printk(KERN_ERR "PnPBIOS: %s: the system is not docked\n", |
189 | module); | ||
190 | break; | 190 | break; |
191 | case PNP_NO_ISA_PNP_CARDS: | 191 | case PNP_NO_ISA_PNP_CARDS: |
192 | printk(KERN_ERR "PnPBIOS: %s: no isapnp cards are installed on this system\n", module); | 192 | printk(KERN_ERR |
193 | "PnPBIOS: %s: no isapnp cards are installed on this system\n", | ||
194 | module); | ||
193 | break; | 195 | break; |
194 | case PNP_UNABLE_TO_DETERMINE_DOCK_CAPABILITIES: | 196 | case PNP_UNABLE_TO_DETERMINE_DOCK_CAPABILITIES: |
195 | printk(KERN_ERR "PnPBIOS: %s: cannot determine the capabilities of the docking station\n", module); | 197 | printk(KERN_ERR |
198 | "PnPBIOS: %s: cannot determine the capabilities of the docking station\n", | ||
199 | module); | ||
196 | break; | 200 | break; |
197 | case PNP_CONFIG_CHANGE_FAILED_NO_BATTERY: | 201 | case PNP_CONFIG_CHANGE_FAILED_NO_BATTERY: |
198 | printk(KERN_ERR "PnPBIOS: %s: unable to undock, the system does not have a battery\n", module); | 202 | printk(KERN_ERR |
203 | "PnPBIOS: %s: unable to undock, the system does not have a battery\n", | ||
204 | module); | ||
199 | break; | 205 | break; |
200 | case PNP_CONFIG_CHANGE_FAILED_RESOURCE_CONFLICT: | 206 | case PNP_CONFIG_CHANGE_FAILED_RESOURCE_CONFLICT: |
201 | printk(KERN_ERR "PnPBIOS: %s: could not dock due to resource conflicts\n", module); | 207 | printk(KERN_ERR |
208 | "PnPBIOS: %s: could not dock due to resource conflicts\n", | ||
209 | module); | ||
202 | break; | 210 | break; |
203 | case PNP_BUFFER_TOO_SMALL: | 211 | case PNP_BUFFER_TOO_SMALL: |
204 | printk(KERN_ERR "PnPBIOS: %s: the buffer passed is too small\n", module); | 212 | printk(KERN_ERR "PnPBIOS: %s: the buffer passed is too small\n", |
213 | module); | ||
205 | break; | 214 | break; |
206 | case PNP_USE_ESCD_SUPPORT: | 215 | case PNP_USE_ESCD_SUPPORT: |
207 | printk(KERN_ERR "PnPBIOS: %s: use ESCD instead\n", module); | 216 | printk(KERN_ERR "PnPBIOS: %s: use ESCD instead\n", module); |
208 | break; | 217 | break; |
209 | case PNP_MESSAGE_NOT_SUPPORTED: | 218 | case PNP_MESSAGE_NOT_SUPPORTED: |
210 | printk(KERN_ERR "PnPBIOS: %s: the message is unsupported\n", module); | 219 | printk(KERN_ERR "PnPBIOS: %s: the message is unsupported\n", |
220 | module); | ||
211 | break; | 221 | break; |
212 | case PNP_HARDWARE_ERROR: | 222 | case PNP_HARDWARE_ERROR: |
213 | printk(KERN_ERR "PnPBIOS: %s: a hardware failure has occured\n", module); | 223 | printk(KERN_ERR "PnPBIOS: %s: a hardware failure has occured\n", |
224 | module); | ||
214 | break; | 225 | break; |
215 | default: | 226 | default: |
216 | printk(KERN_ERR "PnPBIOS: %s: unexpected status 0x%x\n", module, status); | 227 | printk(KERN_ERR "PnPBIOS: %s: unexpected status 0x%x\n", module, |
228 | status); | ||
217 | break; | 229 | break; |
218 | } | 230 | } |
219 | } | 231 | } |
220 | 232 | ||
221 | |||
222 | /* | 233 | /* |
223 | * PnP BIOS Low Level Calls | 234 | * PnP BIOS Low Level Calls |
224 | */ | 235 | */ |
@@ -243,19 +254,22 @@ void pnpbios_print_status(const char * module, u16 status) | |||
243 | static int __pnp_bios_dev_node_info(struct pnp_dev_node_info *data) | 254 | static int __pnp_bios_dev_node_info(struct pnp_dev_node_info *data) |
244 | { | 255 | { |
245 | u16 status; | 256 | u16 status; |
257 | |||
246 | if (!pnp_bios_present()) | 258 | if (!pnp_bios_present()) |
247 | return PNP_FUNCTION_NOT_SUPPORTED; | 259 | return PNP_FUNCTION_NOT_SUPPORTED; |
248 | status = call_pnp_bios(PNP_GET_NUM_SYS_DEV_NODES, 0, PNP_TS1, 2, PNP_TS1, PNP_DS, 0, 0, | 260 | status = call_pnp_bios(PNP_GET_NUM_SYS_DEV_NODES, 0, PNP_TS1, 2, |
249 | data, sizeof(struct pnp_dev_node_info), NULL, 0); | 261 | PNP_TS1, PNP_DS, 0, 0, data, |
262 | sizeof(struct pnp_dev_node_info), NULL, 0); | ||
250 | data->no_nodes &= 0xff; | 263 | data->no_nodes &= 0xff; |
251 | return status; | 264 | return status; |
252 | } | 265 | } |
253 | 266 | ||
254 | int pnp_bios_dev_node_info(struct pnp_dev_node_info *data) | 267 | int pnp_bios_dev_node_info(struct pnp_dev_node_info *data) |
255 | { | 268 | { |
256 | int status = __pnp_bios_dev_node_info( data ); | 269 | int status = __pnp_bios_dev_node_info(data); |
257 | if ( status ) | 270 | |
258 | pnpbios_print_status( "dev_node_info", status ); | 271 | if (status) |
272 | pnpbios_print_status("dev_node_info", status); | ||
259 | return status; | 273 | return status; |
260 | } | 274 | } |
261 | 275 | ||
@@ -273,17 +287,20 @@ int pnp_bios_dev_node_info(struct pnp_dev_node_info *data) | |||
273 | * or volatile current (0) config | 287 | * or volatile current (0) config |
274 | * Output: *nodenum=next node or 0xff if no more nodes | 288 | * Output: *nodenum=next node or 0xff if no more nodes |
275 | */ | 289 | */ |
276 | static int __pnp_bios_get_dev_node(u8 *nodenum, char boot, struct pnp_bios_node *data) | 290 | static int __pnp_bios_get_dev_node(u8 *nodenum, char boot, |
291 | struct pnp_bios_node *data) | ||
277 | { | 292 | { |
278 | u16 status; | 293 | u16 status; |
279 | u16 tmp_nodenum; | 294 | u16 tmp_nodenum; |
295 | |||
280 | if (!pnp_bios_present()) | 296 | if (!pnp_bios_present()) |
281 | return PNP_FUNCTION_NOT_SUPPORTED; | 297 | return PNP_FUNCTION_NOT_SUPPORTED; |
282 | if ( !boot && pnpbios_dont_use_current_config ) | 298 | if (!boot && pnpbios_dont_use_current_config) |
283 | return PNP_FUNCTION_NOT_SUPPORTED; | 299 | return PNP_FUNCTION_NOT_SUPPORTED; |
284 | tmp_nodenum = *nodenum; | 300 | tmp_nodenum = *nodenum; |
285 | status = call_pnp_bios(PNP_GET_SYS_DEV_NODE, 0, PNP_TS1, 0, PNP_TS2, boot ? 2 : 1, PNP_DS, 0, | 301 | status = call_pnp_bios(PNP_GET_SYS_DEV_NODE, 0, PNP_TS1, 0, PNP_TS2, |
286 | &tmp_nodenum, sizeof(tmp_nodenum), data, 65536); | 302 | boot ? 2 : 1, PNP_DS, 0, &tmp_nodenum, |
303 | sizeof(tmp_nodenum), data, 65536); | ||
287 | *nodenum = tmp_nodenum; | 304 | *nodenum = tmp_nodenum; |
288 | return status; | 305 | return status; |
289 | } | 306 | } |
@@ -291,104 +308,66 @@ static int __pnp_bios_get_dev_node(u8 *nodenum, char boot, struct pnp_bios_node | |||
291 | int pnp_bios_get_dev_node(u8 *nodenum, char boot, struct pnp_bios_node *data) | 308 | int pnp_bios_get_dev_node(u8 *nodenum, char boot, struct pnp_bios_node *data) |
292 | { | 309 | { |
293 | int status; | 310 | int status; |
294 | status = __pnp_bios_get_dev_node( nodenum, boot, data ); | 311 | |
295 | if ( status ) | 312 | status = __pnp_bios_get_dev_node(nodenum, boot, data); |
296 | pnpbios_print_status( "get_dev_node", status ); | 313 | if (status) |
314 | pnpbios_print_status("get_dev_node", status); | ||
297 | return status; | 315 | return status; |
298 | } | 316 | } |
299 | 317 | ||
300 | |||
301 | /* | 318 | /* |
302 | * Call PnP BIOS with function 0x02, "set system device node" | 319 | * Call PnP BIOS with function 0x02, "set system device node" |
303 | * Input: *nodenum = desired node, | 320 | * Input: *nodenum = desired node, |
304 | * boot = whether to set nonvolatile boot (!=0) | 321 | * boot = whether to set nonvolatile boot (!=0) |
305 | * or volatile current (0) config | 322 | * or volatile current (0) config |
306 | */ | 323 | */ |
307 | static int __pnp_bios_set_dev_node(u8 nodenum, char boot, struct pnp_bios_node *data) | 324 | static int __pnp_bios_set_dev_node(u8 nodenum, char boot, |
325 | struct pnp_bios_node *data) | ||
308 | { | 326 | { |
309 | u16 status; | 327 | u16 status; |
328 | |||
310 | if (!pnp_bios_present()) | 329 | if (!pnp_bios_present()) |
311 | return PNP_FUNCTION_NOT_SUPPORTED; | 330 | return PNP_FUNCTION_NOT_SUPPORTED; |
312 | if ( !boot && pnpbios_dont_use_current_config ) | 331 | if (!boot && pnpbios_dont_use_current_config) |
313 | return PNP_FUNCTION_NOT_SUPPORTED; | 332 | return PNP_FUNCTION_NOT_SUPPORTED; |
314 | status = call_pnp_bios(PNP_SET_SYS_DEV_NODE, nodenum, 0, PNP_TS1, boot ? 2 : 1, PNP_DS, 0, 0, | 333 | status = call_pnp_bios(PNP_SET_SYS_DEV_NODE, nodenum, 0, PNP_TS1, |
315 | data, 65536, NULL, 0); | 334 | boot ? 2 : 1, PNP_DS, 0, 0, data, 65536, NULL, |
335 | 0); | ||
316 | return status; | 336 | return status; |
317 | } | 337 | } |
318 | 338 | ||
319 | int pnp_bios_set_dev_node(u8 nodenum, char boot, struct pnp_bios_node *data) | 339 | int pnp_bios_set_dev_node(u8 nodenum, char boot, struct pnp_bios_node *data) |
320 | { | 340 | { |
321 | int status; | 341 | int status; |
322 | status = __pnp_bios_set_dev_node( nodenum, boot, data ); | 342 | |
323 | if ( status ) { | 343 | status = __pnp_bios_set_dev_node(nodenum, boot, data); |
324 | pnpbios_print_status( "set_dev_node", status ); | 344 | if (status) { |
345 | pnpbios_print_status("set_dev_node", status); | ||
325 | return status; | 346 | return status; |
326 | } | 347 | } |
327 | if ( !boot ) { /* Update devlist */ | 348 | if (!boot) { /* Update devlist */ |
328 | status = pnp_bios_get_dev_node( &nodenum, boot, data ); | 349 | status = pnp_bios_get_dev_node(&nodenum, boot, data); |
329 | if ( status ) | 350 | if (status) |
330 | return status; | 351 | return status; |
331 | } | 352 | } |
332 | return status; | 353 | return status; |
333 | } | 354 | } |
334 | 355 | ||
335 | #if needed | ||
336 | /* | ||
337 | * Call PnP BIOS with function 0x03, "get event" | ||
338 | */ | ||
339 | static int pnp_bios_get_event(u16 *event) | ||
340 | { | ||
341 | u16 status; | ||
342 | if (!pnp_bios_present()) | ||
343 | return PNP_FUNCTION_NOT_SUPPORTED; | ||
344 | status = call_pnp_bios(PNP_GET_EVENT, 0, PNP_TS1, PNP_DS, 0, 0 ,0 ,0, | ||
345 | event, sizeof(u16), NULL, 0); | ||
346 | return status; | ||
347 | } | ||
348 | #endif | ||
349 | |||
350 | #if needed | ||
351 | /* | ||
352 | * Call PnP BIOS with function 0x04, "send message" | ||
353 | */ | ||
354 | static int pnp_bios_send_message(u16 message) | ||
355 | { | ||
356 | u16 status; | ||
357 | if (!pnp_bios_present()) | ||
358 | return PNP_FUNCTION_NOT_SUPPORTED; | ||
359 | status = call_pnp_bios(PNP_SEND_MESSAGE, message, PNP_DS, 0, 0, 0, 0, 0, 0, 0, 0, 0); | ||
360 | return status; | ||
361 | } | ||
362 | #endif | ||
363 | |||
364 | /* | 356 | /* |
365 | * Call PnP BIOS with function 0x05, "get docking station information" | 357 | * Call PnP BIOS with function 0x05, "get docking station information" |
366 | */ | 358 | */ |
367 | int pnp_bios_dock_station_info(struct pnp_docking_station_info *data) | 359 | int pnp_bios_dock_station_info(struct pnp_docking_station_info *data) |
368 | { | 360 | { |
369 | u16 status; | 361 | u16 status; |
370 | if (!pnp_bios_present()) | ||
371 | return PNP_FUNCTION_NOT_SUPPORTED; | ||
372 | status = call_pnp_bios(PNP_GET_DOCKING_STATION_INFORMATION, 0, PNP_TS1, PNP_DS, 0, 0, 0, 0, | ||
373 | data, sizeof(struct pnp_docking_station_info), NULL, 0); | ||
374 | return status; | ||
375 | } | ||
376 | 362 | ||
377 | #if needed | ||
378 | /* | ||
379 | * Call PnP BIOS with function 0x09, "set statically allocated resource | ||
380 | * information" | ||
381 | */ | ||
382 | static int pnp_bios_set_stat_res(char *info) | ||
383 | { | ||
384 | u16 status; | ||
385 | if (!pnp_bios_present()) | 363 | if (!pnp_bios_present()) |
386 | return PNP_FUNCTION_NOT_SUPPORTED; | 364 | return PNP_FUNCTION_NOT_SUPPORTED; |
387 | status = call_pnp_bios(PNP_SET_STATIC_ALLOCED_RES_INFO, 0, PNP_TS1, PNP_DS, 0, 0, 0, 0, | 365 | status = call_pnp_bios(PNP_GET_DOCKING_STATION_INFORMATION, 0, PNP_TS1, |
388 | info, *((u16 *) info), 0, 0); | 366 | PNP_DS, 0, 0, 0, 0, data, |
367 | sizeof(struct pnp_docking_station_info), NULL, | ||
368 | 0); | ||
389 | return status; | 369 | return status; |
390 | } | 370 | } |
391 | #endif | ||
392 | 371 | ||
393 | /* | 372 | /* |
394 | * Call PnP BIOS with function 0x0a, "get statically allocated resource | 373 | * Call PnP BIOS with function 0x0a, "get statically allocated resource |
@@ -397,36 +376,23 @@ static int pnp_bios_set_stat_res(char *info) | |||
397 | static int __pnp_bios_get_stat_res(char *info) | 376 | static int __pnp_bios_get_stat_res(char *info) |
398 | { | 377 | { |
399 | u16 status; | 378 | u16 status; |
379 | |||
400 | if (!pnp_bios_present()) | 380 | if (!pnp_bios_present()) |
401 | return PNP_FUNCTION_NOT_SUPPORTED; | 381 | return PNP_FUNCTION_NOT_SUPPORTED; |
402 | status = call_pnp_bios(PNP_GET_STATIC_ALLOCED_RES_INFO, 0, PNP_TS1, PNP_DS, 0, 0, 0, 0, | 382 | status = call_pnp_bios(PNP_GET_STATIC_ALLOCED_RES_INFO, 0, PNP_TS1, |
403 | info, 65536, NULL, 0); | 383 | PNP_DS, 0, 0, 0, 0, info, 65536, NULL, 0); |
404 | return status; | 384 | return status; |
405 | } | 385 | } |
406 | 386 | ||
407 | int pnp_bios_get_stat_res(char *info) | 387 | int pnp_bios_get_stat_res(char *info) |
408 | { | 388 | { |
409 | int status; | 389 | int status; |
410 | status = __pnp_bios_get_stat_res( info ); | ||
411 | if ( status ) | ||
412 | pnpbios_print_status( "get_stat_res", status ); | ||
413 | return status; | ||
414 | } | ||
415 | 390 | ||
416 | #if needed | 391 | status = __pnp_bios_get_stat_res(info); |
417 | /* | 392 | if (status) |
418 | * Call PnP BIOS with function 0x0b, "get APM id table" | 393 | pnpbios_print_status("get_stat_res", status); |
419 | */ | ||
420 | static int pnp_bios_apm_id_table(char *table, u16 *size) | ||
421 | { | ||
422 | u16 status; | ||
423 | if (!pnp_bios_present()) | ||
424 | return PNP_FUNCTION_NOT_SUPPORTED; | ||
425 | status = call_pnp_bios(PNP_GET_APM_ID_TABLE, 0, PNP_TS2, 0, PNP_TS1, PNP_DS, 0, 0, | ||
426 | table, *size, size, sizeof(u16)); | ||
427 | return status; | 394 | return status; |
428 | } | 395 | } |
429 | #endif | ||
430 | 396 | ||
431 | /* | 397 | /* |
432 | * Call PnP BIOS with function 0x40, "get isa pnp configuration structure" | 398 | * Call PnP BIOS with function 0x40, "get isa pnp configuration structure" |
@@ -434,19 +400,22 @@ static int pnp_bios_apm_id_table(char *table, u16 *size) | |||
434 | static int __pnp_bios_isapnp_config(struct pnp_isa_config_struc *data) | 400 | static int __pnp_bios_isapnp_config(struct pnp_isa_config_struc *data) |
435 | { | 401 | { |
436 | u16 status; | 402 | u16 status; |
403 | |||
437 | if (!pnp_bios_present()) | 404 | if (!pnp_bios_present()) |
438 | return PNP_FUNCTION_NOT_SUPPORTED; | 405 | return PNP_FUNCTION_NOT_SUPPORTED; |
439 | status = call_pnp_bios(PNP_GET_PNP_ISA_CONFIG_STRUC, 0, PNP_TS1, PNP_DS, 0, 0, 0, 0, | 406 | status = call_pnp_bios(PNP_GET_PNP_ISA_CONFIG_STRUC, 0, PNP_TS1, PNP_DS, |
440 | data, sizeof(struct pnp_isa_config_struc), NULL, 0); | 407 | 0, 0, 0, 0, data, |
408 | sizeof(struct pnp_isa_config_struc), NULL, 0); | ||
441 | return status; | 409 | return status; |
442 | } | 410 | } |
443 | 411 | ||
444 | int pnp_bios_isapnp_config(struct pnp_isa_config_struc *data) | 412 | int pnp_bios_isapnp_config(struct pnp_isa_config_struc *data) |
445 | { | 413 | { |
446 | int status; | 414 | int status; |
447 | status = __pnp_bios_isapnp_config( data ); | 415 | |
448 | if ( status ) | 416 | status = __pnp_bios_isapnp_config(data); |
449 | pnpbios_print_status( "isapnp_config", status ); | 417 | if (status) |
418 | pnpbios_print_status("isapnp_config", status); | ||
450 | return status; | 419 | return status; |
451 | } | 420 | } |
452 | 421 | ||
@@ -456,19 +425,22 @@ int pnp_bios_isapnp_config(struct pnp_isa_config_struc *data) | |||
456 | static int __pnp_bios_escd_info(struct escd_info_struc *data) | 425 | static int __pnp_bios_escd_info(struct escd_info_struc *data) |
457 | { | 426 | { |
458 | u16 status; | 427 | u16 status; |
428 | |||
459 | if (!pnp_bios_present()) | 429 | if (!pnp_bios_present()) |
460 | return ESCD_FUNCTION_NOT_SUPPORTED; | 430 | return ESCD_FUNCTION_NOT_SUPPORTED; |
461 | status = call_pnp_bios(PNP_GET_ESCD_INFO, 0, PNP_TS1, 2, PNP_TS1, 4, PNP_TS1, PNP_DS, | 431 | status = call_pnp_bios(PNP_GET_ESCD_INFO, 0, PNP_TS1, 2, PNP_TS1, 4, |
462 | data, sizeof(struct escd_info_struc), NULL, 0); | 432 | PNP_TS1, PNP_DS, data, |
433 | sizeof(struct escd_info_struc), NULL, 0); | ||
463 | return status; | 434 | return status; |
464 | } | 435 | } |
465 | 436 | ||
466 | int pnp_bios_escd_info(struct escd_info_struc *data) | 437 | int pnp_bios_escd_info(struct escd_info_struc *data) |
467 | { | 438 | { |
468 | int status; | 439 | int status; |
469 | status = __pnp_bios_escd_info( data ); | 440 | |
470 | if ( status ) | 441 | status = __pnp_bios_escd_info(data); |
471 | pnpbios_print_status( "escd_info", status ); | 442 | if (status) |
443 | pnpbios_print_status("escd_info", status); | ||
472 | return status; | 444 | return status; |
473 | } | 445 | } |
474 | 446 | ||
@@ -479,57 +451,42 @@ int pnp_bios_escd_info(struct escd_info_struc *data) | |||
479 | static int __pnp_bios_read_escd(char *data, u32 nvram_base) | 451 | static int __pnp_bios_read_escd(char *data, u32 nvram_base) |
480 | { | 452 | { |
481 | u16 status; | 453 | u16 status; |
454 | |||
482 | if (!pnp_bios_present()) | 455 | if (!pnp_bios_present()) |
483 | return ESCD_FUNCTION_NOT_SUPPORTED; | 456 | return ESCD_FUNCTION_NOT_SUPPORTED; |
484 | status = call_pnp_bios(PNP_READ_ESCD, 0, PNP_TS1, PNP_TS2, PNP_DS, 0, 0, 0, | 457 | status = call_pnp_bios(PNP_READ_ESCD, 0, PNP_TS1, PNP_TS2, PNP_DS, 0, 0, |
485 | data, 65536, __va(nvram_base), 65536); | 458 | 0, data, 65536, __va(nvram_base), 65536); |
486 | return status; | 459 | return status; |
487 | } | 460 | } |
488 | 461 | ||
489 | int pnp_bios_read_escd(char *data, u32 nvram_base) | 462 | int pnp_bios_read_escd(char *data, u32 nvram_base) |
490 | { | 463 | { |
491 | int status; | 464 | int status; |
492 | status = __pnp_bios_read_escd( data, nvram_base ); | ||
493 | if ( status ) | ||
494 | pnpbios_print_status( "read_escd", status ); | ||
495 | return status; | ||
496 | } | ||
497 | 465 | ||
498 | #if needed | 466 | status = __pnp_bios_read_escd(data, nvram_base); |
499 | /* | 467 | if (status) |
500 | * Call PnP BIOS function 0x43, "write ESCD" | 468 | pnpbios_print_status("read_escd", status); |
501 | */ | ||
502 | static int pnp_bios_write_escd(char *data, u32 nvram_base) | ||
503 | { | ||
504 | u16 status; | ||
505 | if (!pnp_bios_present()) | ||
506 | return ESCD_FUNCTION_NOT_SUPPORTED; | ||
507 | status = call_pnp_bios(PNP_WRITE_ESCD, 0, PNP_TS1, PNP_TS2, PNP_DS, 0, 0, 0, | ||
508 | data, 65536, __va(nvram_base), 65536); | ||
509 | return status; | 469 | return status; |
510 | } | 470 | } |
511 | #endif | ||
512 | |||
513 | |||
514 | /* | ||
515 | * Initialization | ||
516 | */ | ||
517 | 471 | ||
518 | void pnpbios_calls_init(union pnp_bios_install_struct *header) | 472 | void pnpbios_calls_init(union pnp_bios_install_struct *header) |
519 | { | 473 | { |
520 | int i; | 474 | int i; |
475 | |||
521 | spin_lock_init(&pnp_bios_lock); | 476 | spin_lock_init(&pnp_bios_lock); |
522 | pnp_bios_callpoint.offset = header->fields.pm16offset; | 477 | pnp_bios_callpoint.offset = header->fields.pm16offset; |
523 | pnp_bios_callpoint.segment = PNP_CS16; | 478 | pnp_bios_callpoint.segment = PNP_CS16; |
524 | 479 | ||
525 | set_base(bad_bios_desc, __va((unsigned long)0x40 << 4)); | 480 | set_base(bad_bios_desc, __va((unsigned long)0x40 << 4)); |
526 | _set_limit((char *)&bad_bios_desc, 4095 - (0x40 << 4)); | 481 | _set_limit((char *)&bad_bios_desc, 4095 - (0x40 << 4)); |
527 | for (i = 0; i < NR_CPUS; i++) { | 482 | for (i = 0; i < NR_CPUS; i++) { |
528 | struct desc_struct *gdt = get_cpu_gdt_table(i); | 483 | struct desc_struct *gdt = get_cpu_gdt_table(i); |
529 | if (!gdt) | 484 | if (!gdt) |
530 | continue; | 485 | continue; |
531 | set_base(gdt[GDT_ENTRY_PNPBIOS_CS32], &pnp_bios_callfunc); | 486 | set_base(gdt[GDT_ENTRY_PNPBIOS_CS32], &pnp_bios_callfunc); |
532 | set_base(gdt[GDT_ENTRY_PNPBIOS_CS16], __va(header->fields.pm16cseg)); | 487 | set_base(gdt[GDT_ENTRY_PNPBIOS_CS16], |
533 | set_base(gdt[GDT_ENTRY_PNPBIOS_DS], __va(header->fields.pm16dseg)); | 488 | __va(header->fields.pm16cseg)); |
534 | } | 489 | set_base(gdt[GDT_ENTRY_PNPBIOS_DS], |
490 | __va(header->fields.pm16dseg)); | ||
491 | } | ||
535 | } | 492 | } |
diff --git a/drivers/pnp/pnpbios/core.c b/drivers/pnp/pnpbios/core.c index ed112ee16012..3692a099b45f 100644 --- a/drivers/pnp/pnpbios/core.c +++ b/drivers/pnp/pnpbios/core.c | |||
@@ -32,7 +32,7 @@ | |||
32 | * along with this program; if not, write to the Free Software | 32 | * along with this program; if not, write to the Free Software |
33 | * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA | 33 | * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA |
34 | */ | 34 | */ |
35 | 35 | ||
36 | /* Change Log | 36 | /* Change Log |
37 | * | 37 | * |
38 | * Adam Belay - <ambx1@neo.rr.com> - March 16, 2003 | 38 | * Adam Belay - <ambx1@neo.rr.com> - March 16, 2003 |
@@ -71,14 +71,13 @@ | |||
71 | 71 | ||
72 | #include "pnpbios.h" | 72 | #include "pnpbios.h" |
73 | 73 | ||
74 | |||
75 | /* | 74 | /* |
76 | * | 75 | * |
77 | * PnP BIOS INTERFACE | 76 | * PnP BIOS INTERFACE |
78 | * | 77 | * |
79 | */ | 78 | */ |
80 | 79 | ||
81 | static union pnp_bios_install_struct * pnp_bios_install = NULL; | 80 | static union pnp_bios_install_struct *pnp_bios_install = NULL; |
82 | 81 | ||
83 | int pnp_bios_present(void) | 82 | int pnp_bios_present(void) |
84 | { | 83 | { |
@@ -101,36 +100,35 @@ static struct completion unload_sem; | |||
101 | /* | 100 | /* |
102 | * (Much of this belongs in a shared routine somewhere) | 101 | * (Much of this belongs in a shared routine somewhere) |
103 | */ | 102 | */ |
104 | |||
105 | static int pnp_dock_event(int dock, struct pnp_docking_station_info *info) | 103 | static int pnp_dock_event(int dock, struct pnp_docking_station_info *info) |
106 | { | 104 | { |
107 | char *argv [3], **envp, *buf, *scratch; | 105 | char *argv[3], **envp, *buf, *scratch; |
108 | int i = 0, value; | 106 | int i = 0, value; |
109 | 107 | ||
110 | if (!current->fs->root) { | 108 | if (!current->fs->root) |
111 | return -EAGAIN; | 109 | return -EAGAIN; |
112 | } | 110 | if (!(envp = kcalloc(20, sizeof(char *), GFP_KERNEL))) |
113 | if (!(envp = kcalloc(20, sizeof (char *), GFP_KERNEL))) { | ||
114 | return -ENOMEM; | 111 | return -ENOMEM; |
115 | } | ||
116 | if (!(buf = kzalloc(256, GFP_KERNEL))) { | 112 | if (!(buf = kzalloc(256, GFP_KERNEL))) { |
117 | kfree (envp); | 113 | kfree(envp); |
118 | return -ENOMEM; | 114 | return -ENOMEM; |
119 | } | 115 | } |
120 | 116 | ||
121 | /* FIXME: if there are actual users of this, it should be integrated into | 117 | /* FIXME: if there are actual users of this, it should be |
122 | * the driver core and use the usual infrastructure like sysfs and uevents */ | 118 | * integrated into the driver core and use the usual infrastructure |
123 | argv [0] = "/sbin/pnpbios"; | 119 | * like sysfs and uevents |
124 | argv [1] = "dock"; | 120 | */ |
125 | argv [2] = NULL; | 121 | argv[0] = "/sbin/pnpbios"; |
122 | argv[1] = "dock"; | ||
123 | argv[2] = NULL; | ||
126 | 124 | ||
127 | /* minimal command environment */ | 125 | /* minimal command environment */ |
128 | envp [i++] = "HOME=/"; | 126 | envp[i++] = "HOME=/"; |
129 | envp [i++] = "PATH=/sbin:/bin:/usr/sbin:/usr/bin"; | 127 | envp[i++] = "PATH=/sbin:/bin:/usr/sbin:/usr/bin"; |
130 | 128 | ||
131 | #ifdef DEBUG | 129 | #ifdef DEBUG |
132 | /* hint that policy agent should enter no-stdout debug mode */ | 130 | /* hint that policy agent should enter no-stdout debug mode */ |
133 | envp [i++] = "DEBUG=kernel"; | 131 | envp[i++] = "DEBUG=kernel"; |
134 | #endif | 132 | #endif |
135 | /* extensible set of named bus-specific parameters, | 133 | /* extensible set of named bus-specific parameters, |
136 | * supporting multiple driver selection algorithms. | 134 | * supporting multiple driver selection algorithms. |
@@ -138,33 +136,33 @@ static int pnp_dock_event(int dock, struct pnp_docking_station_info *info) | |||
138 | scratch = buf; | 136 | scratch = buf; |
139 | 137 | ||
140 | /* action: add, remove */ | 138 | /* action: add, remove */ |
141 | envp [i++] = scratch; | 139 | envp[i++] = scratch; |
142 | scratch += sprintf (scratch, "ACTION=%s", dock?"add":"remove") + 1; | 140 | scratch += sprintf(scratch, "ACTION=%s", dock ? "add" : "remove") + 1; |
143 | 141 | ||
144 | /* Report the ident for the dock */ | 142 | /* Report the ident for the dock */ |
145 | envp [i++] = scratch; | 143 | envp[i++] = scratch; |
146 | scratch += sprintf (scratch, "DOCK=%x/%x/%x", | 144 | scratch += sprintf(scratch, "DOCK=%x/%x/%x", |
147 | info->location_id, info->serial, info->capabilities); | 145 | info->location_id, info->serial, info->capabilities); |
148 | envp[i] = NULL; | 146 | envp[i] = NULL; |
149 | 147 | ||
150 | value = call_usermodehelper (argv [0], argv, envp, UMH_WAIT_EXEC); | 148 | value = call_usermodehelper(argv [0], argv, envp, UMH_WAIT_EXEC); |
151 | kfree (buf); | 149 | kfree(buf); |
152 | kfree (envp); | 150 | kfree(envp); |
153 | return 0; | 151 | return 0; |
154 | } | 152 | } |
155 | 153 | ||
156 | /* | 154 | /* |
157 | * Poll the PnP docking at regular intervals | 155 | * Poll the PnP docking at regular intervals |
158 | */ | 156 | */ |
159 | static int pnp_dock_thread(void * unused) | 157 | static int pnp_dock_thread(void *unused) |
160 | { | 158 | { |
161 | static struct pnp_docking_station_info now; | 159 | static struct pnp_docking_station_info now; |
162 | int docked = -1, d = 0; | 160 | int docked = -1, d = 0; |
161 | |||
163 | set_freezable(); | 162 | set_freezable(); |
164 | while (!unloading) | 163 | while (!unloading) { |
165 | { | ||
166 | int status; | 164 | int status; |
167 | 165 | ||
168 | /* | 166 | /* |
169 | * Poll every 2 seconds | 167 | * Poll every 2 seconds |
170 | */ | 168 | */ |
@@ -175,30 +173,29 @@ static int pnp_dock_thread(void * unused) | |||
175 | 173 | ||
176 | status = pnp_bios_dock_station_info(&now); | 174 | status = pnp_bios_dock_station_info(&now); |
177 | 175 | ||
178 | switch(status) | 176 | switch (status) { |
179 | { | ||
180 | /* | 177 | /* |
181 | * No dock to manage | 178 | * No dock to manage |
182 | */ | 179 | */ |
183 | case PNP_FUNCTION_NOT_SUPPORTED: | 180 | case PNP_FUNCTION_NOT_SUPPORTED: |
184 | complete_and_exit(&unload_sem, 0); | 181 | complete_and_exit(&unload_sem, 0); |
185 | case PNP_SYSTEM_NOT_DOCKED: | 182 | case PNP_SYSTEM_NOT_DOCKED: |
186 | d = 0; | 183 | d = 0; |
187 | break; | 184 | break; |
188 | case PNP_SUCCESS: | 185 | case PNP_SUCCESS: |
189 | d = 1; | 186 | d = 1; |
190 | break; | 187 | break; |
191 | default: | 188 | default: |
192 | pnpbios_print_status( "pnp_dock_thread", status ); | 189 | pnpbios_print_status("pnp_dock_thread", status); |
193 | continue; | 190 | continue; |
194 | } | 191 | } |
195 | if(d != docked) | 192 | if (d != docked) { |
196 | { | 193 | if (pnp_dock_event(d, &now) == 0) { |
197 | if(pnp_dock_event(d, &now)==0) | ||
198 | { | ||
199 | docked = d; | 194 | docked = d; |
200 | #if 0 | 195 | #if 0 |
201 | printk(KERN_INFO "PnPBIOS: Docking station %stached\n", docked?"at":"de"); | 196 | printk(KERN_INFO |
197 | "PnPBIOS: Docking station %stached\n", | ||
198 | docked ? "at" : "de"); | ||
202 | #endif | 199 | #endif |
203 | } | 200 | } |
204 | } | 201 | } |
@@ -206,21 +203,21 @@ static int pnp_dock_thread(void * unused) | |||
206 | complete_and_exit(&unload_sem, 0); | 203 | complete_and_exit(&unload_sem, 0); |
207 | } | 204 | } |
208 | 205 | ||
209 | #endif /* CONFIG_HOTPLUG */ | 206 | #endif /* CONFIG_HOTPLUG */ |
210 | 207 | ||
211 | static int pnpbios_get_resources(struct pnp_dev * dev, struct pnp_resource_table * res) | 208 | static int pnpbios_get_resources(struct pnp_dev *dev, |
209 | struct pnp_resource_table *res) | ||
212 | { | 210 | { |
213 | u8 nodenum = dev->number; | 211 | u8 nodenum = dev->number; |
214 | struct pnp_bios_node * node; | 212 | struct pnp_bios_node *node; |
215 | 213 | ||
216 | /* just in case */ | 214 | if (!pnpbios_is_dynamic(dev)) |
217 | if(!pnpbios_is_dynamic(dev)) | ||
218 | return -EPERM; | 215 | return -EPERM; |
219 | 216 | ||
220 | node = kzalloc(node_info.max_node_size, GFP_KERNEL); | 217 | node = kzalloc(node_info.max_node_size, GFP_KERNEL); |
221 | if (!node) | 218 | if (!node) |
222 | return -1; | 219 | return -1; |
223 | if (pnp_bios_get_dev_node(&nodenum, (char )PNPMODE_DYNAMIC, node)) { | 220 | if (pnp_bios_get_dev_node(&nodenum, (char)PNPMODE_DYNAMIC, node)) { |
224 | kfree(node); | 221 | kfree(node); |
225 | return -ENODEV; | 222 | return -ENODEV; |
226 | } | 223 | } |
@@ -230,24 +227,24 @@ static int pnpbios_get_resources(struct pnp_dev * dev, struct pnp_resource_table | |||
230 | return 0; | 227 | return 0; |
231 | } | 228 | } |
232 | 229 | ||
233 | static int pnpbios_set_resources(struct pnp_dev * dev, struct pnp_resource_table * res) | 230 | static int pnpbios_set_resources(struct pnp_dev *dev, |
231 | struct pnp_resource_table *res) | ||
234 | { | 232 | { |
235 | u8 nodenum = dev->number; | 233 | u8 nodenum = dev->number; |
236 | struct pnp_bios_node * node; | 234 | struct pnp_bios_node *node; |
237 | int ret; | 235 | int ret; |
238 | 236 | ||
239 | /* just in case */ | ||
240 | if (!pnpbios_is_dynamic(dev)) | 237 | if (!pnpbios_is_dynamic(dev)) |
241 | return -EPERM; | 238 | return -EPERM; |
242 | 239 | ||
243 | node = kzalloc(node_info.max_node_size, GFP_KERNEL); | 240 | node = kzalloc(node_info.max_node_size, GFP_KERNEL); |
244 | if (!node) | 241 | if (!node) |
245 | return -1; | 242 | return -1; |
246 | if (pnp_bios_get_dev_node(&nodenum, (char )PNPMODE_DYNAMIC, node)) { | 243 | if (pnp_bios_get_dev_node(&nodenum, (char)PNPMODE_DYNAMIC, node)) { |
247 | kfree(node); | 244 | kfree(node); |
248 | return -ENODEV; | 245 | return -ENODEV; |
249 | } | 246 | } |
250 | if(pnpbios_write_resources_to_node(res, node)<0) { | 247 | if (pnpbios_write_resources_to_node(res, node) < 0) { |
251 | kfree(node); | 248 | kfree(node); |
252 | return -1; | 249 | return -1; |
253 | } | 250 | } |
@@ -258,18 +255,19 @@ static int pnpbios_set_resources(struct pnp_dev * dev, struct pnp_resource_table | |||
258 | return ret; | 255 | return ret; |
259 | } | 256 | } |
260 | 257 | ||
261 | static void pnpbios_zero_data_stream(struct pnp_bios_node * node) | 258 | static void pnpbios_zero_data_stream(struct pnp_bios_node *node) |
262 | { | 259 | { |
263 | unsigned char * p = (char *)node->data; | 260 | unsigned char *p = (char *)node->data; |
264 | unsigned char * end = (char *)(node->data + node->size); | 261 | unsigned char *end = (char *)(node->data + node->size); |
265 | unsigned int len; | 262 | unsigned int len; |
266 | int i; | 263 | int i; |
264 | |||
267 | while ((char *)p < (char *)end) { | 265 | while ((char *)p < (char *)end) { |
268 | if(p[0] & 0x80) { /* large tag */ | 266 | if (p[0] & 0x80) { /* large tag */ |
269 | len = (p[2] << 8) | p[1]; | 267 | len = (p[2] << 8) | p[1]; |
270 | p += 3; | 268 | p += 3; |
271 | } else { | 269 | } else { |
272 | if (((p[0]>>3) & 0x0f) == 0x0f) | 270 | if (((p[0] >> 3) & 0x0f) == 0x0f) |
273 | return; | 271 | return; |
274 | len = p[0] & 0x07; | 272 | len = p[0] & 0x07; |
275 | p += 1; | 273 | p += 1; |
@@ -278,24 +276,24 @@ static void pnpbios_zero_data_stream(struct pnp_bios_node * node) | |||
278 | p[i] = 0; | 276 | p[i] = 0; |
279 | p += len; | 277 | p += len; |
280 | } | 278 | } |
281 | printk(KERN_ERR "PnPBIOS: Resource structure did not contain an end tag.\n"); | 279 | printk(KERN_ERR |
280 | "PnPBIOS: Resource structure did not contain an end tag.\n"); | ||
282 | } | 281 | } |
283 | 282 | ||
284 | static int pnpbios_disable_resources(struct pnp_dev *dev) | 283 | static int pnpbios_disable_resources(struct pnp_dev *dev) |
285 | { | 284 | { |
286 | struct pnp_bios_node * node; | 285 | struct pnp_bios_node *node; |
287 | u8 nodenum = dev->number; | 286 | u8 nodenum = dev->number; |
288 | int ret; | 287 | int ret; |
289 | 288 | ||
290 | /* just in case */ | 289 | if (dev->flags & PNPBIOS_NO_DISABLE || !pnpbios_is_dynamic(dev)) |
291 | if(dev->flags & PNPBIOS_NO_DISABLE || !pnpbios_is_dynamic(dev)) | ||
292 | return -EPERM; | 290 | return -EPERM; |
293 | 291 | ||
294 | node = kzalloc(node_info.max_node_size, GFP_KERNEL); | 292 | node = kzalloc(node_info.max_node_size, GFP_KERNEL); |
295 | if (!node) | 293 | if (!node) |
296 | return -ENOMEM; | 294 | return -ENOMEM; |
297 | 295 | ||
298 | if (pnp_bios_get_dev_node(&nodenum, (char )PNPMODE_DYNAMIC, node)) { | 296 | if (pnp_bios_get_dev_node(&nodenum, (char)PNPMODE_DYNAMIC, node)) { |
299 | kfree(node); | 297 | kfree(node); |
300 | return -ENODEV; | 298 | return -ENODEV; |
301 | } | 299 | } |
@@ -311,22 +309,22 @@ static int pnpbios_disable_resources(struct pnp_dev *dev) | |||
311 | /* PnP Layer support */ | 309 | /* PnP Layer support */ |
312 | 310 | ||
313 | struct pnp_protocol pnpbios_protocol = { | 311 | struct pnp_protocol pnpbios_protocol = { |
314 | .name = "Plug and Play BIOS", | 312 | .name = "Plug and Play BIOS", |
315 | .get = pnpbios_get_resources, | 313 | .get = pnpbios_get_resources, |
316 | .set = pnpbios_set_resources, | 314 | .set = pnpbios_set_resources, |
317 | .disable = pnpbios_disable_resources, | 315 | .disable = pnpbios_disable_resources, |
318 | }; | 316 | }; |
319 | 317 | ||
320 | static int insert_device(struct pnp_dev *dev, struct pnp_bios_node * node) | 318 | static int insert_device(struct pnp_dev *dev, struct pnp_bios_node *node) |
321 | { | 319 | { |
322 | struct list_head * pos; | 320 | struct list_head *pos; |
323 | struct pnp_dev * pnp_dev; | 321 | struct pnp_dev *pnp_dev; |
324 | struct pnp_id *dev_id; | 322 | struct pnp_id *dev_id; |
325 | char id[8]; | 323 | char id[8]; |
326 | 324 | ||
327 | /* check if the device is already added */ | 325 | /* check if the device is already added */ |
328 | dev->number = node->handle; | 326 | dev->number = node->handle; |
329 | list_for_each (pos, &pnpbios_protocol.devices){ | 327 | list_for_each(pos, &pnpbios_protocol.devices) { |
330 | pnp_dev = list_entry(pos, struct pnp_dev, protocol_list); | 328 | pnp_dev = list_entry(pos, struct pnp_dev, protocol_list); |
331 | if (dev->number == pnp_dev->number) | 329 | if (dev->number == pnp_dev->number) |
332 | return -1; | 330 | return -1; |
@@ -336,8 +334,8 @@ static int insert_device(struct pnp_dev *dev, struct pnp_bios_node * node) | |||
336 | dev_id = kzalloc(sizeof(struct pnp_id), GFP_KERNEL); | 334 | dev_id = kzalloc(sizeof(struct pnp_id), GFP_KERNEL); |
337 | if (!dev_id) | 335 | if (!dev_id) |
338 | return -1; | 336 | return -1; |
339 | pnpid32_to_pnpid(node->eisa_id,id); | 337 | pnpid32_to_pnpid(node->eisa_id, id); |
340 | memcpy(dev_id->id,id,7); | 338 | memcpy(dev_id->id, id, 7); |
341 | pnp_add_id(dev_id, dev); | 339 | pnp_add_id(dev_id, dev); |
342 | pnpbios_parse_data_stream(dev, node); | 340 | pnpbios_parse_data_stream(dev, node); |
343 | dev->active = pnp_is_active(dev); | 341 | dev->active = pnp_is_active(dev); |
@@ -375,35 +373,41 @@ static void __init build_devlist(void) | |||
375 | if (!node) | 373 | if (!node) |
376 | return; | 374 | return; |
377 | 375 | ||
378 | for(nodenum=0; nodenum<0xff; ) { | 376 | for (nodenum = 0; nodenum < 0xff;) { |
379 | u8 thisnodenum = nodenum; | 377 | u8 thisnodenum = nodenum; |
380 | /* eventually we will want to use PNPMODE_STATIC here but for now | 378 | /* eventually we will want to use PNPMODE_STATIC here but for now |
381 | * dynamic will help us catch buggy bioses to add to the blacklist. | 379 | * dynamic will help us catch buggy bioses to add to the blacklist. |
382 | */ | 380 | */ |
383 | if (!pnpbios_dont_use_current_config) { | 381 | if (!pnpbios_dont_use_current_config) { |
384 | if (pnp_bios_get_dev_node(&nodenum, (char )PNPMODE_DYNAMIC, node)) | 382 | if (pnp_bios_get_dev_node |
383 | (&nodenum, (char)PNPMODE_DYNAMIC, node)) | ||
385 | break; | 384 | break; |
386 | } else { | 385 | } else { |
387 | if (pnp_bios_get_dev_node(&nodenum, (char )PNPMODE_STATIC, node)) | 386 | if (pnp_bios_get_dev_node |
387 | (&nodenum, (char)PNPMODE_STATIC, node)) | ||
388 | break; | 388 | break; |
389 | } | 389 | } |
390 | nodes_got++; | 390 | nodes_got++; |
391 | dev = kzalloc(sizeof (struct pnp_dev), GFP_KERNEL); | 391 | dev = kzalloc(sizeof(struct pnp_dev), GFP_KERNEL); |
392 | if (!dev) | 392 | if (!dev) |
393 | break; | 393 | break; |
394 | if(insert_device(dev,node)<0) | 394 | if (insert_device(dev, node) < 0) |
395 | kfree(dev); | 395 | kfree(dev); |
396 | else | 396 | else |
397 | devs++; | 397 | devs++; |
398 | if (nodenum <= thisnodenum) { | 398 | if (nodenum <= thisnodenum) { |
399 | printk(KERN_ERR "PnPBIOS: build_devlist: Node number 0x%x is out of sequence following node 0x%x. Aborting.\n", (unsigned int)nodenum, (unsigned int)thisnodenum); | 399 | printk(KERN_ERR |
400 | "PnPBIOS: build_devlist: Node number 0x%x is out of sequence following node 0x%x. Aborting.\n", | ||
401 | (unsigned int)nodenum, | ||
402 | (unsigned int)thisnodenum); | ||
400 | break; | 403 | break; |
401 | } | 404 | } |
402 | } | 405 | } |
403 | kfree(node); | 406 | kfree(node); |
404 | 407 | ||
405 | printk(KERN_INFO "PnPBIOS: %i node%s reported by PnP BIOS; %i recorded by driver\n", | 408 | printk(KERN_INFO |
406 | nodes_got, nodes_got != 1 ? "s" : "", devs); | 409 | "PnPBIOS: %i node%s reported by PnP BIOS; %i recorded by driver\n", |
410 | nodes_got, nodes_got != 1 ? "s" : "", devs); | ||
407 | } | 411 | } |
408 | 412 | ||
409 | /* | 413 | /* |
@@ -412,8 +416,8 @@ static void __init build_devlist(void) | |||
412 | * | 416 | * |
413 | */ | 417 | */ |
414 | 418 | ||
415 | static int pnpbios_disabled; /* = 0 */ | 419 | static int pnpbios_disabled; |
416 | int pnpbios_dont_use_current_config; /* = 0 */ | 420 | int pnpbios_dont_use_current_config; |
417 | 421 | ||
418 | #ifndef MODULE | 422 | #ifndef MODULE |
419 | static int __init pnpbios_setup(char *str) | 423 | static int __init pnpbios_setup(char *str) |
@@ -422,9 +426,9 @@ static int __init pnpbios_setup(char *str) | |||
422 | 426 | ||
423 | while ((str != NULL) && (*str != '\0')) { | 427 | while ((str != NULL) && (*str != '\0')) { |
424 | if (strncmp(str, "off", 3) == 0) | 428 | if (strncmp(str, "off", 3) == 0) |
425 | pnpbios_disabled=1; | 429 | pnpbios_disabled = 1; |
426 | if (strncmp(str, "on", 2) == 0) | 430 | if (strncmp(str, "on", 2) == 0) |
427 | pnpbios_disabled=0; | 431 | pnpbios_disabled = 0; |
428 | invert = (strncmp(str, "no-", 3) == 0); | 432 | invert = (strncmp(str, "no-", 3) == 0); |
429 | if (invert) | 433 | if (invert) |
430 | str += 3; | 434 | str += 3; |
@@ -453,35 +457,41 @@ static int __init pnpbios_probe_system(void) | |||
453 | printk(KERN_INFO "PnPBIOS: Scanning system for PnP BIOS support...\n"); | 457 | printk(KERN_INFO "PnPBIOS: Scanning system for PnP BIOS support...\n"); |
454 | 458 | ||
455 | /* | 459 | /* |
456 | * Search the defined area (0xf0000-0xffff0) for a valid PnP BIOS | 460 | * Search the defined area (0xf0000-0xffff0) for a valid PnP BIOS |
457 | * structure and, if one is found, sets up the selectors and | 461 | * structure and, if one is found, sets up the selectors and |
458 | * entry points | 462 | * entry points |
459 | */ | 463 | */ |
460 | for (check = (union pnp_bios_install_struct *) __va(0xf0000); | 464 | for (check = (union pnp_bios_install_struct *)__va(0xf0000); |
461 | check < (union pnp_bios_install_struct *) __va(0xffff0); | 465 | check < (union pnp_bios_install_struct *)__va(0xffff0); |
462 | check = (void *)check + 16) { | 466 | check = (void *)check + 16) { |
463 | if (check->fields.signature != PNP_SIGNATURE) | 467 | if (check->fields.signature != PNP_SIGNATURE) |
464 | continue; | 468 | continue; |
465 | printk(KERN_INFO "PnPBIOS: Found PnP BIOS installation structure at 0x%p\n", check); | 469 | printk(KERN_INFO |
470 | "PnPBIOS: Found PnP BIOS installation structure at 0x%p\n", | ||
471 | check); | ||
466 | length = check->fields.length; | 472 | length = check->fields.length; |
467 | if (!length) { | 473 | if (!length) { |
468 | printk(KERN_ERR "PnPBIOS: installation structure is invalid, skipping\n"); | 474 | printk(KERN_ERR |
475 | "PnPBIOS: installation structure is invalid, skipping\n"); | ||
469 | continue; | 476 | continue; |
470 | } | 477 | } |
471 | for (sum = 0, i = 0; i < length; i++) | 478 | for (sum = 0, i = 0; i < length; i++) |
472 | sum += check->chars[i]; | 479 | sum += check->chars[i]; |
473 | if (sum) { | 480 | if (sum) { |
474 | printk(KERN_ERR "PnPBIOS: installation structure is corrupted, skipping\n"); | 481 | printk(KERN_ERR |
482 | "PnPBIOS: installation structure is corrupted, skipping\n"); | ||
475 | continue; | 483 | continue; |
476 | } | 484 | } |
477 | if (check->fields.version < 0x10) { | 485 | if (check->fields.version < 0x10) { |
478 | printk(KERN_WARNING "PnPBIOS: PnP BIOS version %d.%d is not supported\n", | 486 | printk(KERN_WARNING |
487 | "PnPBIOS: PnP BIOS version %d.%d is not supported\n", | ||
479 | check->fields.version >> 4, | 488 | check->fields.version >> 4, |
480 | check->fields.version & 15); | 489 | check->fields.version & 15); |
481 | continue; | 490 | continue; |
482 | } | 491 | } |
483 | printk(KERN_INFO "PnPBIOS: PnP BIOS version %d.%d, entry 0x%x:0x%x, dseg 0x%x\n", | 492 | printk(KERN_INFO |
484 | check->fields.version >> 4, check->fields.version & 15, | 493 | "PnPBIOS: PnP BIOS version %d.%d, entry 0x%x:0x%x, dseg 0x%x\n", |
494 | check->fields.version >> 4, check->fields.version & 15, | ||
485 | check->fields.pm16cseg, check->fields.pm16offset, | 495 | check->fields.pm16cseg, check->fields.pm16offset, |
486 | check->fields.pm16dseg); | 496 | check->fields.pm16dseg); |
487 | pnp_bios_install = check; | 497 | pnp_bios_install = check; |
@@ -499,25 +509,25 @@ static int __init exploding_pnp_bios(struct dmi_system_id *d) | |||
499 | } | 509 | } |
500 | 510 | ||
501 | static struct dmi_system_id pnpbios_dmi_table[] __initdata = { | 511 | static struct dmi_system_id pnpbios_dmi_table[] __initdata = { |
502 | { /* PnPBIOS GPF on boot */ | 512 | { /* PnPBIOS GPF on boot */ |
503 | .callback = exploding_pnp_bios, | 513 | .callback = exploding_pnp_bios, |
504 | .ident = "Higraded P14H", | 514 | .ident = "Higraded P14H", |
505 | .matches = { | 515 | .matches = { |
506 | DMI_MATCH(DMI_BIOS_VENDOR, "American Megatrends Inc."), | 516 | DMI_MATCH(DMI_BIOS_VENDOR, "American Megatrends Inc."), |
507 | DMI_MATCH(DMI_BIOS_VERSION, "07.00T"), | 517 | DMI_MATCH(DMI_BIOS_VERSION, "07.00T"), |
508 | DMI_MATCH(DMI_SYS_VENDOR, "Higraded"), | 518 | DMI_MATCH(DMI_SYS_VENDOR, "Higraded"), |
509 | DMI_MATCH(DMI_PRODUCT_NAME, "P14H"), | 519 | DMI_MATCH(DMI_PRODUCT_NAME, "P14H"), |
510 | }, | 520 | }, |
511 | }, | 521 | }, |
512 | { /* PnPBIOS GPF on boot */ | 522 | { /* PnPBIOS GPF on boot */ |
513 | .callback = exploding_pnp_bios, | 523 | .callback = exploding_pnp_bios, |
514 | .ident = "ASUS P4P800", | 524 | .ident = "ASUS P4P800", |
515 | .matches = { | 525 | .matches = { |
516 | DMI_MATCH(DMI_BOARD_VENDOR, "ASUSTeK Computer Inc."), | 526 | DMI_MATCH(DMI_BOARD_VENDOR, "ASUSTeK Computer Inc."), |
517 | DMI_MATCH(DMI_BOARD_NAME, "P4P800"), | 527 | DMI_MATCH(DMI_BOARD_NAME, "P4P800"), |
518 | }, | 528 | }, |
519 | }, | 529 | }, |
520 | { } | 530 | {} |
521 | }; | 531 | }; |
522 | 532 | ||
523 | static int __init pnpbios_init(void) | 533 | static int __init pnpbios_init(void) |
@@ -533,14 +543,13 @@ static int __init pnpbios_init(void) | |||
533 | printk(KERN_INFO "PnPBIOS: Disabled\n"); | 543 | printk(KERN_INFO "PnPBIOS: Disabled\n"); |
534 | return -ENODEV; | 544 | return -ENODEV; |
535 | } | 545 | } |
536 | |||
537 | #ifdef CONFIG_PNPACPI | 546 | #ifdef CONFIG_PNPACPI |
538 | if (!acpi_disabled && !pnpacpi_disabled) { | 547 | if (!acpi_disabled && !pnpacpi_disabled) { |
539 | pnpbios_disabled = 1; | 548 | pnpbios_disabled = 1; |
540 | printk(KERN_INFO "PnPBIOS: Disabled by ACPI PNP\n"); | 549 | printk(KERN_INFO "PnPBIOS: Disabled by ACPI PNP\n"); |
541 | return -ENODEV; | 550 | return -ENODEV; |
542 | } | 551 | } |
543 | #endif /* CONFIG_ACPI */ | 552 | #endif /* CONFIG_ACPI */ |
544 | 553 | ||
545 | /* scan the system for pnpbios support */ | 554 | /* scan the system for pnpbios support */ |
546 | if (!pnpbios_probe_system()) | 555 | if (!pnpbios_probe_system()) |
@@ -552,14 +561,16 @@ static int __init pnpbios_init(void) | |||
552 | /* read the node info */ | 561 | /* read the node info */ |
553 | ret = pnp_bios_dev_node_info(&node_info); | 562 | ret = pnp_bios_dev_node_info(&node_info); |
554 | if (ret) { | 563 | if (ret) { |
555 | printk(KERN_ERR "PnPBIOS: Unable to get node info. Aborting.\n"); | 564 | printk(KERN_ERR |
565 | "PnPBIOS: Unable to get node info. Aborting.\n"); | ||
556 | return ret; | 566 | return ret; |
557 | } | 567 | } |
558 | 568 | ||
559 | /* register with the pnp layer */ | 569 | /* register with the pnp layer */ |
560 | ret = pnp_register_protocol(&pnpbios_protocol); | 570 | ret = pnp_register_protocol(&pnpbios_protocol); |
561 | if (ret) { | 571 | if (ret) { |
562 | printk(KERN_ERR "PnPBIOS: Unable to register driver. Aborting.\n"); | 572 | printk(KERN_ERR |
573 | "PnPBIOS: Unable to register driver. Aborting.\n"); | ||
563 | return ret; | 574 | return ret; |
564 | } | 575 | } |
565 | 576 | ||
diff --git a/drivers/pnp/pnpbios/proc.c b/drivers/pnp/pnpbios/proc.c index 8027073f7919..9c8c07701b65 100644 --- a/drivers/pnp/pnpbios/proc.c +++ b/drivers/pnp/pnpbios/proc.c | |||
@@ -18,9 +18,6 @@ | |||
18 | * The other files are human-readable. | 18 | * The other files are human-readable. |
19 | */ | 19 | */ |
20 | 20 | ||
21 | //#include <pcmcia/config.h> | ||
22 | //#include <pcmcia/k_compat.h> | ||
23 | |||
24 | #include <linux/module.h> | 21 | #include <linux/module.h> |
25 | #include <linux/kernel.h> | 22 | #include <linux/kernel.h> |
26 | #include <linux/slab.h> | 23 | #include <linux/slab.h> |
@@ -37,42 +34,37 @@ static struct proc_dir_entry *proc_pnp = NULL; | |||
37 | static struct proc_dir_entry *proc_pnp_boot = NULL; | 34 | static struct proc_dir_entry *proc_pnp_boot = NULL; |
38 | 35 | ||
39 | static int proc_read_pnpconfig(char *buf, char **start, off_t pos, | 36 | static int proc_read_pnpconfig(char *buf, char **start, off_t pos, |
40 | int count, int *eof, void *data) | 37 | int count, int *eof, void *data) |
41 | { | 38 | { |
42 | struct pnp_isa_config_struc pnps; | 39 | struct pnp_isa_config_struc pnps; |
43 | 40 | ||
44 | if (pnp_bios_isapnp_config(&pnps)) | 41 | if (pnp_bios_isapnp_config(&pnps)) |
45 | return -EIO; | 42 | return -EIO; |
46 | return snprintf(buf, count, | 43 | return snprintf(buf, count, |
47 | "structure_revision %d\n" | 44 | "structure_revision %d\n" |
48 | "number_of_CSNs %d\n" | 45 | "number_of_CSNs %d\n" |
49 | "ISA_read_data_port 0x%x\n", | 46 | "ISA_read_data_port 0x%x\n", |
50 | pnps.revision, | 47 | pnps.revision, pnps.no_csns, pnps.isa_rd_data_port); |
51 | pnps.no_csns, | ||
52 | pnps.isa_rd_data_port | ||
53 | ); | ||
54 | } | 48 | } |
55 | 49 | ||
56 | static int proc_read_escdinfo(char *buf, char **start, off_t pos, | 50 | static int proc_read_escdinfo(char *buf, char **start, off_t pos, |
57 | int count, int *eof, void *data) | 51 | int count, int *eof, void *data) |
58 | { | 52 | { |
59 | struct escd_info_struc escd; | 53 | struct escd_info_struc escd; |
60 | 54 | ||
61 | if (pnp_bios_escd_info(&escd)) | 55 | if (pnp_bios_escd_info(&escd)) |
62 | return -EIO; | 56 | return -EIO; |
63 | return snprintf(buf, count, | 57 | return snprintf(buf, count, |
64 | "min_ESCD_write_size %d\n" | 58 | "min_ESCD_write_size %d\n" |
65 | "ESCD_size %d\n" | 59 | "ESCD_size %d\n" |
66 | "NVRAM_base 0x%x\n", | 60 | "NVRAM_base 0x%x\n", |
67 | escd.min_escd_write_size, | 61 | escd.min_escd_write_size, |
68 | escd.escd_size, | 62 | escd.escd_size, escd.nv_storage_base); |
69 | escd.nv_storage_base | ||
70 | ); | ||
71 | } | 63 | } |
72 | 64 | ||
73 | #define MAX_SANE_ESCD_SIZE (32*1024) | 65 | #define MAX_SANE_ESCD_SIZE (32*1024) |
74 | static int proc_read_escd(char *buf, char **start, off_t pos, | 66 | static int proc_read_escd(char *buf, char **start, off_t pos, |
75 | int count, int *eof, void *data) | 67 | int count, int *eof, void *data) |
76 | { | 68 | { |
77 | struct escd_info_struc escd; | 69 | struct escd_info_struc escd; |
78 | char *tmpbuf; | 70 | char *tmpbuf; |
@@ -83,30 +75,36 @@ static int proc_read_escd(char *buf, char **start, off_t pos, | |||
83 | 75 | ||
84 | /* sanity check */ | 76 | /* sanity check */ |
85 | if (escd.escd_size > MAX_SANE_ESCD_SIZE) { | 77 | if (escd.escd_size > MAX_SANE_ESCD_SIZE) { |
86 | printk(KERN_ERR "PnPBIOS: proc_read_escd: ESCD size reported by BIOS escd_info call is too great\n"); | 78 | printk(KERN_ERR |
79 | "PnPBIOS: proc_read_escd: ESCD size reported by BIOS escd_info call is too great\n"); | ||
87 | return -EFBIG; | 80 | return -EFBIG; |
88 | } | 81 | } |
89 | 82 | ||
90 | tmpbuf = kzalloc(escd.escd_size, GFP_KERNEL); | 83 | tmpbuf = kzalloc(escd.escd_size, GFP_KERNEL); |
91 | if (!tmpbuf) return -ENOMEM; | 84 | if (!tmpbuf) |
85 | return -ENOMEM; | ||
92 | 86 | ||
93 | if (pnp_bios_read_escd(tmpbuf, escd.nv_storage_base)) { | 87 | if (pnp_bios_read_escd(tmpbuf, escd.nv_storage_base)) { |
94 | kfree(tmpbuf); | 88 | kfree(tmpbuf); |
95 | return -EIO; | 89 | return -EIO; |
96 | } | 90 | } |
97 | 91 | ||
98 | escd_size = (unsigned char)(tmpbuf[0]) + (unsigned char)(tmpbuf[1])*256; | 92 | escd_size = |
93 | (unsigned char)(tmpbuf[0]) + (unsigned char)(tmpbuf[1]) * 256; | ||
99 | 94 | ||
100 | /* sanity check */ | 95 | /* sanity check */ |
101 | if (escd_size > MAX_SANE_ESCD_SIZE) { | 96 | if (escd_size > MAX_SANE_ESCD_SIZE) { |
102 | printk(KERN_ERR "PnPBIOS: proc_read_escd: ESCD size reported by BIOS read_escd call is too great\n"); | 97 | printk(KERN_ERR |
98 | "PnPBIOS: proc_read_escd: ESCD size reported by BIOS read_escd call is too great\n"); | ||
103 | return -EFBIG; | 99 | return -EFBIG; |
104 | } | 100 | } |
105 | 101 | ||
106 | escd_left_to_read = escd_size - pos; | 102 | escd_left_to_read = escd_size - pos; |
107 | if (escd_left_to_read < 0) escd_left_to_read = 0; | 103 | if (escd_left_to_read < 0) |
108 | if (escd_left_to_read == 0) *eof = 1; | 104 | escd_left_to_read = 0; |
109 | n = min(count,escd_left_to_read); | 105 | if (escd_left_to_read == 0) |
106 | *eof = 1; | ||
107 | n = min(count, escd_left_to_read); | ||
110 | memcpy(buf, tmpbuf + pos, n); | 108 | memcpy(buf, tmpbuf + pos, n); |
111 | kfree(tmpbuf); | 109 | kfree(tmpbuf); |
112 | *start = buf; | 110 | *start = buf; |
@@ -114,17 +112,17 @@ static int proc_read_escd(char *buf, char **start, off_t pos, | |||
114 | } | 112 | } |
115 | 113 | ||
116 | static int proc_read_legacyres(char *buf, char **start, off_t pos, | 114 | static int proc_read_legacyres(char *buf, char **start, off_t pos, |
117 | int count, int *eof, void *data) | 115 | int count, int *eof, void *data) |
118 | { | 116 | { |
119 | /* Assume that the following won't overflow the buffer */ | 117 | /* Assume that the following won't overflow the buffer */ |
120 | if (pnp_bios_get_stat_res(buf)) | 118 | if (pnp_bios_get_stat_res(buf)) |
121 | return -EIO; | 119 | return -EIO; |
122 | 120 | ||
123 | return count; // FIXME: Return actual length | 121 | return count; // FIXME: Return actual length |
124 | } | 122 | } |
125 | 123 | ||
126 | static int proc_read_devices(char *buf, char **start, off_t pos, | 124 | static int proc_read_devices(char *buf, char **start, off_t pos, |
127 | int count, int *eof, void *data) | 125 | int count, int *eof, void *data) |
128 | { | 126 | { |
129 | struct pnp_bios_node *node; | 127 | struct pnp_bios_node *node; |
130 | u8 nodenum; | 128 | u8 nodenum; |
@@ -134,9 +132,10 @@ static int proc_read_devices(char *buf, char **start, off_t pos, | |||
134 | return 0; | 132 | return 0; |
135 | 133 | ||
136 | node = kzalloc(node_info.max_node_size, GFP_KERNEL); | 134 | node = kzalloc(node_info.max_node_size, GFP_KERNEL); |
137 | if (!node) return -ENOMEM; | 135 | if (!node) |
136 | return -ENOMEM; | ||
138 | 137 | ||
139 | for (nodenum=pos; nodenum<0xff; ) { | 138 | for (nodenum = pos; nodenum < 0xff;) { |
140 | u8 thisnodenum = nodenum; | 139 | u8 thisnodenum = nodenum; |
141 | /* 26 = the number of characters per line sprintf'ed */ | 140 | /* 26 = the number of characters per line sprintf'ed */ |
142 | if ((p - buf + 26) > count) | 141 | if ((p - buf + 26) > count) |
@@ -148,7 +147,11 @@ static int proc_read_devices(char *buf, char **start, off_t pos, | |||
148 | node->type_code[0], node->type_code[1], | 147 | node->type_code[0], node->type_code[1], |
149 | node->type_code[2], node->flags); | 148 | node->type_code[2], node->flags); |
150 | if (nodenum <= thisnodenum) { | 149 | if (nodenum <= thisnodenum) { |
151 | printk(KERN_ERR "%s Node number 0x%x is out of sequence following node 0x%x. Aborting.\n", "PnPBIOS: proc_read_devices:", (unsigned int)nodenum, (unsigned int)thisnodenum); | 150 | printk(KERN_ERR |
151 | "%s Node number 0x%x is out of sequence following node 0x%x. Aborting.\n", | ||
152 | "PnPBIOS: proc_read_devices:", | ||
153 | (unsigned int)nodenum, | ||
154 | (unsigned int)thisnodenum); | ||
152 | *eof = 1; | 155 | *eof = 1; |
153 | break; | 156 | break; |
154 | } | 157 | } |
@@ -156,12 +159,12 @@ static int proc_read_devices(char *buf, char **start, off_t pos, | |||
156 | kfree(node); | 159 | kfree(node); |
157 | if (nodenum == 0xff) | 160 | if (nodenum == 0xff) |
158 | *eof = 1; | 161 | *eof = 1; |
159 | *start = (char *)((off_t)nodenum - pos); | 162 | *start = (char *)((off_t) nodenum - pos); |
160 | return p - buf; | 163 | return p - buf; |
161 | } | 164 | } |
162 | 165 | ||
163 | static int proc_read_node(char *buf, char **start, off_t pos, | 166 | static int proc_read_node(char *buf, char **start, off_t pos, |
164 | int count, int *eof, void *data) | 167 | int count, int *eof, void *data) |
165 | { | 168 | { |
166 | struct pnp_bios_node *node; | 169 | struct pnp_bios_node *node; |
167 | int boot = (long)data >> 8; | 170 | int boot = (long)data >> 8; |
@@ -169,7 +172,8 @@ static int proc_read_node(char *buf, char **start, off_t pos, | |||
169 | int len; | 172 | int len; |
170 | 173 | ||
171 | node = kzalloc(node_info.max_node_size, GFP_KERNEL); | 174 | node = kzalloc(node_info.max_node_size, GFP_KERNEL); |
172 | if (!node) return -ENOMEM; | 175 | if (!node) |
176 | return -ENOMEM; | ||
173 | if (pnp_bios_get_dev_node(&nodenum, boot, node)) { | 177 | if (pnp_bios_get_dev_node(&nodenum, boot, node)) { |
174 | kfree(node); | 178 | kfree(node); |
175 | return -EIO; | 179 | return -EIO; |
@@ -180,8 +184,8 @@ static int proc_read_node(char *buf, char **start, off_t pos, | |||
180 | return len; | 184 | return len; |
181 | } | 185 | } |
182 | 186 | ||
183 | static int proc_write_node(struct file *file, const char __user *buf, | 187 | static int proc_write_node(struct file *file, const char __user * buf, |
184 | unsigned long count, void *data) | 188 | unsigned long count, void *data) |
185 | { | 189 | { |
186 | struct pnp_bios_node *node; | 190 | struct pnp_bios_node *node; |
187 | int boot = (long)data >> 8; | 191 | int boot = (long)data >> 8; |
@@ -208,12 +212,12 @@ static int proc_write_node(struct file *file, const char __user *buf, | |||
208 | goto out; | 212 | goto out; |
209 | } | 213 | } |
210 | ret = count; | 214 | ret = count; |
211 | out: | 215 | out: |
212 | kfree(node); | 216 | kfree(node); |
213 | return ret; | 217 | return ret; |
214 | } | 218 | } |
215 | 219 | ||
216 | int pnpbios_interface_attach_device(struct pnp_bios_node * node) | 220 | int pnpbios_interface_attach_device(struct pnp_bios_node *node) |
217 | { | 221 | { |
218 | char name[3]; | 222 | char name[3]; |
219 | struct proc_dir_entry *ent; | 223 | struct proc_dir_entry *ent; |
@@ -222,7 +226,7 @@ int pnpbios_interface_attach_device(struct pnp_bios_node * node) | |||
222 | 226 | ||
223 | if (!proc_pnp) | 227 | if (!proc_pnp) |
224 | return -EIO; | 228 | return -EIO; |
225 | if ( !pnpbios_dont_use_current_config ) { | 229 | if (!pnpbios_dont_use_current_config) { |
226 | ent = create_proc_entry(name, 0, proc_pnp); | 230 | ent = create_proc_entry(name, 0, proc_pnp); |
227 | if (ent) { | 231 | if (ent) { |
228 | ent->read_proc = proc_read_node; | 232 | ent->read_proc = proc_read_node; |
@@ -237,7 +241,7 @@ int pnpbios_interface_attach_device(struct pnp_bios_node * node) | |||
237 | if (ent) { | 241 | if (ent) { |
238 | ent->read_proc = proc_read_node; | 242 | ent->read_proc = proc_read_node; |
239 | ent->write_proc = proc_write_node; | 243 | ent->write_proc = proc_write_node; |
240 | ent->data = (void *)(long)(node->handle+0x100); | 244 | ent->data = (void *)(long)(node->handle + 0x100); |
241 | return 0; | 245 | return 0; |
242 | } | 246 | } |
243 | 247 | ||
@@ -249,7 +253,7 @@ int pnpbios_interface_attach_device(struct pnp_bios_node * node) | |||
249 | * work and the pnpbios_dont_use_current_config flag | 253 | * work and the pnpbios_dont_use_current_config flag |
250 | * should already have been set to the appropriate value | 254 | * should already have been set to the appropriate value |
251 | */ | 255 | */ |
252 | int __init pnpbios_proc_init( void ) | 256 | int __init pnpbios_proc_init(void) |
253 | { | 257 | { |
254 | proc_pnp = proc_mkdir("pnp", proc_bus); | 258 | proc_pnp = proc_mkdir("pnp", proc_bus); |
255 | if (!proc_pnp) | 259 | if (!proc_pnp) |
@@ -258,10 +262,13 @@ int __init pnpbios_proc_init( void ) | |||
258 | if (!proc_pnp_boot) | 262 | if (!proc_pnp_boot) |
259 | return -EIO; | 263 | return -EIO; |
260 | create_proc_read_entry("devices", 0, proc_pnp, proc_read_devices, NULL); | 264 | create_proc_read_entry("devices", 0, proc_pnp, proc_read_devices, NULL); |
261 | create_proc_read_entry("configuration_info", 0, proc_pnp, proc_read_pnpconfig, NULL); | 265 | create_proc_read_entry("configuration_info", 0, proc_pnp, |
262 | create_proc_read_entry("escd_info", 0, proc_pnp, proc_read_escdinfo, NULL); | 266 | proc_read_pnpconfig, NULL); |
267 | create_proc_read_entry("escd_info", 0, proc_pnp, proc_read_escdinfo, | ||
268 | NULL); | ||
263 | create_proc_read_entry("escd", S_IRUSR, proc_pnp, proc_read_escd, NULL); | 269 | create_proc_read_entry("escd", S_IRUSR, proc_pnp, proc_read_escd, NULL); |
264 | create_proc_read_entry("legacy_device_resources", 0, proc_pnp, proc_read_legacyres, NULL); | 270 | create_proc_read_entry("legacy_device_resources", 0, proc_pnp, |
271 | proc_read_legacyres, NULL); | ||
265 | 272 | ||
266 | return 0; | 273 | return 0; |
267 | } | 274 | } |
@@ -274,9 +281,9 @@ void __exit pnpbios_proc_exit(void) | |||
274 | if (!proc_pnp) | 281 | if (!proc_pnp) |
275 | return; | 282 | return; |
276 | 283 | ||
277 | for (i=0; i<0xff; i++) { | 284 | for (i = 0; i < 0xff; i++) { |
278 | sprintf(name, "%02x", i); | 285 | sprintf(name, "%02x", i); |
279 | if ( !pnpbios_dont_use_current_config ) | 286 | if (!pnpbios_dont_use_current_config) |
280 | remove_proc_entry(name, proc_pnp); | 287 | remove_proc_entry(name, proc_pnp); |
281 | remove_proc_entry(name, proc_pnp_boot); | 288 | remove_proc_entry(name, proc_pnp_boot); |
282 | } | 289 | } |
@@ -287,6 +294,4 @@ void __exit pnpbios_proc_exit(void) | |||
287 | remove_proc_entry("devices", proc_pnp); | 294 | remove_proc_entry("devices", proc_pnp); |
288 | remove_proc_entry("boot", proc_pnp); | 295 | remove_proc_entry("boot", proc_pnp); |
289 | remove_proc_entry("pnp", proc_bus); | 296 | remove_proc_entry("pnp", proc_bus); |
290 | |||
291 | return; | ||
292 | } | 297 | } |
diff --git a/drivers/pnp/pnpbios/rsparser.c b/drivers/pnp/pnpbios/rsparser.c index 3c2ab8394e3f..04ecd7b67230 100644 --- a/drivers/pnp/pnpbios/rsparser.c +++ b/drivers/pnp/pnpbios/rsparser.c | |||
@@ -1,6 +1,5 @@ | |||
1 | /* | 1 | /* |
2 | * rsparser.c - parses and encodes pnpbios resource data streams | 2 | * rsparser.c - parses and encodes pnpbios resource data streams |
3 | * | ||
4 | */ | 3 | */ |
5 | 4 | ||
6 | #include <linux/ctype.h> | 5 | #include <linux/ctype.h> |
@@ -12,8 +11,10 @@ | |||
12 | #ifdef CONFIG_PCI | 11 | #ifdef CONFIG_PCI |
13 | #include <linux/pci.h> | 12 | #include <linux/pci.h> |
14 | #else | 13 | #else |
15 | inline void pcibios_penalize_isa_irq(int irq, int active) {} | 14 | inline void pcibios_penalize_isa_irq(int irq, int active) |
16 | #endif /* CONFIG_PCI */ | 15 | { |
16 | } | ||
17 | #endif /* CONFIG_PCI */ | ||
17 | 18 | ||
18 | #include "pnpbios.h" | 19 | #include "pnpbios.h" |
19 | 20 | ||
@@ -52,75 +53,88 @@ inline void pcibios_penalize_isa_irq(int irq, int active) {} | |||
52 | * Allocated Resources | 53 | * Allocated Resources |
53 | */ | 54 | */ |
54 | 55 | ||
55 | static void | 56 | static void pnpbios_parse_allocated_irqresource(struct pnp_resource_table *res, |
56 | pnpbios_parse_allocated_irqresource(struct pnp_resource_table * res, int irq) | 57 | int irq) |
57 | { | 58 | { |
58 | int i = 0; | 59 | int i = 0; |
59 | while (!(res->irq_resource[i].flags & IORESOURCE_UNSET) && i < PNP_MAX_IRQ) i++; | 60 | |
61 | while (!(res->irq_resource[i].flags & IORESOURCE_UNSET) | ||
62 | && i < PNP_MAX_IRQ) | ||
63 | i++; | ||
60 | if (i < PNP_MAX_IRQ) { | 64 | if (i < PNP_MAX_IRQ) { |
61 | res->irq_resource[i].flags = IORESOURCE_IRQ; // Also clears _UNSET flag | 65 | res->irq_resource[i].flags = IORESOURCE_IRQ; // Also clears _UNSET flag |
62 | if (irq == -1) { | 66 | if (irq == -1) { |
63 | res->irq_resource[i].flags |= IORESOURCE_DISABLED; | 67 | res->irq_resource[i].flags |= IORESOURCE_DISABLED; |
64 | return; | 68 | return; |
65 | } | 69 | } |
66 | res->irq_resource[i].start = | 70 | res->irq_resource[i].start = |
67 | res->irq_resource[i].end = (unsigned long) irq; | 71 | res->irq_resource[i].end = (unsigned long)irq; |
68 | pcibios_penalize_isa_irq(irq, 1); | 72 | pcibios_penalize_isa_irq(irq, 1); |
69 | } | 73 | } |
70 | } | 74 | } |
71 | 75 | ||
72 | static void | 76 | static void pnpbios_parse_allocated_dmaresource(struct pnp_resource_table *res, |
73 | pnpbios_parse_allocated_dmaresource(struct pnp_resource_table * res, int dma) | 77 | int dma) |
74 | { | 78 | { |
75 | int i = 0; | 79 | int i = 0; |
80 | |||
76 | while (i < PNP_MAX_DMA && | 81 | while (i < PNP_MAX_DMA && |
77 | !(res->dma_resource[i].flags & IORESOURCE_UNSET)) | 82 | !(res->dma_resource[i].flags & IORESOURCE_UNSET)) |
78 | i++; | 83 | i++; |
79 | if (i < PNP_MAX_DMA) { | 84 | if (i < PNP_MAX_DMA) { |
80 | res->dma_resource[i].flags = IORESOURCE_DMA; // Also clears _UNSET flag | 85 | res->dma_resource[i].flags = IORESOURCE_DMA; // Also clears _UNSET flag |
81 | if (dma == -1) { | 86 | if (dma == -1) { |
82 | res->dma_resource[i].flags |= IORESOURCE_DISABLED; | 87 | res->dma_resource[i].flags |= IORESOURCE_DISABLED; |
83 | return; | 88 | return; |
84 | } | 89 | } |
85 | res->dma_resource[i].start = | 90 | res->dma_resource[i].start = |
86 | res->dma_resource[i].end = (unsigned long) dma; | 91 | res->dma_resource[i].end = (unsigned long)dma; |
87 | } | 92 | } |
88 | } | 93 | } |
89 | 94 | ||
90 | static void | 95 | static void pnpbios_parse_allocated_ioresource(struct pnp_resource_table *res, |
91 | pnpbios_parse_allocated_ioresource(struct pnp_resource_table * res, int io, int len) | 96 | int io, int len) |
92 | { | 97 | { |
93 | int i = 0; | 98 | int i = 0; |
94 | while (!(res->port_resource[i].flags & IORESOURCE_UNSET) && i < PNP_MAX_PORT) i++; | 99 | |
100 | while (!(res->port_resource[i].flags & IORESOURCE_UNSET) | ||
101 | && i < PNP_MAX_PORT) | ||
102 | i++; | ||
95 | if (i < PNP_MAX_PORT) { | 103 | if (i < PNP_MAX_PORT) { |
96 | res->port_resource[i].flags = IORESOURCE_IO; // Also clears _UNSET flag | 104 | res->port_resource[i].flags = IORESOURCE_IO; // Also clears _UNSET flag |
97 | if (len <= 0 || (io + len -1) >= 0x10003) { | 105 | if (len <= 0 || (io + len - 1) >= 0x10003) { |
98 | res->port_resource[i].flags |= IORESOURCE_DISABLED; | 106 | res->port_resource[i].flags |= IORESOURCE_DISABLED; |
99 | return; | 107 | return; |
100 | } | 108 | } |
101 | res->port_resource[i].start = (unsigned long) io; | 109 | res->port_resource[i].start = (unsigned long)io; |
102 | res->port_resource[i].end = (unsigned long)(io + len - 1); | 110 | res->port_resource[i].end = (unsigned long)(io + len - 1); |
103 | } | 111 | } |
104 | } | 112 | } |
105 | 113 | ||
106 | static void | 114 | static void pnpbios_parse_allocated_memresource(struct pnp_resource_table *res, |
107 | pnpbios_parse_allocated_memresource(struct pnp_resource_table * res, int mem, int len) | 115 | int mem, int len) |
108 | { | 116 | { |
109 | int i = 0; | 117 | int i = 0; |
110 | while (!(res->mem_resource[i].flags & IORESOURCE_UNSET) && i < PNP_MAX_MEM) i++; | 118 | |
119 | while (!(res->mem_resource[i].flags & IORESOURCE_UNSET) | ||
120 | && i < PNP_MAX_MEM) | ||
121 | i++; | ||
111 | if (i < PNP_MAX_MEM) { | 122 | if (i < PNP_MAX_MEM) { |
112 | res->mem_resource[i].flags = IORESOURCE_MEM; // Also clears _UNSET flag | 123 | res->mem_resource[i].flags = IORESOURCE_MEM; // Also clears _UNSET flag |
113 | if (len <= 0) { | 124 | if (len <= 0) { |
114 | res->mem_resource[i].flags |= IORESOURCE_DISABLED; | 125 | res->mem_resource[i].flags |= IORESOURCE_DISABLED; |
115 | return; | 126 | return; |
116 | } | 127 | } |
117 | res->mem_resource[i].start = (unsigned long) mem; | 128 | res->mem_resource[i].start = (unsigned long)mem; |
118 | res->mem_resource[i].end = (unsigned long)(mem + len - 1); | 129 | res->mem_resource[i].end = (unsigned long)(mem + len - 1); |
119 | } | 130 | } |
120 | } | 131 | } |
121 | 132 | ||
122 | static unsigned char * | 133 | static unsigned char *pnpbios_parse_allocated_resource_data(unsigned char *p, |
123 | pnpbios_parse_allocated_resource_data(unsigned char * p, unsigned char * end, struct pnp_resource_table * res) | 134 | unsigned char *end, |
135 | struct | ||
136 | pnp_resource_table | ||
137 | *res) | ||
124 | { | 138 | { |
125 | unsigned int len, tag; | 139 | unsigned int len, tag; |
126 | int io, size, mask, i; | 140 | int io, size, mask, i; |
@@ -134,12 +148,12 @@ pnpbios_parse_allocated_resource_data(unsigned char * p, unsigned char * end, st | |||
134 | while ((char *)p < (char *)end) { | 148 | while ((char *)p < (char *)end) { |
135 | 149 | ||
136 | /* determine the type of tag */ | 150 | /* determine the type of tag */ |
137 | if (p[0] & LARGE_TAG) { /* large tag */ | 151 | if (p[0] & LARGE_TAG) { /* large tag */ |
138 | len = (p[2] << 8) | p[1]; | 152 | len = (p[2] << 8) | p[1]; |
139 | tag = p[0]; | 153 | tag = p[0]; |
140 | } else { /* small tag */ | 154 | } else { /* small tag */ |
141 | len = p[0] & 0x07; | 155 | len = p[0] & 0x07; |
142 | tag = ((p[0]>>3) & 0x0f); | 156 | tag = ((p[0] >> 3) & 0x0f); |
143 | } | 157 | } |
144 | 158 | ||
145 | switch (tag) { | 159 | switch (tag) { |
@@ -147,8 +161,8 @@ pnpbios_parse_allocated_resource_data(unsigned char * p, unsigned char * end, st | |||
147 | case LARGE_TAG_MEM: | 161 | case LARGE_TAG_MEM: |
148 | if (len != 9) | 162 | if (len != 9) |
149 | goto len_err; | 163 | goto len_err; |
150 | io = *(short *) &p[4]; | 164 | io = *(short *)&p[4]; |
151 | size = *(short *) &p[10]; | 165 | size = *(short *)&p[10]; |
152 | pnpbios_parse_allocated_memresource(res, io, size); | 166 | pnpbios_parse_allocated_memresource(res, io, size); |
153 | break; | 167 | break; |
154 | 168 | ||
@@ -163,16 +177,16 @@ pnpbios_parse_allocated_resource_data(unsigned char * p, unsigned char * end, st | |||
163 | case LARGE_TAG_MEM32: | 177 | case LARGE_TAG_MEM32: |
164 | if (len != 17) | 178 | if (len != 17) |
165 | goto len_err; | 179 | goto len_err; |
166 | io = *(int *) &p[4]; | 180 | io = *(int *)&p[4]; |
167 | size = *(int *) &p[16]; | 181 | size = *(int *)&p[16]; |
168 | pnpbios_parse_allocated_memresource(res, io, size); | 182 | pnpbios_parse_allocated_memresource(res, io, size); |
169 | break; | 183 | break; |
170 | 184 | ||
171 | case LARGE_TAG_FIXEDMEM32: | 185 | case LARGE_TAG_FIXEDMEM32: |
172 | if (len != 9) | 186 | if (len != 9) |
173 | goto len_err; | 187 | goto len_err; |
174 | io = *(int *) &p[4]; | 188 | io = *(int *)&p[4]; |
175 | size = *(int *) &p[8]; | 189 | size = *(int *)&p[8]; |
176 | pnpbios_parse_allocated_memresource(res, io, size); | 190 | pnpbios_parse_allocated_memresource(res, io, size); |
177 | break; | 191 | break; |
178 | 192 | ||
@@ -180,9 +194,10 @@ pnpbios_parse_allocated_resource_data(unsigned char * p, unsigned char * end, st | |||
180 | if (len < 2 || len > 3) | 194 | if (len < 2 || len > 3) |
181 | goto len_err; | 195 | goto len_err; |
182 | io = -1; | 196 | io = -1; |
183 | mask= p[1] + p[2]*256; | 197 | mask = p[1] + p[2] * 256; |
184 | for (i=0;i<16;i++, mask=mask>>1) | 198 | for (i = 0; i < 16; i++, mask = mask >> 1) |
185 | if(mask & 0x01) io=i; | 199 | if (mask & 0x01) |
200 | io = i; | ||
186 | pnpbios_parse_allocated_irqresource(res, io); | 201 | pnpbios_parse_allocated_irqresource(res, io); |
187 | break; | 202 | break; |
188 | 203 | ||
@@ -191,15 +206,16 @@ pnpbios_parse_allocated_resource_data(unsigned char * p, unsigned char * end, st | |||
191 | goto len_err; | 206 | goto len_err; |
192 | io = -1; | 207 | io = -1; |
193 | mask = p[1]; | 208 | mask = p[1]; |
194 | for (i=0;i<8;i++, mask = mask>>1) | 209 | for (i = 0; i < 8; i++, mask = mask >> 1) |
195 | if(mask & 0x01) io=i; | 210 | if (mask & 0x01) |
211 | io = i; | ||
196 | pnpbios_parse_allocated_dmaresource(res, io); | 212 | pnpbios_parse_allocated_dmaresource(res, io); |
197 | break; | 213 | break; |
198 | 214 | ||
199 | case SMALL_TAG_PORT: | 215 | case SMALL_TAG_PORT: |
200 | if (len != 7) | 216 | if (len != 7) |
201 | goto len_err; | 217 | goto len_err; |
202 | io = p[2] + p[3] *256; | 218 | io = p[2] + p[3] * 256; |
203 | size = p[7]; | 219 | size = p[7]; |
204 | pnpbios_parse_allocated_ioresource(res, io, size); | 220 | pnpbios_parse_allocated_ioresource(res, io, size); |
205 | break; | 221 | break; |
@@ -218,12 +234,14 @@ pnpbios_parse_allocated_resource_data(unsigned char * p, unsigned char * end, st | |||
218 | 234 | ||
219 | case SMALL_TAG_END: | 235 | case SMALL_TAG_END: |
220 | p = p + 2; | 236 | p = p + 2; |
221 | return (unsigned char *)p; | 237 | return (unsigned char *)p; |
222 | break; | 238 | break; |
223 | 239 | ||
224 | default: /* an unkown tag */ | 240 | default: /* an unkown tag */ |
225 | len_err: | 241 | len_err: |
226 | printk(KERN_ERR "PnPBIOS: Unknown tag '0x%x', length '%d'.\n", tag, len); | 242 | printk(KERN_ERR |
243 | "PnPBIOS: Unknown tag '0x%x', length '%d'.\n", | ||
244 | tag, len); | ||
227 | break; | 245 | break; |
228 | } | 246 | } |
229 | 247 | ||
@@ -234,20 +252,21 @@ pnpbios_parse_allocated_resource_data(unsigned char * p, unsigned char * end, st | |||
234 | p += len + 1; | 252 | p += len + 1; |
235 | } | 253 | } |
236 | 254 | ||
237 | printk(KERN_ERR "PnPBIOS: Resource structure does not contain an end tag.\n"); | 255 | printk(KERN_ERR |
256 | "PnPBIOS: Resource structure does not contain an end tag.\n"); | ||
238 | 257 | ||
239 | return NULL; | 258 | return NULL; |
240 | } | 259 | } |
241 | 260 | ||
242 | |||
243 | /* | 261 | /* |
244 | * Resource Configuration Options | 262 | * Resource Configuration Options |
245 | */ | 263 | */ |
246 | 264 | ||
247 | static void | 265 | static void pnpbios_parse_mem_option(unsigned char *p, int size, |
248 | pnpbios_parse_mem_option(unsigned char *p, int size, struct pnp_option *option) | 266 | struct pnp_option *option) |
249 | { | 267 | { |
250 | struct pnp_mem * mem; | 268 | struct pnp_mem *mem; |
269 | |||
251 | mem = kzalloc(sizeof(struct pnp_mem), GFP_KERNEL); | 270 | mem = kzalloc(sizeof(struct pnp_mem), GFP_KERNEL); |
252 | if (!mem) | 271 | if (!mem) |
253 | return; | 272 | return; |
@@ -256,14 +275,14 @@ pnpbios_parse_mem_option(unsigned char *p, int size, struct pnp_option *option) | |||
256 | mem->align = (p[9] << 8) | p[8]; | 275 | mem->align = (p[9] << 8) | p[8]; |
257 | mem->size = ((p[11] << 8) | p[10]) << 8; | 276 | mem->size = ((p[11] << 8) | p[10]) << 8; |
258 | mem->flags = p[3]; | 277 | mem->flags = p[3]; |
259 | pnp_register_mem_resource(option,mem); | 278 | pnp_register_mem_resource(option, mem); |
260 | return; | ||
261 | } | 279 | } |
262 | 280 | ||
263 | static void | 281 | static void pnpbios_parse_mem32_option(unsigned char *p, int size, |
264 | pnpbios_parse_mem32_option(unsigned char *p, int size, struct pnp_option *option) | 282 | struct pnp_option *option) |
265 | { | 283 | { |
266 | struct pnp_mem * mem; | 284 | struct pnp_mem *mem; |
285 | |||
267 | mem = kzalloc(sizeof(struct pnp_mem), GFP_KERNEL); | 286 | mem = kzalloc(sizeof(struct pnp_mem), GFP_KERNEL); |
268 | if (!mem) | 287 | if (!mem) |
269 | return; | 288 | return; |
@@ -272,14 +291,13 @@ pnpbios_parse_mem32_option(unsigned char *p, int size, struct pnp_option *option | |||
272 | mem->align = (p[15] << 24) | (p[14] << 16) | (p[13] << 8) | p[12]; | 291 | mem->align = (p[15] << 24) | (p[14] << 16) | (p[13] << 8) | p[12]; |
273 | mem->size = (p[19] << 24) | (p[18] << 16) | (p[17] << 8) | p[16]; | 292 | mem->size = (p[19] << 24) | (p[18] << 16) | (p[17] << 8) | p[16]; |
274 | mem->flags = p[3]; | 293 | mem->flags = p[3]; |
275 | pnp_register_mem_resource(option,mem); | 294 | pnp_register_mem_resource(option, mem); |
276 | return; | ||
277 | } | 295 | } |
278 | 296 | ||
279 | static void | 297 | static void pnpbios_parse_fixed_mem32_option(unsigned char *p, int size, |
280 | pnpbios_parse_fixed_mem32_option(unsigned char *p, int size, struct pnp_option *option) | 298 | struct pnp_option *option) |
281 | { | 299 | { |
282 | struct pnp_mem * mem; | 300 | struct pnp_mem *mem; |
283 | mem = kzalloc(sizeof(struct pnp_mem), GFP_KERNEL); | 301 | mem = kzalloc(sizeof(struct pnp_mem), GFP_KERNEL); |
284 | if (!mem) | 302 | if (!mem) |
285 | return; | 303 | return; |
@@ -287,14 +305,13 @@ pnpbios_parse_fixed_mem32_option(unsigned char *p, int size, struct pnp_option * | |||
287 | mem->size = (p[11] << 24) | (p[10] << 16) | (p[9] << 8) | p[8]; | 305 | mem->size = (p[11] << 24) | (p[10] << 16) | (p[9] << 8) | p[8]; |
288 | mem->align = 0; | 306 | mem->align = 0; |
289 | mem->flags = p[3]; | 307 | mem->flags = p[3]; |
290 | pnp_register_mem_resource(option,mem); | 308 | pnp_register_mem_resource(option, mem); |
291 | return; | ||
292 | } | 309 | } |
293 | 310 | ||
294 | static void | 311 | static void pnpbios_parse_irq_option(unsigned char *p, int size, |
295 | pnpbios_parse_irq_option(unsigned char *p, int size, struct pnp_option *option) | 312 | struct pnp_option *option) |
296 | { | 313 | { |
297 | struct pnp_irq * irq; | 314 | struct pnp_irq *irq; |
298 | unsigned long bits; | 315 | unsigned long bits; |
299 | 316 | ||
300 | irq = kzalloc(sizeof(struct pnp_irq), GFP_KERNEL); | 317 | irq = kzalloc(sizeof(struct pnp_irq), GFP_KERNEL); |
@@ -306,27 +323,27 @@ pnpbios_parse_irq_option(unsigned char *p, int size, struct pnp_option *option) | |||
306 | irq->flags = p[3]; | 323 | irq->flags = p[3]; |
307 | else | 324 | else |
308 | irq->flags = IORESOURCE_IRQ_HIGHEDGE; | 325 | irq->flags = IORESOURCE_IRQ_HIGHEDGE; |
309 | pnp_register_irq_resource(option,irq); | 326 | pnp_register_irq_resource(option, irq); |
310 | return; | ||
311 | } | 327 | } |
312 | 328 | ||
313 | static void | 329 | static void pnpbios_parse_dma_option(unsigned char *p, int size, |
314 | pnpbios_parse_dma_option(unsigned char *p, int size, struct pnp_option *option) | 330 | struct pnp_option *option) |
315 | { | 331 | { |
316 | struct pnp_dma * dma; | 332 | struct pnp_dma *dma; |
333 | |||
317 | dma = kzalloc(sizeof(struct pnp_dma), GFP_KERNEL); | 334 | dma = kzalloc(sizeof(struct pnp_dma), GFP_KERNEL); |
318 | if (!dma) | 335 | if (!dma) |
319 | return; | 336 | return; |
320 | dma->map = p[1]; | 337 | dma->map = p[1]; |
321 | dma->flags = p[2]; | 338 | dma->flags = p[2]; |
322 | pnp_register_dma_resource(option,dma); | 339 | pnp_register_dma_resource(option, dma); |
323 | return; | ||
324 | } | 340 | } |
325 | 341 | ||
326 | static void | 342 | static void pnpbios_parse_port_option(unsigned char *p, int size, |
327 | pnpbios_parse_port_option(unsigned char *p, int size, struct pnp_option *option) | 343 | struct pnp_option *option) |
328 | { | 344 | { |
329 | struct pnp_port * port; | 345 | struct pnp_port *port; |
346 | |||
330 | port = kzalloc(sizeof(struct pnp_port), GFP_KERNEL); | 347 | port = kzalloc(sizeof(struct pnp_port), GFP_KERNEL); |
331 | if (!port) | 348 | if (!port) |
332 | return; | 349 | return; |
@@ -335,14 +352,14 @@ pnpbios_parse_port_option(unsigned char *p, int size, struct pnp_option *option) | |||
335 | port->align = p[6]; | 352 | port->align = p[6]; |
336 | port->size = p[7]; | 353 | port->size = p[7]; |
337 | port->flags = p[1] ? PNP_PORT_FLAG_16BITADDR : 0; | 354 | port->flags = p[1] ? PNP_PORT_FLAG_16BITADDR : 0; |
338 | pnp_register_port_resource(option,port); | 355 | pnp_register_port_resource(option, port); |
339 | return; | ||
340 | } | 356 | } |
341 | 357 | ||
342 | static void | 358 | static void pnpbios_parse_fixed_port_option(unsigned char *p, int size, |
343 | pnpbios_parse_fixed_port_option(unsigned char *p, int size, struct pnp_option *option) | 359 | struct pnp_option *option) |
344 | { | 360 | { |
345 | struct pnp_port * port; | 361 | struct pnp_port *port; |
362 | |||
346 | port = kzalloc(sizeof(struct pnp_port), GFP_KERNEL); | 363 | port = kzalloc(sizeof(struct pnp_port), GFP_KERNEL); |
347 | if (!port) | 364 | if (!port) |
348 | return; | 365 | return; |
@@ -350,12 +367,12 @@ pnpbios_parse_fixed_port_option(unsigned char *p, int size, struct pnp_option *o | |||
350 | port->size = p[3]; | 367 | port->size = p[3]; |
351 | port->align = 0; | 368 | port->align = 0; |
352 | port->flags = PNP_PORT_FLAG_FIXED; | 369 | port->flags = PNP_PORT_FLAG_FIXED; |
353 | pnp_register_port_resource(option,port); | 370 | pnp_register_port_resource(option, port); |
354 | return; | ||
355 | } | 371 | } |
356 | 372 | ||
357 | static unsigned char * | 373 | static unsigned char *pnpbios_parse_resource_option_data(unsigned char *p, |
358 | pnpbios_parse_resource_option_data(unsigned char * p, unsigned char * end, struct pnp_dev *dev) | 374 | unsigned char *end, |
375 | struct pnp_dev *dev) | ||
359 | { | 376 | { |
360 | unsigned int len, tag; | 377 | unsigned int len, tag; |
361 | int priority = 0; | 378 | int priority = 0; |
@@ -371,12 +388,12 @@ pnpbios_parse_resource_option_data(unsigned char * p, unsigned char * end, struc | |||
371 | while ((char *)p < (char *)end) { | 388 | while ((char *)p < (char *)end) { |
372 | 389 | ||
373 | /* determine the type of tag */ | 390 | /* determine the type of tag */ |
374 | if (p[0] & LARGE_TAG) { /* large tag */ | 391 | if (p[0] & LARGE_TAG) { /* large tag */ |
375 | len = (p[2] << 8) | p[1]; | 392 | len = (p[2] << 8) | p[1]; |
376 | tag = p[0]; | 393 | tag = p[0]; |
377 | } else { /* small tag */ | 394 | } else { /* small tag */ |
378 | len = p[0] & 0x07; | 395 | len = p[0] & 0x07; |
379 | tag = ((p[0]>>3) & 0x0f); | 396 | tag = ((p[0] >> 3) & 0x0f); |
380 | } | 397 | } |
381 | 398 | ||
382 | switch (tag) { | 399 | switch (tag) { |
@@ -442,16 +459,19 @@ pnpbios_parse_resource_option_data(unsigned char * p, unsigned char * end, struc | |||
442 | if (len != 0) | 459 | if (len != 0) |
443 | goto len_err; | 460 | goto len_err; |
444 | if (option_independent == option) | 461 | if (option_independent == option) |
445 | printk(KERN_WARNING "PnPBIOS: Missing SMALL_TAG_STARTDEP tag\n"); | 462 | printk(KERN_WARNING |
463 | "PnPBIOS: Missing SMALL_TAG_STARTDEP tag\n"); | ||
446 | option = option_independent; | 464 | option = option_independent; |
447 | break; | 465 | break; |
448 | 466 | ||
449 | case SMALL_TAG_END: | 467 | case SMALL_TAG_END: |
450 | return p + 2; | 468 | return p + 2; |
451 | 469 | ||
452 | default: /* an unkown tag */ | 470 | default: /* an unkown tag */ |
453 | len_err: | 471 | len_err: |
454 | printk(KERN_ERR "PnPBIOS: Unknown tag '0x%x', length '%d'.\n", tag, len); | 472 | printk(KERN_ERR |
473 | "PnPBIOS: Unknown tag '0x%x', length '%d'.\n", | ||
474 | tag, len); | ||
455 | break; | 475 | break; |
456 | } | 476 | } |
457 | 477 | ||
@@ -462,19 +482,18 @@ pnpbios_parse_resource_option_data(unsigned char * p, unsigned char * end, struc | |||
462 | p += len + 1; | 482 | p += len + 1; |
463 | } | 483 | } |
464 | 484 | ||
465 | printk(KERN_ERR "PnPBIOS: Resource structure does not contain an end tag.\n"); | 485 | printk(KERN_ERR |
486 | "PnPBIOS: Resource structure does not contain an end tag.\n"); | ||
466 | 487 | ||
467 | return NULL; | 488 | return NULL; |
468 | } | 489 | } |
469 | 490 | ||
470 | |||
471 | /* | 491 | /* |
472 | * Compatible Device IDs | 492 | * Compatible Device IDs |
473 | */ | 493 | */ |
474 | 494 | ||
475 | #define HEX(id,a) hex[((id)>>a) & 15] | 495 | #define HEX(id,a) hex[((id)>>a) & 15] |
476 | #define CHAR(id,a) (0x40 + (((id)>>a) & 31)) | 496 | #define CHAR(id,a) (0x40 + (((id)>>a) & 31)) |
477 | // | ||
478 | 497 | ||
479 | void pnpid32_to_pnpid(u32 id, char *str) | 498 | void pnpid32_to_pnpid(u32 id, char *str) |
480 | { | 499 | { |
@@ -483,21 +502,20 @@ void pnpid32_to_pnpid(u32 id, char *str) | |||
483 | id = be32_to_cpu(id); | 502 | id = be32_to_cpu(id); |
484 | str[0] = CHAR(id, 26); | 503 | str[0] = CHAR(id, 26); |
485 | str[1] = CHAR(id, 21); | 504 | str[1] = CHAR(id, 21); |
486 | str[2] = CHAR(id,16); | 505 | str[2] = CHAR(id, 16); |
487 | str[3] = HEX(id, 12); | 506 | str[3] = HEX(id, 12); |
488 | str[4] = HEX(id, 8); | 507 | str[4] = HEX(id, 8); |
489 | str[5] = HEX(id, 4); | 508 | str[5] = HEX(id, 4); |
490 | str[6] = HEX(id, 0); | 509 | str[6] = HEX(id, 0); |
491 | str[7] = '\0'; | 510 | str[7] = '\0'; |
492 | |||
493 | return; | ||
494 | } | 511 | } |
495 | // | 512 | |
496 | #undef CHAR | 513 | #undef CHAR |
497 | #undef HEX | 514 | #undef HEX |
498 | 515 | ||
499 | static unsigned char * | 516 | static unsigned char *pnpbios_parse_compatible_ids(unsigned char *p, |
500 | pnpbios_parse_compatible_ids(unsigned char *p, unsigned char *end, struct pnp_dev *dev) | 517 | unsigned char *end, |
518 | struct pnp_dev *dev) | ||
501 | { | 519 | { |
502 | int len, tag; | 520 | int len, tag; |
503 | char id[8]; | 521 | char id[8]; |
@@ -509,40 +527,45 @@ pnpbios_parse_compatible_ids(unsigned char *p, unsigned char *end, struct pnp_de | |||
509 | while ((char *)p < (char *)end) { | 527 | while ((char *)p < (char *)end) { |
510 | 528 | ||
511 | /* determine the type of tag */ | 529 | /* determine the type of tag */ |
512 | if (p[0] & LARGE_TAG) { /* large tag */ | 530 | if (p[0] & LARGE_TAG) { /* large tag */ |
513 | len = (p[2] << 8) | p[1]; | 531 | len = (p[2] << 8) | p[1]; |
514 | tag = p[0]; | 532 | tag = p[0]; |
515 | } else { /* small tag */ | 533 | } else { /* small tag */ |
516 | len = p[0] & 0x07; | 534 | len = p[0] & 0x07; |
517 | tag = ((p[0]>>3) & 0x0f); | 535 | tag = ((p[0] >> 3) & 0x0f); |
518 | } | 536 | } |
519 | 537 | ||
520 | switch (tag) { | 538 | switch (tag) { |
521 | 539 | ||
522 | case LARGE_TAG_ANSISTR: | 540 | case LARGE_TAG_ANSISTR: |
523 | strncpy(dev->name, p + 3, len >= PNP_NAME_LEN ? PNP_NAME_LEN - 2 : len); | 541 | strncpy(dev->name, p + 3, |
524 | dev->name[len >= PNP_NAME_LEN ? PNP_NAME_LEN - 1 : len] = '\0'; | 542 | len >= PNP_NAME_LEN ? PNP_NAME_LEN - 2 : len); |
543 | dev->name[len >= | ||
544 | PNP_NAME_LEN ? PNP_NAME_LEN - 1 : len] = '\0'; | ||
525 | break; | 545 | break; |
526 | 546 | ||
527 | case SMALL_TAG_COMPATDEVID: /* compatible ID */ | 547 | case SMALL_TAG_COMPATDEVID: /* compatible ID */ |
528 | if (len != 4) | 548 | if (len != 4) |
529 | goto len_err; | 549 | goto len_err; |
530 | dev_id = kzalloc(sizeof (struct pnp_id), GFP_KERNEL); | 550 | dev_id = kzalloc(sizeof(struct pnp_id), GFP_KERNEL); |
531 | if (!dev_id) | 551 | if (!dev_id) |
532 | return NULL; | 552 | return NULL; |
533 | pnpid32_to_pnpid(p[1] | p[2] << 8 | p[3] << 16 | p[4] << 24,id); | 553 | pnpid32_to_pnpid(p[1] | p[2] << 8 | p[3] << 16 | p[4] << |
554 | 24, id); | ||
534 | memcpy(&dev_id->id, id, 7); | 555 | memcpy(&dev_id->id, id, 7); |
535 | pnp_add_id(dev_id, dev); | 556 | pnp_add_id(dev_id, dev); |
536 | break; | 557 | break; |
537 | 558 | ||
538 | case SMALL_TAG_END: | 559 | case SMALL_TAG_END: |
539 | p = p + 2; | 560 | p = p + 2; |
540 | return (unsigned char *)p; | 561 | return (unsigned char *)p; |
541 | break; | 562 | break; |
542 | 563 | ||
543 | default: /* an unkown tag */ | 564 | default: /* an unkown tag */ |
544 | len_err: | 565 | len_err: |
545 | printk(KERN_ERR "PnPBIOS: Unknown tag '0x%x', length '%d'.\n", tag, len); | 566 | printk(KERN_ERR |
567 | "PnPBIOS: Unknown tag '0x%x', length '%d'.\n", | ||
568 | tag, len); | ||
546 | break; | 569 | break; |
547 | } | 570 | } |
548 | 571 | ||
@@ -553,33 +576,34 @@ pnpbios_parse_compatible_ids(unsigned char *p, unsigned char *end, struct pnp_de | |||
553 | p += len + 1; | 576 | p += len + 1; |
554 | } | 577 | } |
555 | 578 | ||
556 | printk(KERN_ERR "PnPBIOS: Resource structure does not contain an end tag.\n"); | 579 | printk(KERN_ERR |
580 | "PnPBIOS: Resource structure does not contain an end tag.\n"); | ||
557 | 581 | ||
558 | return NULL; | 582 | return NULL; |
559 | } | 583 | } |
560 | 584 | ||
561 | |||
562 | /* | 585 | /* |
563 | * Allocated Resource Encoding | 586 | * Allocated Resource Encoding |
564 | */ | 587 | */ |
565 | 588 | ||
566 | static void pnpbios_encode_mem(unsigned char *p, struct resource * res) | 589 | static void pnpbios_encode_mem(unsigned char *p, struct resource *res) |
567 | { | 590 | { |
568 | unsigned long base = res->start; | 591 | unsigned long base = res->start; |
569 | unsigned long len = res->end - res->start + 1; | 592 | unsigned long len = res->end - res->start + 1; |
593 | |||
570 | p[4] = (base >> 8) & 0xff; | 594 | p[4] = (base >> 8) & 0xff; |
571 | p[5] = ((base >> 8) >> 8) & 0xff; | 595 | p[5] = ((base >> 8) >> 8) & 0xff; |
572 | p[6] = (base >> 8) & 0xff; | 596 | p[6] = (base >> 8) & 0xff; |
573 | p[7] = ((base >> 8) >> 8) & 0xff; | 597 | p[7] = ((base >> 8) >> 8) & 0xff; |
574 | p[10] = (len >> 8) & 0xff; | 598 | p[10] = (len >> 8) & 0xff; |
575 | p[11] = ((len >> 8) >> 8) & 0xff; | 599 | p[11] = ((len >> 8) >> 8) & 0xff; |
576 | return; | ||
577 | } | 600 | } |
578 | 601 | ||
579 | static void pnpbios_encode_mem32(unsigned char *p, struct resource * res) | 602 | static void pnpbios_encode_mem32(unsigned char *p, struct resource *res) |
580 | { | 603 | { |
581 | unsigned long base = res->start; | 604 | unsigned long base = res->start; |
582 | unsigned long len = res->end - res->start + 1; | 605 | unsigned long len = res->end - res->start + 1; |
606 | |||
583 | p[4] = base & 0xff; | 607 | p[4] = base & 0xff; |
584 | p[5] = (base >> 8) & 0xff; | 608 | p[5] = (base >> 8) & 0xff; |
585 | p[6] = (base >> 16) & 0xff; | 609 | p[6] = (base >> 16) & 0xff; |
@@ -592,12 +616,13 @@ static void pnpbios_encode_mem32(unsigned char *p, struct resource * res) | |||
592 | p[17] = (len >> 8) & 0xff; | 616 | p[17] = (len >> 8) & 0xff; |
593 | p[18] = (len >> 16) & 0xff; | 617 | p[18] = (len >> 16) & 0xff; |
594 | p[19] = (len >> 24) & 0xff; | 618 | p[19] = (len >> 24) & 0xff; |
595 | return; | ||
596 | } | 619 | } |
597 | 620 | ||
598 | static void pnpbios_encode_fixed_mem32(unsigned char *p, struct resource * res) | 621 | static void pnpbios_encode_fixed_mem32(unsigned char *p, struct resource *res) |
599 | { unsigned long base = res->start; | 622 | { |
623 | unsigned long base = res->start; | ||
600 | unsigned long len = res->end - res->start + 1; | 624 | unsigned long len = res->end - res->start + 1; |
625 | |||
601 | p[4] = base & 0xff; | 626 | p[4] = base & 0xff; |
602 | p[5] = (base >> 8) & 0xff; | 627 | p[5] = (base >> 8) & 0xff; |
603 | p[6] = (base >> 16) & 0xff; | 628 | p[6] = (base >> 16) & 0xff; |
@@ -606,50 +631,52 @@ static void pnpbios_encode_fixed_mem32(unsigned char *p, struct resource * res) | |||
606 | p[9] = (len >> 8) & 0xff; | 631 | p[9] = (len >> 8) & 0xff; |
607 | p[10] = (len >> 16) & 0xff; | 632 | p[10] = (len >> 16) & 0xff; |
608 | p[11] = (len >> 24) & 0xff; | 633 | p[11] = (len >> 24) & 0xff; |
609 | return; | ||
610 | } | 634 | } |
611 | 635 | ||
612 | static void pnpbios_encode_irq(unsigned char *p, struct resource * res) | 636 | static void pnpbios_encode_irq(unsigned char *p, struct resource *res) |
613 | { | 637 | { |
614 | unsigned long map = 0; | 638 | unsigned long map = 0; |
639 | |||
615 | map = 1 << res->start; | 640 | map = 1 << res->start; |
616 | p[1] = map & 0xff; | 641 | p[1] = map & 0xff; |
617 | p[2] = (map >> 8) & 0xff; | 642 | p[2] = (map >> 8) & 0xff; |
618 | return; | ||
619 | } | 643 | } |
620 | 644 | ||
621 | static void pnpbios_encode_dma(unsigned char *p, struct resource * res) | 645 | static void pnpbios_encode_dma(unsigned char *p, struct resource *res) |
622 | { | 646 | { |
623 | unsigned long map = 0; | 647 | unsigned long map = 0; |
648 | |||
624 | map = 1 << res->start; | 649 | map = 1 << res->start; |
625 | p[1] = map & 0xff; | 650 | p[1] = map & 0xff; |
626 | return; | ||
627 | } | 651 | } |
628 | 652 | ||
629 | static void pnpbios_encode_port(unsigned char *p, struct resource * res) | 653 | static void pnpbios_encode_port(unsigned char *p, struct resource *res) |
630 | { | 654 | { |
631 | unsigned long base = res->start; | 655 | unsigned long base = res->start; |
632 | unsigned long len = res->end - res->start + 1; | 656 | unsigned long len = res->end - res->start + 1; |
657 | |||
633 | p[2] = base & 0xff; | 658 | p[2] = base & 0xff; |
634 | p[3] = (base >> 8) & 0xff; | 659 | p[3] = (base >> 8) & 0xff; |
635 | p[4] = base & 0xff; | 660 | p[4] = base & 0xff; |
636 | p[5] = (base >> 8) & 0xff; | 661 | p[5] = (base >> 8) & 0xff; |
637 | p[7] = len & 0xff; | 662 | p[7] = len & 0xff; |
638 | return; | ||
639 | } | 663 | } |
640 | 664 | ||
641 | static void pnpbios_encode_fixed_port(unsigned char *p, struct resource * res) | 665 | static void pnpbios_encode_fixed_port(unsigned char *p, struct resource *res) |
642 | { | 666 | { |
643 | unsigned long base = res->start; | 667 | unsigned long base = res->start; |
644 | unsigned long len = res->end - res->start + 1; | 668 | unsigned long len = res->end - res->start + 1; |
669 | |||
645 | p[1] = base & 0xff; | 670 | p[1] = base & 0xff; |
646 | p[2] = (base >> 8) & 0xff; | 671 | p[2] = (base >> 8) & 0xff; |
647 | p[3] = len & 0xff; | 672 | p[3] = len & 0xff; |
648 | return; | ||
649 | } | 673 | } |
650 | 674 | ||
651 | static unsigned char * | 675 | static unsigned char *pnpbios_encode_allocated_resource_data(unsigned char *p, |
652 | pnpbios_encode_allocated_resource_data(unsigned char * p, unsigned char * end, struct pnp_resource_table * res) | 676 | unsigned char *end, |
677 | struct | ||
678 | pnp_resource_table | ||
679 | *res) | ||
653 | { | 680 | { |
654 | unsigned int len, tag; | 681 | unsigned int len, tag; |
655 | int port = 0, irq = 0, dma = 0, mem = 0; | 682 | int port = 0, irq = 0, dma = 0, mem = 0; |
@@ -660,12 +687,12 @@ pnpbios_encode_allocated_resource_data(unsigned char * p, unsigned char * end, s | |||
660 | while ((char *)p < (char *)end) { | 687 | while ((char *)p < (char *)end) { |
661 | 688 | ||
662 | /* determine the type of tag */ | 689 | /* determine the type of tag */ |
663 | if (p[0] & LARGE_TAG) { /* large tag */ | 690 | if (p[0] & LARGE_TAG) { /* large tag */ |
664 | len = (p[2] << 8) | p[1]; | 691 | len = (p[2] << 8) | p[1]; |
665 | tag = p[0]; | 692 | tag = p[0]; |
666 | } else { /* small tag */ | 693 | } else { /* small tag */ |
667 | len = p[0] & 0x07; | 694 | len = p[0] & 0x07; |
668 | tag = ((p[0]>>3) & 0x0f); | 695 | tag = ((p[0] >> 3) & 0x0f); |
669 | } | 696 | } |
670 | 697 | ||
671 | switch (tag) { | 698 | switch (tag) { |
@@ -725,12 +752,14 @@ pnpbios_encode_allocated_resource_data(unsigned char * p, unsigned char * end, s | |||
725 | 752 | ||
726 | case SMALL_TAG_END: | 753 | case SMALL_TAG_END: |
727 | p = p + 2; | 754 | p = p + 2; |
728 | return (unsigned char *)p; | 755 | return (unsigned char *)p; |
729 | break; | 756 | break; |
730 | 757 | ||
731 | default: /* an unkown tag */ | 758 | default: /* an unkown tag */ |
732 | len_err: | 759 | len_err: |
733 | printk(KERN_ERR "PnPBIOS: Unknown tag '0x%x', length '%d'.\n", tag, len); | 760 | printk(KERN_ERR |
761 | "PnPBIOS: Unknown tag '0x%x', length '%d'.\n", | ||
762 | tag, len); | ||
734 | break; | 763 | break; |
735 | } | 764 | } |
736 | 765 | ||
@@ -741,52 +770,52 @@ pnpbios_encode_allocated_resource_data(unsigned char * p, unsigned char * end, s | |||
741 | p += len + 1; | 770 | p += len + 1; |
742 | } | 771 | } |
743 | 772 | ||
744 | printk(KERN_ERR "PnPBIOS: Resource structure does not contain an end tag.\n"); | 773 | printk(KERN_ERR |
774 | "PnPBIOS: Resource structure does not contain an end tag.\n"); | ||
745 | 775 | ||
746 | return NULL; | 776 | return NULL; |
747 | } | 777 | } |
748 | 778 | ||
749 | |||
750 | /* | 779 | /* |
751 | * Core Parsing Functions | 780 | * Core Parsing Functions |
752 | */ | 781 | */ |
753 | 782 | ||
754 | int | 783 | int pnpbios_parse_data_stream(struct pnp_dev *dev, struct pnp_bios_node *node) |
755 | pnpbios_parse_data_stream(struct pnp_dev *dev, struct pnp_bios_node * node) | ||
756 | { | 784 | { |
757 | unsigned char * p = (char *)node->data; | 785 | unsigned char *p = (char *)node->data; |
758 | unsigned char * end = (char *)(node->data + node->size); | 786 | unsigned char *end = (char *)(node->data + node->size); |
759 | p = pnpbios_parse_allocated_resource_data(p,end,&dev->res); | 787 | |
788 | p = pnpbios_parse_allocated_resource_data(p, end, &dev->res); | ||
760 | if (!p) | 789 | if (!p) |
761 | return -EIO; | 790 | return -EIO; |
762 | p = pnpbios_parse_resource_option_data(p,end,dev); | 791 | p = pnpbios_parse_resource_option_data(p, end, dev); |
763 | if (!p) | 792 | if (!p) |
764 | return -EIO; | 793 | return -EIO; |
765 | p = pnpbios_parse_compatible_ids(p,end,dev); | 794 | p = pnpbios_parse_compatible_ids(p, end, dev); |
766 | if (!p) | 795 | if (!p) |
767 | return -EIO; | 796 | return -EIO; |
768 | return 0; | 797 | return 0; |
769 | } | 798 | } |
770 | 799 | ||
771 | int | 800 | int pnpbios_read_resources_from_node(struct pnp_resource_table *res, |
772 | pnpbios_read_resources_from_node(struct pnp_resource_table *res, | 801 | struct pnp_bios_node *node) |
773 | struct pnp_bios_node * node) | ||
774 | { | 802 | { |
775 | unsigned char * p = (char *)node->data; | 803 | unsigned char *p = (char *)node->data; |
776 | unsigned char * end = (char *)(node->data + node->size); | 804 | unsigned char *end = (char *)(node->data + node->size); |
777 | p = pnpbios_parse_allocated_resource_data(p,end,res); | 805 | |
806 | p = pnpbios_parse_allocated_resource_data(p, end, res); | ||
778 | if (!p) | 807 | if (!p) |
779 | return -EIO; | 808 | return -EIO; |
780 | return 0; | 809 | return 0; |
781 | } | 810 | } |
782 | 811 | ||
783 | int | 812 | int pnpbios_write_resources_to_node(struct pnp_resource_table *res, |
784 | pnpbios_write_resources_to_node(struct pnp_resource_table *res, | 813 | struct pnp_bios_node *node) |
785 | struct pnp_bios_node * node) | ||
786 | { | 814 | { |
787 | unsigned char * p = (char *)node->data; | 815 | unsigned char *p = (char *)node->data; |
788 | unsigned char * end = (char *)(node->data + node->size); | 816 | unsigned char *end = (char *)(node->data + node->size); |
789 | p = pnpbios_encode_allocated_resource_data(p,end,res); | 817 | |
818 | p = pnpbios_encode_allocated_resource_data(p, end, res); | ||
790 | if (!p) | 819 | if (!p) |
791 | return -EIO; | 820 | return -EIO; |
792 | return 0; | 821 | return 0; |
diff --git a/drivers/pnp/quirks.c b/drivers/pnp/quirks.c index 7c3236690cc3..90755d4cdb9f 100644 --- a/drivers/pnp/quirks.c +++ b/drivers/pnp/quirks.c | |||
@@ -19,7 +19,6 @@ | |||
19 | #include <linux/io.h> | 19 | #include <linux/io.h> |
20 | #include "base.h" | 20 | #include "base.h" |
21 | 21 | ||
22 | |||
23 | static void quirk_awe32_resources(struct pnp_dev *dev) | 22 | static void quirk_awe32_resources(struct pnp_dev *dev) |
24 | { | 23 | { |
25 | struct pnp_port *port, *port2, *port3; | 24 | struct pnp_port *port, *port2, *port3; |
@@ -31,7 +30,7 @@ static void quirk_awe32_resources(struct pnp_dev *dev) | |||
31 | * two extra ports (at offset 0x400 and 0x800 from the one given) by | 30 | * two extra ports (at offset 0x400 and 0x800 from the one given) by |
32 | * hand. | 31 | * hand. |
33 | */ | 32 | */ |
34 | for ( ; res ; res = res->next ) { | 33 | for (; res; res = res->next) { |
35 | port2 = pnp_alloc(sizeof(struct pnp_port)); | 34 | port2 = pnp_alloc(sizeof(struct pnp_port)); |
36 | if (!port2) | 35 | if (!port2) |
37 | return; | 36 | return; |
@@ -58,18 +57,19 @@ static void quirk_cmi8330_resources(struct pnp_dev *dev) | |||
58 | struct pnp_option *res = dev->dependent; | 57 | struct pnp_option *res = dev->dependent; |
59 | unsigned long tmp; | 58 | unsigned long tmp; |
60 | 59 | ||
61 | for ( ; res ; res = res->next ) { | 60 | for (; res; res = res->next) { |
62 | 61 | ||
63 | struct pnp_irq *irq; | 62 | struct pnp_irq *irq; |
64 | struct pnp_dma *dma; | 63 | struct pnp_dma *dma; |
65 | 64 | ||
66 | for( irq = res->irq; irq; irq = irq->next ) { // Valid irqs are 5, 7, 10 | 65 | for (irq = res->irq; irq; irq = irq->next) { // Valid irqs are 5, 7, 10 |
67 | tmp = 0x04A0; | 66 | tmp = 0x04A0; |
68 | bitmap_copy(irq->map, &tmp, 16); // 0000 0100 1010 0000 | 67 | bitmap_copy(irq->map, &tmp, 16); // 0000 0100 1010 0000 |
69 | } | 68 | } |
70 | 69 | ||
71 | for( dma = res->dma; dma; dma = dma->next ) // Valid 8bit dma channels are 1,3 | 70 | for (dma = res->dma; dma; dma = dma->next) // Valid 8bit dma channels are 1,3 |
72 | if( ( dma->flags & IORESOURCE_DMA_TYPE_MASK ) == IORESOURCE_DMA_8BIT ) | 71 | if ((dma->flags & IORESOURCE_DMA_TYPE_MASK) == |
72 | IORESOURCE_DMA_8BIT) | ||
73 | dma->map = 0x000A; | 73 | dma->map = 0x000A; |
74 | } | 74 | } |
75 | printk(KERN_INFO "pnp: CMI8330 quirk - fixing interrupts and dma\n"); | 75 | printk(KERN_INFO "pnp: CMI8330 quirk - fixing interrupts and dma\n"); |
@@ -79,7 +79,7 @@ static void quirk_sb16audio_resources(struct pnp_dev *dev) | |||
79 | { | 79 | { |
80 | struct pnp_port *port; | 80 | struct pnp_port *port; |
81 | struct pnp_option *res = dev->dependent; | 81 | struct pnp_option *res = dev->dependent; |
82 | int changed = 0; | 82 | int changed = 0; |
83 | 83 | ||
84 | /* | 84 | /* |
85 | * The default range on the mpu port for these devices is 0x388-0x388. | 85 | * The default range on the mpu port for these devices is 0x388-0x388. |
@@ -87,24 +87,24 @@ static void quirk_sb16audio_resources(struct pnp_dev *dev) | |||
87 | * auto-configured. | 87 | * auto-configured. |
88 | */ | 88 | */ |
89 | 89 | ||
90 | for( ; res ; res = res->next ) { | 90 | for (; res; res = res->next) { |
91 | port = res->port; | 91 | port = res->port; |
92 | if(!port) | 92 | if (!port) |
93 | continue; | 93 | continue; |
94 | port = port->next; | 94 | port = port->next; |
95 | if(!port) | 95 | if (!port) |
96 | continue; | 96 | continue; |
97 | port = port->next; | 97 | port = port->next; |
98 | if(!port) | 98 | if (!port) |
99 | continue; | 99 | continue; |
100 | if(port->min != port->max) | 100 | if (port->min != port->max) |
101 | continue; | 101 | continue; |
102 | port->max += 0x70; | 102 | port->max += 0x70; |
103 | changed = 1; | 103 | changed = 1; |
104 | } | 104 | } |
105 | if(changed) | 105 | if (changed) |
106 | printk(KERN_INFO "pnp: SB audio device quirk - increasing port range\n"); | 106 | printk(KERN_INFO |
107 | return; | 107 | "pnp: SB audio device quirk - increasing port range\n"); |
108 | } | 108 | } |
109 | 109 | ||
110 | static int quirk_smc_fir_enabled(struct pnp_dev *dev) | 110 | static int quirk_smc_fir_enabled(struct pnp_dev *dev) |
@@ -124,7 +124,7 @@ static int quirk_smc_fir_enabled(struct pnp_dev *dev) | |||
124 | outb(bank, firbase + 7); | 124 | outb(bank, firbase + 7); |
125 | 125 | ||
126 | high = inb(firbase + 0); | 126 | high = inb(firbase + 0); |
127 | low = inb(firbase + 1); | 127 | low = inb(firbase + 1); |
128 | chip = inb(firbase + 2); | 128 | chip = inb(firbase + 2); |
129 | 129 | ||
130 | /* This corresponds to the check in smsc_ircc_present() */ | 130 | /* This corresponds to the check in smsc_ircc_present() */ |
@@ -153,8 +153,8 @@ static void quirk_smc_enable(struct pnp_dev *dev) | |||
153 | */ | 153 | */ |
154 | dev_err(&dev->dev, "%s not responding at SIR 0x%lx, FIR 0x%lx; " | 154 | dev_err(&dev->dev, "%s not responding at SIR 0x%lx, FIR 0x%lx; " |
155 | "auto-configuring\n", dev->id->id, | 155 | "auto-configuring\n", dev->id->id, |
156 | (unsigned long) pnp_port_start(dev, 0), | 156 | (unsigned long)pnp_port_start(dev, 0), |
157 | (unsigned long) pnp_port_start(dev, 1)); | 157 | (unsigned long)pnp_port_start(dev, 1)); |
158 | 158 | ||
159 | pnp_disable_dev(dev); | 159 | pnp_disable_dev(dev); |
160 | pnp_init_resource_table(&dev->res); | 160 | pnp_init_resource_table(&dev->res); |
@@ -162,8 +162,8 @@ static void quirk_smc_enable(struct pnp_dev *dev) | |||
162 | pnp_activate_dev(dev); | 162 | pnp_activate_dev(dev); |
163 | if (quirk_smc_fir_enabled(dev)) { | 163 | if (quirk_smc_fir_enabled(dev)) { |
164 | dev_err(&dev->dev, "responds at SIR 0x%lx, FIR 0x%lx\n", | 164 | dev_err(&dev->dev, "responds at SIR 0x%lx, FIR 0x%lx\n", |
165 | (unsigned long) pnp_port_start(dev, 0), | 165 | (unsigned long)pnp_port_start(dev, 0), |
166 | (unsigned long) pnp_port_start(dev, 1)); | 166 | (unsigned long)pnp_port_start(dev, 1)); |
167 | return; | 167 | return; |
168 | } | 168 | } |
169 | 169 | ||
@@ -175,8 +175,8 @@ static void quirk_smc_enable(struct pnp_dev *dev) | |||
175 | */ | 175 | */ |
176 | dev_err(&dev->dev, "not responding at SIR 0x%lx, FIR 0x%lx; " | 176 | dev_err(&dev->dev, "not responding at SIR 0x%lx, FIR 0x%lx; " |
177 | "swapping SIR/FIR and reconfiguring\n", | 177 | "swapping SIR/FIR and reconfiguring\n", |
178 | (unsigned long) pnp_port_start(dev, 0), | 178 | (unsigned long)pnp_port_start(dev, 0), |
179 | (unsigned long) pnp_port_start(dev, 1)); | 179 | (unsigned long)pnp_port_start(dev, 1)); |
180 | 180 | ||
181 | /* | 181 | /* |
182 | * Clear IORESOURCE_AUTO so pnp_activate_dev() doesn't reassign | 182 | * Clear IORESOURCE_AUTO so pnp_activate_dev() doesn't reassign |
@@ -200,8 +200,8 @@ static void quirk_smc_enable(struct pnp_dev *dev) | |||
200 | 200 | ||
201 | if (quirk_smc_fir_enabled(dev)) { | 201 | if (quirk_smc_fir_enabled(dev)) { |
202 | dev_err(&dev->dev, "responds at SIR 0x%lx, FIR 0x%lx\n", | 202 | dev_err(&dev->dev, "responds at SIR 0x%lx, FIR 0x%lx\n", |
203 | (unsigned long) pnp_port_start(dev, 0), | 203 | (unsigned long)pnp_port_start(dev, 0), |
204 | (unsigned long) pnp_port_start(dev, 1)); | 204 | (unsigned long)pnp_port_start(dev, 1)); |
205 | return; | 205 | return; |
206 | } | 206 | } |
207 | 207 | ||
@@ -209,7 +209,6 @@ static void quirk_smc_enable(struct pnp_dev *dev) | |||
209 | "email bjorn.helgaas@hp.com\n"); | 209 | "email bjorn.helgaas@hp.com\n"); |
210 | } | 210 | } |
211 | 211 | ||
212 | |||
213 | /* | 212 | /* |
214 | * PnP Quirks | 213 | * PnP Quirks |
215 | * Cards or devices that need some tweaking due to incomplete resource info | 214 | * Cards or devices that need some tweaking due to incomplete resource info |
@@ -217,21 +216,21 @@ static void quirk_smc_enable(struct pnp_dev *dev) | |||
217 | 216 | ||
218 | static struct pnp_fixup pnp_fixups[] = { | 217 | static struct pnp_fixup pnp_fixups[] = { |
219 | /* Soundblaster awe io port quirk */ | 218 | /* Soundblaster awe io port quirk */ |
220 | { "CTL0021", quirk_awe32_resources }, | 219 | {"CTL0021", quirk_awe32_resources}, |
221 | { "CTL0022", quirk_awe32_resources }, | 220 | {"CTL0022", quirk_awe32_resources}, |
222 | { "CTL0023", quirk_awe32_resources }, | 221 | {"CTL0023", quirk_awe32_resources}, |
223 | /* CMI 8330 interrupt and dma fix */ | 222 | /* CMI 8330 interrupt and dma fix */ |
224 | { "@X@0001", quirk_cmi8330_resources }, | 223 | {"@X@0001", quirk_cmi8330_resources}, |
225 | /* Soundblaster audio device io port range quirk */ | 224 | /* Soundblaster audio device io port range quirk */ |
226 | { "CTL0001", quirk_sb16audio_resources }, | 225 | {"CTL0001", quirk_sb16audio_resources}, |
227 | { "CTL0031", quirk_sb16audio_resources }, | 226 | {"CTL0031", quirk_sb16audio_resources}, |
228 | { "CTL0041", quirk_sb16audio_resources }, | 227 | {"CTL0041", quirk_sb16audio_resources}, |
229 | { "CTL0042", quirk_sb16audio_resources }, | 228 | {"CTL0042", quirk_sb16audio_resources}, |
230 | { "CTL0043", quirk_sb16audio_resources }, | 229 | {"CTL0043", quirk_sb16audio_resources}, |
231 | { "CTL0044", quirk_sb16audio_resources }, | 230 | {"CTL0044", quirk_sb16audio_resources}, |
232 | { "CTL0045", quirk_sb16audio_resources }, | 231 | {"CTL0045", quirk_sb16audio_resources}, |
233 | { "SMCf010", quirk_smc_enable }, | 232 | {"SMCf010", quirk_smc_enable}, |
234 | { "" } | 233 | {""} |
235 | }; | 234 | }; |
236 | 235 | ||
237 | void pnp_fixup_device(struct pnp_dev *dev) | 236 | void pnp_fixup_device(struct pnp_dev *dev) |
@@ -239,9 +238,8 @@ void pnp_fixup_device(struct pnp_dev *dev) | |||
239 | int i = 0; | 238 | int i = 0; |
240 | 239 | ||
241 | while (*pnp_fixups[i].id) { | 240 | while (*pnp_fixups[i].id) { |
242 | if (compare_pnp_id(dev->id,pnp_fixups[i].id)) { | 241 | if (compare_pnp_id(dev->id, pnp_fixups[i].id)) { |
243 | pnp_dbg("Calling quirk for %s", | 242 | pnp_dbg("Calling quirk for %s", dev->dev.bus_id); |
244 | dev->dev.bus_id); | ||
245 | pnp_fixups[i].quirk_function(dev); | 243 | pnp_fixups[i].quirk_function(dev); |
246 | } | 244 | } |
247 | i++; | 245 | i++; |
diff --git a/drivers/pnp/resource.c b/drivers/pnp/resource.c index a685fbec4604..ea6ec14a0559 100644 --- a/drivers/pnp/resource.c +++ b/drivers/pnp/resource.c | |||
@@ -3,7 +3,6 @@ | |||
3 | * | 3 | * |
4 | * based on isapnp.c resource management (c) Jaroslav Kysela <perex@suse.cz> | 4 | * based on isapnp.c resource management (c) Jaroslav Kysela <perex@suse.cz> |
5 | * Copyright 2003 Adam Belay <ambx1@neo.rr.com> | 5 | * Copyright 2003 Adam Belay <ambx1@neo.rr.com> |
6 | * | ||
7 | */ | 6 | */ |
8 | 7 | ||
9 | #include <linux/module.h> | 8 | #include <linux/module.h> |
@@ -20,21 +19,19 @@ | |||
20 | #include <linux/pnp.h> | 19 | #include <linux/pnp.h> |
21 | #include "base.h" | 20 | #include "base.h" |
22 | 21 | ||
23 | static int pnp_reserve_irq[16] = { [0 ... 15] = -1 }; /* reserve (don't use) some IRQ */ | 22 | static int pnp_reserve_irq[16] = {[0 ... 15] = -1 }; /* reserve (don't use) some IRQ */ |
24 | static int pnp_reserve_dma[8] = { [0 ... 7] = -1 }; /* reserve (don't use) some DMA */ | 23 | static int pnp_reserve_dma[8] = {[0 ... 7] = -1 }; /* reserve (don't use) some DMA */ |
25 | static int pnp_reserve_io[16] = { [0 ... 15] = -1 }; /* reserve (don't use) some I/O region */ | 24 | static int pnp_reserve_io[16] = {[0 ... 15] = -1 }; /* reserve (don't use) some I/O region */ |
26 | static int pnp_reserve_mem[16] = { [0 ... 15] = -1 }; /* reserve (don't use) some memory region */ | 25 | static int pnp_reserve_mem[16] = {[0 ... 15] = -1 }; /* reserve (don't use) some memory region */ |
27 | |||
28 | 26 | ||
29 | /* | 27 | /* |
30 | * option registration | 28 | * option registration |
31 | */ | 29 | */ |
32 | 30 | ||
33 | static struct pnp_option * pnp_build_option(int priority) | 31 | static struct pnp_option *pnp_build_option(int priority) |
34 | { | 32 | { |
35 | struct pnp_option *option = pnp_alloc(sizeof(struct pnp_option)); | 33 | struct pnp_option *option = pnp_alloc(sizeof(struct pnp_option)); |
36 | 34 | ||
37 | /* check if pnp_alloc ran out of memory */ | ||
38 | if (!option) | 35 | if (!option) |
39 | return NULL; | 36 | return NULL; |
40 | 37 | ||
@@ -46,9 +43,10 @@ static struct pnp_option * pnp_build_option(int priority) | |||
46 | return option; | 43 | return option; |
47 | } | 44 | } |
48 | 45 | ||
49 | struct pnp_option * pnp_register_independent_option(struct pnp_dev *dev) | 46 | struct pnp_option *pnp_register_independent_option(struct pnp_dev *dev) |
50 | { | 47 | { |
51 | struct pnp_option *option; | 48 | struct pnp_option *option; |
49 | |||
52 | if (!dev) | 50 | if (!dev) |
53 | return NULL; | 51 | return NULL; |
54 | 52 | ||
@@ -61,9 +59,11 @@ struct pnp_option * pnp_register_independent_option(struct pnp_dev *dev) | |||
61 | return option; | 59 | return option; |
62 | } | 60 | } |
63 | 61 | ||
64 | struct pnp_option * pnp_register_dependent_option(struct pnp_dev *dev, int priority) | 62 | struct pnp_option *pnp_register_dependent_option(struct pnp_dev *dev, |
63 | int priority) | ||
65 | { | 64 | { |
66 | struct pnp_option *option; | 65 | struct pnp_option *option; |
66 | |||
67 | if (!dev) | 67 | if (!dev) |
68 | return NULL; | 68 | return NULL; |
69 | 69 | ||
@@ -82,6 +82,7 @@ struct pnp_option * pnp_register_dependent_option(struct pnp_dev *dev, int prior | |||
82 | int pnp_register_irq_resource(struct pnp_option *option, struct pnp_irq *data) | 82 | int pnp_register_irq_resource(struct pnp_option *option, struct pnp_irq *data) |
83 | { | 83 | { |
84 | struct pnp_irq *ptr; | 84 | struct pnp_irq *ptr; |
85 | |||
85 | if (!option) | 86 | if (!option) |
86 | return -EINVAL; | 87 | return -EINVAL; |
87 | if (!data) | 88 | if (!data) |
@@ -110,6 +111,7 @@ int pnp_register_irq_resource(struct pnp_option *option, struct pnp_irq *data) | |||
110 | int pnp_register_dma_resource(struct pnp_option *option, struct pnp_dma *data) | 111 | int pnp_register_dma_resource(struct pnp_option *option, struct pnp_dma *data) |
111 | { | 112 | { |
112 | struct pnp_dma *ptr; | 113 | struct pnp_dma *ptr; |
114 | |||
113 | if (!option) | 115 | if (!option) |
114 | return -EINVAL; | 116 | return -EINVAL; |
115 | if (!data) | 117 | if (!data) |
@@ -129,6 +131,7 @@ int pnp_register_dma_resource(struct pnp_option *option, struct pnp_dma *data) | |||
129 | int pnp_register_port_resource(struct pnp_option *option, struct pnp_port *data) | 131 | int pnp_register_port_resource(struct pnp_option *option, struct pnp_port *data) |
130 | { | 132 | { |
131 | struct pnp_port *ptr; | 133 | struct pnp_port *ptr; |
134 | |||
132 | if (!option) | 135 | if (!option) |
133 | return -EINVAL; | 136 | return -EINVAL; |
134 | if (!data) | 137 | if (!data) |
@@ -148,6 +151,7 @@ int pnp_register_port_resource(struct pnp_option *option, struct pnp_port *data) | |||
148 | int pnp_register_mem_resource(struct pnp_option *option, struct pnp_mem *data) | 151 | int pnp_register_mem_resource(struct pnp_option *option, struct pnp_mem *data) |
149 | { | 152 | { |
150 | struct pnp_mem *ptr; | 153 | struct pnp_mem *ptr; |
154 | |||
151 | if (!option) | 155 | if (!option) |
152 | return -EINVAL; | 156 | return -EINVAL; |
153 | if (!data) | 157 | if (!data) |
@@ -222,7 +226,6 @@ void pnp_free_option(struct pnp_option *option) | |||
222 | } | 226 | } |
223 | } | 227 | } |
224 | 228 | ||
225 | |||
226 | /* | 229 | /* |
227 | * resource validity checking | 230 | * resource validity checking |
228 | */ | 231 | */ |
@@ -236,11 +239,12 @@ void pnp_free_option(struct pnp_option *option) | |||
236 | #define cannot_compare(flags) \ | 239 | #define cannot_compare(flags) \ |
237 | ((flags) & (IORESOURCE_UNSET | IORESOURCE_DISABLED)) | 240 | ((flags) & (IORESOURCE_UNSET | IORESOURCE_DISABLED)) |
238 | 241 | ||
239 | int pnp_check_port(struct pnp_dev * dev, int idx) | 242 | int pnp_check_port(struct pnp_dev *dev, int idx) |
240 | { | 243 | { |
241 | int tmp; | 244 | int tmp; |
242 | struct pnp_dev *tdev; | 245 | struct pnp_dev *tdev; |
243 | resource_size_t *port, *end, *tport, *tend; | 246 | resource_size_t *port, *end, *tport, *tend; |
247 | |||
244 | port = &dev->res.port_resource[idx].start; | 248 | port = &dev->res.port_resource[idx].start; |
245 | end = &dev->res.port_resource[idx].end; | 249 | end = &dev->res.port_resource[idx].end; |
246 | 250 | ||
@@ -250,8 +254,8 @@ int pnp_check_port(struct pnp_dev * dev, int idx) | |||
250 | 254 | ||
251 | /* check if the resource is already in use, skip if the | 255 | /* check if the resource is already in use, skip if the |
252 | * device is active because it itself may be in use */ | 256 | * device is active because it itself may be in use */ |
253 | if(!dev->active) { | 257 | if (!dev->active) { |
254 | if (__check_region(&ioport_resource, *port, length(port,end))) | 258 | if (__check_region(&ioport_resource, *port, length(port, end))) |
255 | return 0; | 259 | return 0; |
256 | } | 260 | } |
257 | 261 | ||
@@ -259,7 +263,7 @@ int pnp_check_port(struct pnp_dev * dev, int idx) | |||
259 | for (tmp = 0; tmp < 8; tmp++) { | 263 | for (tmp = 0; tmp < 8; tmp++) { |
260 | int rport = pnp_reserve_io[tmp << 1]; | 264 | int rport = pnp_reserve_io[tmp << 1]; |
261 | int rend = pnp_reserve_io[(tmp << 1) + 1] + rport - 1; | 265 | int rend = pnp_reserve_io[(tmp << 1) + 1] + rport - 1; |
262 | if (ranged_conflict(port,end,&rport,&rend)) | 266 | if (ranged_conflict(port, end, &rport, &rend)) |
263 | return 0; | 267 | return 0; |
264 | } | 268 | } |
265 | 269 | ||
@@ -268,7 +272,7 @@ int pnp_check_port(struct pnp_dev * dev, int idx) | |||
268 | if (dev->res.port_resource[tmp].flags & IORESOURCE_IO) { | 272 | if (dev->res.port_resource[tmp].flags & IORESOURCE_IO) { |
269 | tport = &dev->res.port_resource[tmp].start; | 273 | tport = &dev->res.port_resource[tmp].start; |
270 | tend = &dev->res.port_resource[tmp].end; | 274 | tend = &dev->res.port_resource[tmp].end; |
271 | if (ranged_conflict(port,end,tport,tend)) | 275 | if (ranged_conflict(port, end, tport, tend)) |
272 | return 0; | 276 | return 0; |
273 | } | 277 | } |
274 | } | 278 | } |
@@ -279,11 +283,12 @@ int pnp_check_port(struct pnp_dev * dev, int idx) | |||
279 | continue; | 283 | continue; |
280 | for (tmp = 0; tmp < PNP_MAX_PORT; tmp++) { | 284 | for (tmp = 0; tmp < PNP_MAX_PORT; tmp++) { |
281 | if (tdev->res.port_resource[tmp].flags & IORESOURCE_IO) { | 285 | if (tdev->res.port_resource[tmp].flags & IORESOURCE_IO) { |
282 | if (cannot_compare(tdev->res.port_resource[tmp].flags)) | 286 | if (cannot_compare |
287 | (tdev->res.port_resource[tmp].flags)) | ||
283 | continue; | 288 | continue; |
284 | tport = &tdev->res.port_resource[tmp].start; | 289 | tport = &tdev->res.port_resource[tmp].start; |
285 | tend = &tdev->res.port_resource[tmp].end; | 290 | tend = &tdev->res.port_resource[tmp].end; |
286 | if (ranged_conflict(port,end,tport,tend)) | 291 | if (ranged_conflict(port, end, tport, tend)) |
287 | return 0; | 292 | return 0; |
288 | } | 293 | } |
289 | } | 294 | } |
@@ -292,11 +297,12 @@ int pnp_check_port(struct pnp_dev * dev, int idx) | |||
292 | return 1; | 297 | return 1; |
293 | } | 298 | } |
294 | 299 | ||
295 | int pnp_check_mem(struct pnp_dev * dev, int idx) | 300 | int pnp_check_mem(struct pnp_dev *dev, int idx) |
296 | { | 301 | { |
297 | int tmp; | 302 | int tmp; |
298 | struct pnp_dev *tdev; | 303 | struct pnp_dev *tdev; |
299 | resource_size_t *addr, *end, *taddr, *tend; | 304 | resource_size_t *addr, *end, *taddr, *tend; |
305 | |||
300 | addr = &dev->res.mem_resource[idx].start; | 306 | addr = &dev->res.mem_resource[idx].start; |
301 | end = &dev->res.mem_resource[idx].end; | 307 | end = &dev->res.mem_resource[idx].end; |
302 | 308 | ||
@@ -306,8 +312,8 @@ int pnp_check_mem(struct pnp_dev * dev, int idx) | |||
306 | 312 | ||
307 | /* check if the resource is already in use, skip if the | 313 | /* check if the resource is already in use, skip if the |
308 | * device is active because it itself may be in use */ | 314 | * device is active because it itself may be in use */ |
309 | if(!dev->active) { | 315 | if (!dev->active) { |
310 | if (check_mem_region(*addr, length(addr,end))) | 316 | if (check_mem_region(*addr, length(addr, end))) |
311 | return 0; | 317 | return 0; |
312 | } | 318 | } |
313 | 319 | ||
@@ -315,7 +321,7 @@ int pnp_check_mem(struct pnp_dev * dev, int idx) | |||
315 | for (tmp = 0; tmp < 8; tmp++) { | 321 | for (tmp = 0; tmp < 8; tmp++) { |
316 | int raddr = pnp_reserve_mem[tmp << 1]; | 322 | int raddr = pnp_reserve_mem[tmp << 1]; |
317 | int rend = pnp_reserve_mem[(tmp << 1) + 1] + raddr - 1; | 323 | int rend = pnp_reserve_mem[(tmp << 1) + 1] + raddr - 1; |
318 | if (ranged_conflict(addr,end,&raddr,&rend)) | 324 | if (ranged_conflict(addr, end, &raddr, &rend)) |
319 | return 0; | 325 | return 0; |
320 | } | 326 | } |
321 | 327 | ||
@@ -324,7 +330,7 @@ int pnp_check_mem(struct pnp_dev * dev, int idx) | |||
324 | if (dev->res.mem_resource[tmp].flags & IORESOURCE_MEM) { | 330 | if (dev->res.mem_resource[tmp].flags & IORESOURCE_MEM) { |
325 | taddr = &dev->res.mem_resource[tmp].start; | 331 | taddr = &dev->res.mem_resource[tmp].start; |
326 | tend = &dev->res.mem_resource[tmp].end; | 332 | tend = &dev->res.mem_resource[tmp].end; |
327 | if (ranged_conflict(addr,end,taddr,tend)) | 333 | if (ranged_conflict(addr, end, taddr, tend)) |
328 | return 0; | 334 | return 0; |
329 | } | 335 | } |
330 | } | 336 | } |
@@ -335,11 +341,12 @@ int pnp_check_mem(struct pnp_dev * dev, int idx) | |||
335 | continue; | 341 | continue; |
336 | for (tmp = 0; tmp < PNP_MAX_MEM; tmp++) { | 342 | for (tmp = 0; tmp < PNP_MAX_MEM; tmp++) { |
337 | if (tdev->res.mem_resource[tmp].flags & IORESOURCE_MEM) { | 343 | if (tdev->res.mem_resource[tmp].flags & IORESOURCE_MEM) { |
338 | if (cannot_compare(tdev->res.mem_resource[tmp].flags)) | 344 | if (cannot_compare |
345 | (tdev->res.mem_resource[tmp].flags)) | ||
339 | continue; | 346 | continue; |
340 | taddr = &tdev->res.mem_resource[tmp].start; | 347 | taddr = &tdev->res.mem_resource[tmp].start; |
341 | tend = &tdev->res.mem_resource[tmp].end; | 348 | tend = &tdev->res.mem_resource[tmp].end; |
342 | if (ranged_conflict(addr,end,taddr,tend)) | 349 | if (ranged_conflict(addr, end, taddr, tend)) |
343 | return 0; | 350 | return 0; |
344 | } | 351 | } |
345 | } | 352 | } |
@@ -353,11 +360,11 @@ static irqreturn_t pnp_test_handler(int irq, void *dev_id) | |||
353 | return IRQ_HANDLED; | 360 | return IRQ_HANDLED; |
354 | } | 361 | } |
355 | 362 | ||
356 | int pnp_check_irq(struct pnp_dev * dev, int idx) | 363 | int pnp_check_irq(struct pnp_dev *dev, int idx) |
357 | { | 364 | { |
358 | int tmp; | 365 | int tmp; |
359 | struct pnp_dev *tdev; | 366 | struct pnp_dev *tdev; |
360 | resource_size_t * irq = &dev->res.irq_resource[idx].start; | 367 | resource_size_t *irq = &dev->res.irq_resource[idx].start; |
361 | 368 | ||
362 | /* if the resource doesn't exist, don't complain about it */ | 369 | /* if the resource doesn't exist, don't complain about it */ |
363 | if (cannot_compare(dev->res.irq_resource[idx].flags)) | 370 | if (cannot_compare(dev->res.irq_resource[idx].flags)) |
@@ -394,9 +401,9 @@ int pnp_check_irq(struct pnp_dev * dev, int idx) | |||
394 | 401 | ||
395 | /* check if the resource is already in use, skip if the | 402 | /* check if the resource is already in use, skip if the |
396 | * device is active because it itself may be in use */ | 403 | * device is active because it itself may be in use */ |
397 | if(!dev->active) { | 404 | if (!dev->active) { |
398 | if (request_irq(*irq, pnp_test_handler, | 405 | if (request_irq(*irq, pnp_test_handler, |
399 | IRQF_DISABLED|IRQF_PROBE_SHARED, "pnp", NULL)) | 406 | IRQF_DISABLED | IRQF_PROBE_SHARED, "pnp", NULL)) |
400 | return 0; | 407 | return 0; |
401 | free_irq(*irq, NULL); | 408 | free_irq(*irq, NULL); |
402 | } | 409 | } |
@@ -407,7 +414,8 @@ int pnp_check_irq(struct pnp_dev * dev, int idx) | |||
407 | continue; | 414 | continue; |
408 | for (tmp = 0; tmp < PNP_MAX_IRQ; tmp++) { | 415 | for (tmp = 0; tmp < PNP_MAX_IRQ; tmp++) { |
409 | if (tdev->res.irq_resource[tmp].flags & IORESOURCE_IRQ) { | 416 | if (tdev->res.irq_resource[tmp].flags & IORESOURCE_IRQ) { |
410 | if (cannot_compare(tdev->res.irq_resource[tmp].flags)) | 417 | if (cannot_compare |
418 | (tdev->res.irq_resource[tmp].flags)) | ||
411 | continue; | 419 | continue; |
412 | if ((tdev->res.irq_resource[tmp].start == *irq)) | 420 | if ((tdev->res.irq_resource[tmp].start == *irq)) |
413 | return 0; | 421 | return 0; |
@@ -418,12 +426,12 @@ int pnp_check_irq(struct pnp_dev * dev, int idx) | |||
418 | return 1; | 426 | return 1; |
419 | } | 427 | } |
420 | 428 | ||
421 | int pnp_check_dma(struct pnp_dev * dev, int idx) | 429 | int pnp_check_dma(struct pnp_dev *dev, int idx) |
422 | { | 430 | { |
423 | #ifndef CONFIG_IA64 | 431 | #ifndef CONFIG_IA64 |
424 | int tmp; | 432 | int tmp; |
425 | struct pnp_dev *tdev; | 433 | struct pnp_dev *tdev; |
426 | resource_size_t * dma = &dev->res.dma_resource[idx].start; | 434 | resource_size_t *dma = &dev->res.dma_resource[idx].start; |
427 | 435 | ||
428 | /* if the resource doesn't exist, don't complain about it */ | 436 | /* if the resource doesn't exist, don't complain about it */ |
429 | if (cannot_compare(dev->res.dma_resource[idx].flags)) | 437 | if (cannot_compare(dev->res.dma_resource[idx].flags)) |
@@ -449,7 +457,7 @@ int pnp_check_dma(struct pnp_dev * dev, int idx) | |||
449 | 457 | ||
450 | /* check if the resource is already in use, skip if the | 458 | /* check if the resource is already in use, skip if the |
451 | * device is active because it itself may be in use */ | 459 | * device is active because it itself may be in use */ |
452 | if(!dev->active) { | 460 | if (!dev->active) { |
453 | if (request_dma(*dma, "pnp")) | 461 | if (request_dma(*dma, "pnp")) |
454 | return 0; | 462 | return 0; |
455 | free_dma(*dma); | 463 | free_dma(*dma); |
@@ -461,7 +469,8 @@ int pnp_check_dma(struct pnp_dev * dev, int idx) | |||
461 | continue; | 469 | continue; |
462 | for (tmp = 0; tmp < PNP_MAX_DMA; tmp++) { | 470 | for (tmp = 0; tmp < PNP_MAX_DMA; tmp++) { |
463 | if (tdev->res.dma_resource[tmp].flags & IORESOURCE_DMA) { | 471 | if (tdev->res.dma_resource[tmp].flags & IORESOURCE_DMA) { |
464 | if (cannot_compare(tdev->res.dma_resource[tmp].flags)) | 472 | if (cannot_compare |
473 | (tdev->res.dma_resource[tmp].flags)) | ||
465 | continue; | 474 | continue; |
466 | if ((tdev->res.dma_resource[tmp].start == *dma)) | 475 | if ((tdev->res.dma_resource[tmp].start == *dma)) |
467 | return 0; | 476 | return 0; |
@@ -471,30 +480,18 @@ int pnp_check_dma(struct pnp_dev * dev, int idx) | |||
471 | 480 | ||
472 | return 1; | 481 | return 1; |
473 | #else | 482 | #else |
474 | /* IA64 hasn't legacy DMA */ | 483 | /* IA64 does not have legacy DMA */ |
475 | return 0; | 484 | return 0; |
476 | #endif | 485 | #endif |
477 | } | 486 | } |
478 | 487 | ||
479 | |||
480 | #if 0 | ||
481 | EXPORT_SYMBOL(pnp_register_dependent_option); | ||
482 | EXPORT_SYMBOL(pnp_register_independent_option); | ||
483 | EXPORT_SYMBOL(pnp_register_irq_resource); | ||
484 | EXPORT_SYMBOL(pnp_register_dma_resource); | ||
485 | EXPORT_SYMBOL(pnp_register_port_resource); | ||
486 | EXPORT_SYMBOL(pnp_register_mem_resource); | ||
487 | #endif /* 0 */ | ||
488 | |||
489 | |||
490 | /* format is: pnp_reserve_irq=irq1[,irq2] .... */ | 488 | /* format is: pnp_reserve_irq=irq1[,irq2] .... */ |
491 | |||
492 | static int __init pnp_setup_reserve_irq(char *str) | 489 | static int __init pnp_setup_reserve_irq(char *str) |
493 | { | 490 | { |
494 | int i; | 491 | int i; |
495 | 492 | ||
496 | for (i = 0; i < 16; i++) | 493 | for (i = 0; i < 16; i++) |
497 | if (get_option(&str,&pnp_reserve_irq[i]) != 2) | 494 | if (get_option(&str, &pnp_reserve_irq[i]) != 2) |
498 | break; | 495 | break; |
499 | return 1; | 496 | return 1; |
500 | } | 497 | } |
@@ -502,13 +499,12 @@ static int __init pnp_setup_reserve_irq(char *str) | |||
502 | __setup("pnp_reserve_irq=", pnp_setup_reserve_irq); | 499 | __setup("pnp_reserve_irq=", pnp_setup_reserve_irq); |
503 | 500 | ||
504 | /* format is: pnp_reserve_dma=dma1[,dma2] .... */ | 501 | /* format is: pnp_reserve_dma=dma1[,dma2] .... */ |
505 | |||
506 | static int __init pnp_setup_reserve_dma(char *str) | 502 | static int __init pnp_setup_reserve_dma(char *str) |
507 | { | 503 | { |
508 | int i; | 504 | int i; |
509 | 505 | ||
510 | for (i = 0; i < 8; i++) | 506 | for (i = 0; i < 8; i++) |
511 | if (get_option(&str,&pnp_reserve_dma[i]) != 2) | 507 | if (get_option(&str, &pnp_reserve_dma[i]) != 2) |
512 | break; | 508 | break; |
513 | return 1; | 509 | return 1; |
514 | } | 510 | } |
@@ -516,13 +512,12 @@ static int __init pnp_setup_reserve_dma(char *str) | |||
516 | __setup("pnp_reserve_dma=", pnp_setup_reserve_dma); | 512 | __setup("pnp_reserve_dma=", pnp_setup_reserve_dma); |
517 | 513 | ||
518 | /* format is: pnp_reserve_io=io1,size1[,io2,size2] .... */ | 514 | /* format is: pnp_reserve_io=io1,size1[,io2,size2] .... */ |
519 | |||
520 | static int __init pnp_setup_reserve_io(char *str) | 515 | static int __init pnp_setup_reserve_io(char *str) |
521 | { | 516 | { |
522 | int i; | 517 | int i; |
523 | 518 | ||
524 | for (i = 0; i < 16; i++) | 519 | for (i = 0; i < 16; i++) |
525 | if (get_option(&str,&pnp_reserve_io[i]) != 2) | 520 | if (get_option(&str, &pnp_reserve_io[i]) != 2) |
526 | break; | 521 | break; |
527 | return 1; | 522 | return 1; |
528 | } | 523 | } |
@@ -530,13 +525,12 @@ static int __init pnp_setup_reserve_io(char *str) | |||
530 | __setup("pnp_reserve_io=", pnp_setup_reserve_io); | 525 | __setup("pnp_reserve_io=", pnp_setup_reserve_io); |
531 | 526 | ||
532 | /* format is: pnp_reserve_mem=mem1,size1[,mem2,size2] .... */ | 527 | /* format is: pnp_reserve_mem=mem1,size1[,mem2,size2] .... */ |
533 | |||
534 | static int __init pnp_setup_reserve_mem(char *str) | 528 | static int __init pnp_setup_reserve_mem(char *str) |
535 | { | 529 | { |
536 | int i; | 530 | int i; |
537 | 531 | ||
538 | for (i = 0; i < 16; i++) | 532 | for (i = 0; i < 16; i++) |
539 | if (get_option(&str,&pnp_reserve_mem[i]) != 2) | 533 | if (get_option(&str, &pnp_reserve_mem[i]) != 2) |
540 | break; | 534 | break; |
541 | return 1; | 535 | return 1; |
542 | } | 536 | } |
diff --git a/drivers/pnp/support.c b/drivers/pnp/support.c index 946a0dcd627d..13c608f5fb30 100644 --- a/drivers/pnp/support.c +++ b/drivers/pnp/support.c | |||
@@ -1,8 +1,7 @@ | |||
1 | /* | 1 | /* |
2 | * support.c - provides standard pnp functions for the use of pnp protocol drivers, | 2 | * support.c - standard functions for the use of pnp protocol drivers |
3 | * | 3 | * |
4 | * Copyright 2003 Adam Belay <ambx1@neo.rr.com> | 4 | * Copyright 2003 Adam Belay <ambx1@neo.rr.com> |
5 | * | ||
6 | */ | 5 | */ |
7 | 6 | ||
8 | #include <linux/module.h> | 7 | #include <linux/module.h> |
@@ -11,22 +10,18 @@ | |||
11 | #include "base.h" | 10 | #include "base.h" |
12 | 11 | ||
13 | /** | 12 | /** |
14 | * pnp_is_active - Determines if a device is active based on its current resources | 13 | * pnp_is_active - Determines if a device is active based on its current |
14 | * resources | ||
15 | * @dev: pointer to the desired PnP device | 15 | * @dev: pointer to the desired PnP device |
16 | * | ||
17 | */ | 16 | */ |
18 | 17 | int pnp_is_active(struct pnp_dev *dev) | |
19 | int pnp_is_active(struct pnp_dev * dev) | ||
20 | { | 18 | { |
21 | if (!pnp_port_start(dev, 0) && pnp_port_len(dev, 0) <= 1 && | 19 | if (!pnp_port_start(dev, 0) && pnp_port_len(dev, 0) <= 1 && |
22 | !pnp_mem_start(dev, 0) && pnp_mem_len(dev, 0) <= 1 && | 20 | !pnp_mem_start(dev, 0) && pnp_mem_len(dev, 0) <= 1 && |
23 | pnp_irq(dev, 0) == -1 && | 21 | pnp_irq(dev, 0) == -1 && pnp_dma(dev, 0) == -1) |
24 | pnp_dma(dev, 0) == -1) | 22 | return 0; |
25 | return 0; | ||
26 | else | 23 | else |
27 | return 1; | 24 | return 1; |
28 | } | 25 | } |
29 | 26 | ||
30 | |||
31 | |||
32 | EXPORT_SYMBOL(pnp_is_active); | 27 | EXPORT_SYMBOL(pnp_is_active); |
diff --git a/drivers/pnp/system.c b/drivers/pnp/system.c index a8a95540b1ef..a06f980b3ac9 100644 --- a/drivers/pnp/system.c +++ b/drivers/pnp/system.c | |||
@@ -16,13 +16,14 @@ | |||
16 | 16 | ||
17 | static const struct pnp_device_id pnp_dev_table[] = { | 17 | static const struct pnp_device_id pnp_dev_table[] = { |
18 | /* General ID for reserving resources */ | 18 | /* General ID for reserving resources */ |
19 | { "PNP0c02", 0 }, | 19 | {"PNP0c02", 0}, |
20 | /* memory controller */ | 20 | /* memory controller */ |
21 | { "PNP0c01", 0 }, | 21 | {"PNP0c01", 0}, |
22 | { "", 0 } | 22 | {"", 0} |
23 | }; | 23 | }; |
24 | 24 | ||
25 | static void reserve_range(const char *pnpid, resource_size_t start, resource_size_t end, int port) | 25 | static void reserve_range(const char *pnpid, resource_size_t start, |
26 | resource_size_t end, int port) | ||
26 | { | 27 | { |
27 | struct resource *res; | 28 | struct resource *res; |
28 | char *regionid; | 29 | char *regionid; |
@@ -32,9 +33,9 @@ static void reserve_range(const char *pnpid, resource_size_t start, resource_siz | |||
32 | return; | 33 | return; |
33 | snprintf(regionid, 16, "pnp %s", pnpid); | 34 | snprintf(regionid, 16, "pnp %s", pnpid); |
34 | if (port) | 35 | if (port) |
35 | res = request_region(start, end-start+1, regionid); | 36 | res = request_region(start, end - start + 1, regionid); |
36 | else | 37 | else |
37 | res = request_mem_region(start, end-start+1, regionid); | 38 | res = request_mem_region(start, end - start + 1, regionid); |
38 | if (res == NULL) | 39 | if (res == NULL) |
39 | kfree(regionid); | 40 | kfree(regionid); |
40 | else | 41 | else |
@@ -44,11 +45,10 @@ static void reserve_range(const char *pnpid, resource_size_t start, resource_siz | |||
44 | * example do reserve stuff they know about too, so we may well | 45 | * example do reserve stuff they know about too, so we may well |
45 | * have double reservations. | 46 | * have double reservations. |
46 | */ | 47 | */ |
47 | printk(KERN_INFO | 48 | printk(KERN_INFO "pnp: %s: %s range 0x%llx-0x%llx %s reserved\n", |
48 | "pnp: %s: %s range 0x%llx-0x%llx %s reserved\n", | 49 | pnpid, port ? "ioport" : "iomem", |
49 | pnpid, port ? "ioport" : "iomem", | 50 | (unsigned long long)start, (unsigned long long)end, |
50 | (unsigned long long)start, (unsigned long long)end, | 51 | NULL != res ? "has been" : "could not be"); |
51 | NULL != res ? "has been" : "could not be"); | ||
52 | } | 52 | } |
53 | 53 | ||
54 | static void reserve_resources_of_dev(const struct pnp_dev *dev) | 54 | static void reserve_resources_of_dev(const struct pnp_dev *dev) |
@@ -74,7 +74,7 @@ static void reserve_resources_of_dev(const struct pnp_dev *dev) | |||
74 | continue; /* invalid */ | 74 | continue; /* invalid */ |
75 | 75 | ||
76 | reserve_range(dev->dev.bus_id, pnp_port_start(dev, i), | 76 | reserve_range(dev->dev.bus_id, pnp_port_start(dev, i), |
77 | pnp_port_end(dev, i), 1); | 77 | pnp_port_end(dev, i), 1); |
78 | } | 78 | } |
79 | 79 | ||
80 | for (i = 0; i < PNP_MAX_MEM; i++) { | 80 | for (i = 0; i < PNP_MAX_MEM; i++) { |
@@ -82,24 +82,22 @@ static void reserve_resources_of_dev(const struct pnp_dev *dev) | |||
82 | continue; | 82 | continue; |
83 | 83 | ||
84 | reserve_range(dev->dev.bus_id, pnp_mem_start(dev, i), | 84 | reserve_range(dev->dev.bus_id, pnp_mem_start(dev, i), |
85 | pnp_mem_end(dev, i), 0); | 85 | pnp_mem_end(dev, i), 0); |
86 | } | 86 | } |
87 | |||
88 | return; | ||
89 | } | 87 | } |
90 | 88 | ||
91 | static int system_pnp_probe(struct pnp_dev * dev, const struct pnp_device_id *dev_id) | 89 | static int system_pnp_probe(struct pnp_dev *dev, |
90 | const struct pnp_device_id *dev_id) | ||
92 | { | 91 | { |
93 | reserve_resources_of_dev(dev); | 92 | reserve_resources_of_dev(dev); |
94 | return 0; | 93 | return 0; |
95 | } | 94 | } |
96 | 95 | ||
97 | static struct pnp_driver system_pnp_driver = { | 96 | static struct pnp_driver system_pnp_driver = { |
98 | .name = "system", | 97 | .name = "system", |
99 | .id_table = pnp_dev_table, | 98 | .id_table = pnp_dev_table, |
100 | .flags = PNP_DRIVER_RES_DO_NOT_CHANGE, | 99 | .flags = PNP_DRIVER_RES_DO_NOT_CHANGE, |
101 | .probe = system_pnp_probe, | 100 | .probe = system_pnp_probe, |
102 | .remove = NULL, | ||
103 | }; | 101 | }; |
104 | 102 | ||
105 | static int __init pnp_system_init(void) | 103 | static int __init pnp_system_init(void) |
diff --git a/drivers/rtc/Makefile b/drivers/rtc/Makefile index 7ede9e725360..d3a33aa2696f 100644 --- a/drivers/rtc/Makefile +++ b/drivers/rtc/Makefile | |||
@@ -15,34 +15,36 @@ rtc-core-$(CONFIG_RTC_INTF_DEV) += rtc-dev.o | |||
15 | rtc-core-$(CONFIG_RTC_INTF_PROC) += rtc-proc.o | 15 | rtc-core-$(CONFIG_RTC_INTF_PROC) += rtc-proc.o |
16 | rtc-core-$(CONFIG_RTC_INTF_SYSFS) += rtc-sysfs.o | 16 | rtc-core-$(CONFIG_RTC_INTF_SYSFS) += rtc-sysfs.o |
17 | 17 | ||
18 | # Keep the list ordered. | ||
19 | |||
20 | obj-$(CONFIG_RTC_DRV_AT32AP700X)+= rtc-at32ap700x.o | ||
21 | obj-$(CONFIG_RTC_DRV_AT91RM9200)+= rtc-at91rm9200.o | ||
22 | obj-$(CONFIG_RTC_DRV_BFIN) += rtc-bfin.o | ||
18 | obj-$(CONFIG_RTC_DRV_CMOS) += rtc-cmos.o | 23 | obj-$(CONFIG_RTC_DRV_CMOS) += rtc-cmos.o |
19 | obj-$(CONFIG_RTC_DRV_X1205) += rtc-x1205.o | 24 | obj-$(CONFIG_RTC_DRV_DS1216) += rtc-ds1216.o |
20 | obj-$(CONFIG_RTC_DRV_ISL1208) += rtc-isl1208.o | ||
21 | obj-$(CONFIG_RTC_DRV_TEST) += rtc-test.o | ||
22 | obj-$(CONFIG_RTC_DRV_AT32AP700X) += rtc-at32ap700x.o | ||
23 | obj-$(CONFIG_RTC_DRV_DS1307) += rtc-ds1307.o | 25 | obj-$(CONFIG_RTC_DRV_DS1307) += rtc-ds1307.o |
26 | obj-$(CONFIG_RTC_DRV_DS1553) += rtc-ds1553.o | ||
24 | obj-$(CONFIG_RTC_DRV_DS1672) += rtc-ds1672.o | 27 | obj-$(CONFIG_RTC_DRV_DS1672) += rtc-ds1672.o |
25 | obj-$(CONFIG_RTC_DRV_DS1742) += rtc-ds1742.o | 28 | obj-$(CONFIG_RTC_DRV_DS1742) += rtc-ds1742.o |
29 | obj-$(CONFIG_RTC_DRV_EP93XX) += rtc-ep93xx.o | ||
30 | obj-$(CONFIG_RTC_DRV_ISL1208) += rtc-isl1208.o | ||
31 | obj-$(CONFIG_RTC_DRV_M41T80) += rtc-m41t80.o | ||
32 | obj-$(CONFIG_RTC_DRV_M48T59) += rtc-m48t59.o | ||
33 | obj-$(CONFIG_RTC_DRV_M48T86) += rtc-m48t86.o | ||
34 | obj-$(CONFIG_RTC_DRV_MAX6900) += rtc-max6900.o | ||
35 | obj-$(CONFIG_RTC_DRV_MAX6902) += rtc-max6902.o | ||
26 | obj-$(CONFIG_RTC_DRV_OMAP) += rtc-omap.o | 36 | obj-$(CONFIG_RTC_DRV_OMAP) += rtc-omap.o |
27 | obj-$(CONFIG_RTC_DRV_PCF8563) += rtc-pcf8563.o | 37 | obj-$(CONFIG_RTC_DRV_PCF8563) += rtc-pcf8563.o |
28 | obj-$(CONFIG_RTC_DRV_PCF8583) += rtc-pcf8583.o | 38 | obj-$(CONFIG_RTC_DRV_PCF8583) += rtc-pcf8583.o |
39 | obj-$(CONFIG_RTC_DRV_PL031) += rtc-pl031.o | ||
40 | obj-$(CONFIG_RTC_DRV_RS5C313) += rtc-rs5c313.o | ||
41 | obj-$(CONFIG_RTC_DRV_RS5C348) += rtc-rs5c348.o | ||
29 | obj-$(CONFIG_RTC_DRV_RS5C372) += rtc-rs5c372.o | 42 | obj-$(CONFIG_RTC_DRV_RS5C372) += rtc-rs5c372.o |
30 | obj-$(CONFIG_RTC_DRV_S3C) += rtc-s3c.o | 43 | obj-$(CONFIG_RTC_DRV_S3C) += rtc-s3c.o |
31 | obj-$(CONFIG_RTC_DRV_RS5C348) += rtc-rs5c348.o | ||
32 | obj-$(CONFIG_RTC_DRV_M41T80) += rtc-m41t80.o | ||
33 | obj-$(CONFIG_RTC_DRV_M48T86) += rtc-m48t86.o | ||
34 | obj-$(CONFIG_RTC_DRV_DS1553) += rtc-ds1553.o | ||
35 | obj-$(CONFIG_RTC_DRV_STK17TA8) += rtc-stk17ta8.o | ||
36 | obj-$(CONFIG_RTC_DRV_RS5C313) += rtc-rs5c313.o | ||
37 | obj-$(CONFIG_RTC_DRV_EP93XX) += rtc-ep93xx.o | ||
38 | obj-$(CONFIG_RTC_DRV_SA1100) += rtc-sa1100.o | 44 | obj-$(CONFIG_RTC_DRV_SA1100) += rtc-sa1100.o |
39 | obj-$(CONFIG_RTC_DRV_VR41XX) += rtc-vr41xx.o | ||
40 | obj-$(CONFIG_RTC_DRV_PL031) += rtc-pl031.o | ||
41 | obj-$(CONFIG_RTC_DRV_MAX6900) += rtc-max6900.o | ||
42 | obj-$(CONFIG_RTC_DRV_MAX6902) += rtc-max6902.o | ||
43 | obj-$(CONFIG_RTC_DRV_V3020) += rtc-v3020.o | ||
44 | obj-$(CONFIG_RTC_DRV_AT91RM9200)+= rtc-at91rm9200.o | ||
45 | obj-$(CONFIG_RTC_DRV_SH) += rtc-sh.o | 45 | obj-$(CONFIG_RTC_DRV_SH) += rtc-sh.o |
46 | obj-$(CONFIG_RTC_DRV_BFIN) += rtc-bfin.o | 46 | obj-$(CONFIG_RTC_DRV_STK17TA8) += rtc-stk17ta8.o |
47 | obj-$(CONFIG_RTC_DRV_M48T59) += rtc-m48t59.o | 47 | obj-$(CONFIG_RTC_DRV_TEST) += rtc-test.o |
48 | obj-$(CONFIG_RTC_DRV_DS1216) += rtc-ds1216.o | 48 | obj-$(CONFIG_RTC_DRV_V3020) += rtc-v3020.o |
49 | obj-$(CONFIG_RTC_DRV_VR41XX) += rtc-vr41xx.o | ||
50 | obj-$(CONFIG_RTC_DRV_X1205) += rtc-x1205.o | ||
diff --git a/drivers/rtc/class.c b/drivers/rtc/class.c index 8b3cd31d6a61..10ab3b71ffc6 100644 --- a/drivers/rtc/class.c +++ b/drivers/rtc/class.c | |||
@@ -46,6 +46,7 @@ static int rtc_suspend(struct device *dev, pm_message_t mesg) | |||
46 | { | 46 | { |
47 | struct rtc_device *rtc = to_rtc_device(dev); | 47 | struct rtc_device *rtc = to_rtc_device(dev); |
48 | struct rtc_time tm; | 48 | struct rtc_time tm; |
49 | struct timespec ts = current_kernel_time(); | ||
49 | 50 | ||
50 | if (strncmp(rtc->dev.bus_id, | 51 | if (strncmp(rtc->dev.bus_id, |
51 | CONFIG_RTC_HCTOSYS_DEVICE, | 52 | CONFIG_RTC_HCTOSYS_DEVICE, |
@@ -57,8 +58,8 @@ static int rtc_suspend(struct device *dev, pm_message_t mesg) | |||
57 | 58 | ||
58 | /* RTC precision is 1 second; adjust delta for avg 1/2 sec err */ | 59 | /* RTC precision is 1 second; adjust delta for avg 1/2 sec err */ |
59 | set_normalized_timespec(&delta, | 60 | set_normalized_timespec(&delta, |
60 | xtime.tv_sec - oldtime, | 61 | ts.tv_sec - oldtime, |
61 | xtime.tv_nsec - (NSEC_PER_SEC >> 1)); | 62 | ts.tv_nsec - (NSEC_PER_SEC >> 1)); |
62 | 63 | ||
63 | return 0; | 64 | return 0; |
64 | } | 65 | } |
diff --git a/drivers/rtc/rtc-ds1307.c b/drivers/rtc/rtc-ds1307.c index 5158a625671f..db6f3f0d8982 100644 --- a/drivers/rtc/rtc-ds1307.c +++ b/drivers/rtc/rtc-ds1307.c | |||
@@ -352,7 +352,7 @@ read_rtc: | |||
352 | /* oscillator fault? clear flag, and warn */ | 352 | /* oscillator fault? clear flag, and warn */ |
353 | if (ds1307->regs[DS1307_REG_CONTROL] & DS1338_BIT_OSF) { | 353 | if (ds1307->regs[DS1307_REG_CONTROL] & DS1338_BIT_OSF) { |
354 | i2c_smbus_write_byte_data(client, DS1307_REG_CONTROL, | 354 | i2c_smbus_write_byte_data(client, DS1307_REG_CONTROL, |
355 | ds1307->regs[DS1337_REG_CONTROL] | 355 | ds1307->regs[DS1307_REG_CONTROL] |
356 | & ~DS1338_BIT_OSF); | 356 | & ~DS1338_BIT_OSF); |
357 | dev_warn(&client->dev, "SET TIME!\n"); | 357 | dev_warn(&client->dev, "SET TIME!\n"); |
358 | goto read_rtc; | 358 | goto read_rtc; |
diff --git a/drivers/rtc/rtc-stk17ta8.c b/drivers/rtc/rtc-stk17ta8.c index f10d3facecbe..8288b6b2bf2b 100644 --- a/drivers/rtc/rtc-stk17ta8.c +++ b/drivers/rtc/rtc-stk17ta8.c | |||
@@ -258,7 +258,8 @@ static const struct rtc_class_ops stk17ta8_rtc_ops = { | |||
258 | .ioctl = stk17ta8_rtc_ioctl, | 258 | .ioctl = stk17ta8_rtc_ioctl, |
259 | }; | 259 | }; |
260 | 260 | ||
261 | static ssize_t stk17ta8_nvram_read(struct kobject *kobj, char *buf, | 261 | static ssize_t stk17ta8_nvram_read(struct kobject *kobj, |
262 | struct bin_attribute *attr, char *buf, | ||
262 | loff_t pos, size_t size) | 263 | loff_t pos, size_t size) |
263 | { | 264 | { |
264 | struct platform_device *pdev = | 265 | struct platform_device *pdev = |
@@ -272,7 +273,8 @@ static ssize_t stk17ta8_nvram_read(struct kobject *kobj, char *buf, | |||
272 | return count; | 273 | return count; |
273 | } | 274 | } |
274 | 275 | ||
275 | static ssize_t stk17ta8_nvram_write(struct kobject *kobj, char *buf, | 276 | static ssize_t stk17ta8_nvram_write(struct kobject *kobj, |
277 | struct bin_attribute *attr, char *buf, | ||
276 | loff_t pos, size_t size) | 278 | loff_t pos, size_t size) |
277 | { | 279 | { |
278 | struct platform_device *pdev = | 280 | struct platform_device *pdev = |
diff --git a/drivers/s390/block/dasd.c b/drivers/s390/block/dasd.c index bfeca57098fa..e6bfce690ca3 100644 --- a/drivers/s390/block/dasd.c +++ b/drivers/s390/block/dasd.c | |||
@@ -1187,7 +1187,7 @@ dasd_end_request_cb(struct dasd_ccw_req * cqr, void *data) | |||
1187 | static void | 1187 | static void |
1188 | __dasd_process_blk_queue(struct dasd_device * device) | 1188 | __dasd_process_blk_queue(struct dasd_device * device) |
1189 | { | 1189 | { |
1190 | request_queue_t *queue; | 1190 | struct request_queue *queue; |
1191 | struct request *req; | 1191 | struct request *req; |
1192 | struct dasd_ccw_req *cqr; | 1192 | struct dasd_ccw_req *cqr; |
1193 | int nr_queued; | 1193 | int nr_queued; |
@@ -1740,7 +1740,7 @@ dasd_cancel_req(struct dasd_ccw_req *cqr) | |||
1740 | * Dasd request queue function. Called from ll_rw_blk.c | 1740 | * Dasd request queue function. Called from ll_rw_blk.c |
1741 | */ | 1741 | */ |
1742 | static void | 1742 | static void |
1743 | do_dasd_request(request_queue_t * queue) | 1743 | do_dasd_request(struct request_queue * queue) |
1744 | { | 1744 | { |
1745 | struct dasd_device *device; | 1745 | struct dasd_device *device; |
1746 | 1746 | ||
diff --git a/drivers/s390/block/dasd_int.h b/drivers/s390/block/dasd_int.h index 241294cba415..aeda52682446 100644 --- a/drivers/s390/block/dasd_int.h +++ b/drivers/s390/block/dasd_int.h | |||
@@ -293,7 +293,7 @@ struct dasd_uid { | |||
293 | struct dasd_device { | 293 | struct dasd_device { |
294 | /* Block device stuff. */ | 294 | /* Block device stuff. */ |
295 | struct gendisk *gdp; | 295 | struct gendisk *gdp; |
296 | request_queue_t *request_queue; | 296 | struct request_queue *request_queue; |
297 | spinlock_t request_queue_lock; | 297 | spinlock_t request_queue_lock; |
298 | struct block_device *bdev; | 298 | struct block_device *bdev; |
299 | unsigned int devindex; | 299 | unsigned int devindex; |
diff --git a/drivers/s390/block/dcssblk.c b/drivers/s390/block/dcssblk.c index 35765f6a86e0..4d8798bacf97 100644 --- a/drivers/s390/block/dcssblk.c +++ b/drivers/s390/block/dcssblk.c | |||
@@ -621,7 +621,7 @@ out: | |||
621 | } | 621 | } |
622 | 622 | ||
623 | static int | 623 | static int |
624 | dcssblk_make_request(request_queue_t *q, struct bio *bio) | 624 | dcssblk_make_request(struct request_queue *q, struct bio *bio) |
625 | { | 625 | { |
626 | struct dcssblk_dev_info *dev_info; | 626 | struct dcssblk_dev_info *dev_info; |
627 | struct bio_vec *bvec; | 627 | struct bio_vec *bvec; |
diff --git a/drivers/s390/block/xpram.c b/drivers/s390/block/xpram.c index a04d9120cef0..354a060e5bec 100644 --- a/drivers/s390/block/xpram.c +++ b/drivers/s390/block/xpram.c | |||
@@ -191,7 +191,7 @@ static unsigned long __init xpram_highest_page_index(void) | |||
191 | /* | 191 | /* |
192 | * Block device make request function. | 192 | * Block device make request function. |
193 | */ | 193 | */ |
194 | static int xpram_make_request(request_queue_t *q, struct bio *bio) | 194 | static int xpram_make_request(struct request_queue *q, struct bio *bio) |
195 | { | 195 | { |
196 | xpram_device_t *xdev = bio->bi_bdev->bd_disk->private_data; | 196 | xpram_device_t *xdev = bio->bi_bdev->bd_disk->private_data; |
197 | struct bio_vec *bvec; | 197 | struct bio_vec *bvec; |
diff --git a/drivers/s390/char/tape.h b/drivers/s390/char/tape.h index 3b52f5c1dbef..dddf8d62c153 100644 --- a/drivers/s390/char/tape.h +++ b/drivers/s390/char/tape.h | |||
@@ -188,7 +188,7 @@ struct tape_blk_data | |||
188 | { | 188 | { |
189 | struct tape_device * device; | 189 | struct tape_device * device; |
190 | /* Block device request queue. */ | 190 | /* Block device request queue. */ |
191 | request_queue_t * request_queue; | 191 | struct request_queue * request_queue; |
192 | spinlock_t request_queue_lock; | 192 | spinlock_t request_queue_lock; |
193 | 193 | ||
194 | /* Task to move entries from block request to CCS request queue. */ | 194 | /* Task to move entries from block request to CCS request queue. */ |
diff --git a/drivers/s390/char/tape_block.c b/drivers/s390/char/tape_block.c index dd0ecaed592e..eeb92e2ed0cc 100644 --- a/drivers/s390/char/tape_block.c +++ b/drivers/s390/char/tape_block.c | |||
@@ -147,7 +147,7 @@ static void | |||
147 | tapeblock_requeue(struct work_struct *work) { | 147 | tapeblock_requeue(struct work_struct *work) { |
148 | struct tape_blk_data * blkdat; | 148 | struct tape_blk_data * blkdat; |
149 | struct tape_device * device; | 149 | struct tape_device * device; |
150 | request_queue_t * queue; | 150 | struct request_queue * queue; |
151 | int nr_queued; | 151 | int nr_queued; |
152 | struct request * req; | 152 | struct request * req; |
153 | struct list_head * l; | 153 | struct list_head * l; |
@@ -194,7 +194,7 @@ tapeblock_requeue(struct work_struct *work) { | |||
194 | * Tape request queue function. Called from ll_rw_blk.c | 194 | * Tape request queue function. Called from ll_rw_blk.c |
195 | */ | 195 | */ |
196 | static void | 196 | static void |
197 | tapeblock_request_fn(request_queue_t *queue) | 197 | tapeblock_request_fn(struct request_queue *queue) |
198 | { | 198 | { |
199 | struct tape_device *device; | 199 | struct tape_device *device; |
200 | 200 | ||
diff --git a/drivers/s390/net/ctcmain.c b/drivers/s390/net/ctcmain.c index b20fd0681733..92e8a37b5022 100644 --- a/drivers/s390/net/ctcmain.c +++ b/drivers/s390/net/ctcmain.c | |||
@@ -674,7 +674,7 @@ ch_action_txdone(fsm_instance * fi, int event, void *arg) | |||
674 | int first = 1; | 674 | int first = 1; |
675 | int i; | 675 | int i; |
676 | unsigned long duration; | 676 | unsigned long duration; |
677 | struct timespec done_stamp = xtime; | 677 | struct timespec done_stamp = current_kernel_time(); |
678 | 678 | ||
679 | DBF_TEXT(trace, 4, __FUNCTION__); | 679 | DBF_TEXT(trace, 4, __FUNCTION__); |
680 | 680 | ||
@@ -730,7 +730,7 @@ ch_action_txdone(fsm_instance * fi, int event, void *arg) | |||
730 | spin_unlock(&ch->collect_lock); | 730 | spin_unlock(&ch->collect_lock); |
731 | ch->ccw[1].count = ch->trans_skb->len; | 731 | ch->ccw[1].count = ch->trans_skb->len; |
732 | fsm_addtimer(&ch->timer, CTC_TIMEOUT_5SEC, CH_EVENT_TIMER, ch); | 732 | fsm_addtimer(&ch->timer, CTC_TIMEOUT_5SEC, CH_EVENT_TIMER, ch); |
733 | ch->prof.send_stamp = xtime; | 733 | ch->prof.send_stamp = current_kernel_time(); |
734 | rc = ccw_device_start(ch->cdev, &ch->ccw[0], | 734 | rc = ccw_device_start(ch->cdev, &ch->ccw[0], |
735 | (unsigned long) ch, 0xff, 0); | 735 | (unsigned long) ch, 0xff, 0); |
736 | ch->prof.doios_multi++; | 736 | ch->prof.doios_multi++; |
@@ -2281,7 +2281,7 @@ transmit_skb(struct channel *ch, struct sk_buff *skb) | |||
2281 | fsm_newstate(ch->fsm, CH_STATE_TX); | 2281 | fsm_newstate(ch->fsm, CH_STATE_TX); |
2282 | fsm_addtimer(&ch->timer, CTC_TIMEOUT_5SEC, CH_EVENT_TIMER, ch); | 2282 | fsm_addtimer(&ch->timer, CTC_TIMEOUT_5SEC, CH_EVENT_TIMER, ch); |
2283 | spin_lock_irqsave(get_ccwdev_lock(ch->cdev), saveflags); | 2283 | spin_lock_irqsave(get_ccwdev_lock(ch->cdev), saveflags); |
2284 | ch->prof.send_stamp = xtime; | 2284 | ch->prof.send_stamp = current_kernel_time(); |
2285 | rc = ccw_device_start(ch->cdev, &ch->ccw[ccw_idx], | 2285 | rc = ccw_device_start(ch->cdev, &ch->ccw[ccw_idx], |
2286 | (unsigned long) ch, 0xff, 0); | 2286 | (unsigned long) ch, 0xff, 0); |
2287 | spin_unlock_irqrestore(get_ccwdev_lock(ch->cdev), saveflags); | 2287 | spin_unlock_irqrestore(get_ccwdev_lock(ch->cdev), saveflags); |
diff --git a/drivers/s390/net/netiucv.c b/drivers/s390/net/netiucv.c index 3d28e1a5bf79..268889474339 100644 --- a/drivers/s390/net/netiucv.c +++ b/drivers/s390/net/netiucv.c | |||
@@ -753,7 +753,7 @@ static void conn_action_txdone(fsm_instance *fi, int event, void *arg) | |||
753 | 753 | ||
754 | header.next = 0; | 754 | header.next = 0; |
755 | memcpy(skb_put(conn->tx_buff, NETIUCV_HDRLEN), &header, NETIUCV_HDRLEN); | 755 | memcpy(skb_put(conn->tx_buff, NETIUCV_HDRLEN), &header, NETIUCV_HDRLEN); |
756 | conn->prof.send_stamp = xtime; | 756 | conn->prof.send_stamp = current_kernel_time(); |
757 | txmsg.class = 0; | 757 | txmsg.class = 0; |
758 | txmsg.tag = 0; | 758 | txmsg.tag = 0; |
759 | rc = iucv_message_send(conn->path, &txmsg, 0, 0, | 759 | rc = iucv_message_send(conn->path, &txmsg, 0, 0, |
@@ -1185,7 +1185,7 @@ static int netiucv_transmit_skb(struct iucv_connection *conn, | |||
1185 | memcpy(skb_put(nskb, NETIUCV_HDRLEN), &header, NETIUCV_HDRLEN); | 1185 | memcpy(skb_put(nskb, NETIUCV_HDRLEN), &header, NETIUCV_HDRLEN); |
1186 | 1186 | ||
1187 | fsm_newstate(conn->fsm, CONN_STATE_TX); | 1187 | fsm_newstate(conn->fsm, CONN_STATE_TX); |
1188 | conn->prof.send_stamp = xtime; | 1188 | conn->prof.send_stamp = current_kernel_time(); |
1189 | 1189 | ||
1190 | msg.tag = 1; | 1190 | msg.tag = 1; |
1191 | msg.class = 0; | 1191 | msg.class = 0; |
diff --git a/drivers/sbus/char/Kconfig b/drivers/sbus/char/Kconfig index 35a73168333f..400c65bfb8c7 100644 --- a/drivers/sbus/char/Kconfig +++ b/drivers/sbus/char/Kconfig | |||
@@ -15,6 +15,7 @@ config SUN_OPENPROMIO | |||
15 | 15 | ||
16 | config SUN_MOSTEK_RTC | 16 | config SUN_MOSTEK_RTC |
17 | tristate "Mostek real time clock support" | 17 | tristate "Mostek real time clock support" |
18 | depends on SPARC32 | ||
18 | help | 19 | help |
19 | The Mostek RTC chip is used on all known Sun computers except | 20 | The Mostek RTC chip is used on all known Sun computers except |
20 | some JavaStations. For a JavaStation you need to say Y both here | 21 | some JavaStations. For a JavaStation you need to say Y both here |
diff --git a/drivers/sbus/char/jsflash.c b/drivers/sbus/char/jsflash.c index 5157a2abc58d..4b7079fdc10c 100644 --- a/drivers/sbus/char/jsflash.c +++ b/drivers/sbus/char/jsflash.c | |||
@@ -185,7 +185,7 @@ static void jsfd_read(char *buf, unsigned long p, size_t togo) { | |||
185 | } | 185 | } |
186 | } | 186 | } |
187 | 187 | ||
188 | static void jsfd_do_request(request_queue_t *q) | 188 | static void jsfd_do_request(struct request_queue *q) |
189 | { | 189 | { |
190 | struct request *req; | 190 | struct request *req; |
191 | 191 | ||
diff --git a/drivers/scsi/aacraid/linit.c b/drivers/scsi/aacraid/linit.c index d76e1a8cb93a..c709dc8ad99d 100644 --- a/drivers/scsi/aacraid/linit.c +++ b/drivers/scsi/aacraid/linit.c | |||
@@ -636,6 +636,8 @@ static int aac_cfg_open(struct inode *inode, struct file *file) | |||
636 | static int aac_cfg_ioctl(struct inode *inode, struct file *file, | 636 | static int aac_cfg_ioctl(struct inode *inode, struct file *file, |
637 | unsigned int cmd, unsigned long arg) | 637 | unsigned int cmd, unsigned long arg) |
638 | { | 638 | { |
639 | if (!capable(CAP_SYS_ADMIN)) | ||
640 | return -EPERM; | ||
639 | return aac_do_ioctl(file->private_data, cmd, (void __user *)arg); | 641 | return aac_do_ioctl(file->private_data, cmd, (void __user *)arg); |
640 | } | 642 | } |
641 | 643 | ||
@@ -689,6 +691,8 @@ static int aac_compat_ioctl(struct scsi_device *sdev, int cmd, void __user *arg) | |||
689 | 691 | ||
690 | static long aac_compat_cfg_ioctl(struct file *file, unsigned cmd, unsigned long arg) | 692 | static long aac_compat_cfg_ioctl(struct file *file, unsigned cmd, unsigned long arg) |
691 | { | 693 | { |
694 | if (!capable(CAP_SYS_ADMIN)) | ||
695 | return -EPERM; | ||
692 | return aac_compat_do_ioctl((struct aac_dev *)file->private_data, cmd, arg); | 696 | return aac_compat_do_ioctl((struct aac_dev *)file->private_data, cmd, arg); |
693 | } | 697 | } |
694 | #endif | 698 | #endif |
diff --git a/drivers/scsi/scsi_lib.c b/drivers/scsi/scsi_lib.c index da63c544919b..21c075d44db1 100644 --- a/drivers/scsi/scsi_lib.c +++ b/drivers/scsi/scsi_lib.c | |||
@@ -654,7 +654,7 @@ void scsi_run_host_queues(struct Scsi_Host *shost) | |||
654 | static struct scsi_cmnd *scsi_end_request(struct scsi_cmnd *cmd, int uptodate, | 654 | static struct scsi_cmnd *scsi_end_request(struct scsi_cmnd *cmd, int uptodate, |
655 | int bytes, int requeue) | 655 | int bytes, int requeue) |
656 | { | 656 | { |
657 | request_queue_t *q = cmd->device->request_queue; | 657 | struct request_queue *q = cmd->device->request_queue; |
658 | struct request *req = cmd->request; | 658 | struct request *req = cmd->request; |
659 | unsigned long flags; | 659 | unsigned long flags; |
660 | 660 | ||
@@ -818,7 +818,7 @@ void scsi_io_completion(struct scsi_cmnd *cmd, unsigned int good_bytes) | |||
818 | { | 818 | { |
819 | int result = cmd->result; | 819 | int result = cmd->result; |
820 | int this_count = cmd->request_bufflen; | 820 | int this_count = cmd->request_bufflen; |
821 | request_queue_t *q = cmd->device->request_queue; | 821 | struct request_queue *q = cmd->device->request_queue; |
822 | struct request *req = cmd->request; | 822 | struct request *req = cmd->request; |
823 | int clear_errors = 1; | 823 | int clear_errors = 1; |
824 | struct scsi_sense_hdr sshdr; | 824 | struct scsi_sense_hdr sshdr; |
@@ -1038,7 +1038,7 @@ static int scsi_init_io(struct scsi_cmnd *cmd) | |||
1038 | return BLKPREP_KILL; | 1038 | return BLKPREP_KILL; |
1039 | } | 1039 | } |
1040 | 1040 | ||
1041 | static int scsi_issue_flush_fn(request_queue_t *q, struct gendisk *disk, | 1041 | static int scsi_issue_flush_fn(struct request_queue *q, struct gendisk *disk, |
1042 | sector_t *error_sector) | 1042 | sector_t *error_sector) |
1043 | { | 1043 | { |
1044 | struct scsi_device *sdev = q->queuedata; | 1044 | struct scsi_device *sdev = q->queuedata; |
@@ -1340,7 +1340,7 @@ static inline int scsi_host_queue_ready(struct request_queue *q, | |||
1340 | /* | 1340 | /* |
1341 | * Kill a request for a dead device | 1341 | * Kill a request for a dead device |
1342 | */ | 1342 | */ |
1343 | static void scsi_kill_request(struct request *req, request_queue_t *q) | 1343 | static void scsi_kill_request(struct request *req, struct request_queue *q) |
1344 | { | 1344 | { |
1345 | struct scsi_cmnd *cmd = req->special; | 1345 | struct scsi_cmnd *cmd = req->special; |
1346 | struct scsi_device *sdev = cmd->device; | 1346 | struct scsi_device *sdev = cmd->device; |
@@ -2119,7 +2119,7 @@ EXPORT_SYMBOL(scsi_target_resume); | |||
2119 | int | 2119 | int |
2120 | scsi_internal_device_block(struct scsi_device *sdev) | 2120 | scsi_internal_device_block(struct scsi_device *sdev) |
2121 | { | 2121 | { |
2122 | request_queue_t *q = sdev->request_queue; | 2122 | struct request_queue *q = sdev->request_queue; |
2123 | unsigned long flags; | 2123 | unsigned long flags; |
2124 | int err = 0; | 2124 | int err = 0; |
2125 | 2125 | ||
@@ -2159,7 +2159,7 @@ EXPORT_SYMBOL_GPL(scsi_internal_device_block); | |||
2159 | int | 2159 | int |
2160 | scsi_internal_device_unblock(struct scsi_device *sdev) | 2160 | scsi_internal_device_unblock(struct scsi_device *sdev) |
2161 | { | 2161 | { |
2162 | request_queue_t *q = sdev->request_queue; | 2162 | struct request_queue *q = sdev->request_queue; |
2163 | int err; | 2163 | int err; |
2164 | unsigned long flags; | 2164 | unsigned long flags; |
2165 | 2165 | ||
diff --git a/drivers/scsi/sd.c b/drivers/scsi/sd.c index 424d557284a9..e21c7142a3ea 100644 --- a/drivers/scsi/sd.c +++ b/drivers/scsi/sd.c | |||
@@ -814,7 +814,7 @@ static int sd_issue_flush(struct device *dev, sector_t *error_sector) | |||
814 | return ret; | 814 | return ret; |
815 | } | 815 | } |
816 | 816 | ||
817 | static void sd_prepare_flush(request_queue_t *q, struct request *rq) | 817 | static void sd_prepare_flush(struct request_queue *q, struct request *rq) |
818 | { | 818 | { |
819 | memset(rq->cmd, 0, sizeof(rq->cmd)); | 819 | memset(rq->cmd, 0, sizeof(rq->cmd)); |
820 | rq->cmd_type = REQ_TYPE_BLOCK_PC; | 820 | rq->cmd_type = REQ_TYPE_BLOCK_PC; |
@@ -1285,7 +1285,7 @@ got_data: | |||
1285 | */ | 1285 | */ |
1286 | int hard_sector = sector_size; | 1286 | int hard_sector = sector_size; |
1287 | sector_t sz = (sdkp->capacity/2) * (hard_sector/256); | 1287 | sector_t sz = (sdkp->capacity/2) * (hard_sector/256); |
1288 | request_queue_t *queue = sdp->request_queue; | 1288 | struct request_queue *queue = sdp->request_queue; |
1289 | sector_t mb = sz; | 1289 | sector_t mb = sz; |
1290 | 1290 | ||
1291 | blk_queue_hardsect_size(queue, hard_sector); | 1291 | blk_queue_hardsect_size(queue, hard_sector); |
diff --git a/drivers/scsi/sr.c b/drivers/scsi/sr.c index e7b6a7fde1cb..902eb11ffe8a 100644 --- a/drivers/scsi/sr.c +++ b/drivers/scsi/sr.c | |||
@@ -624,7 +624,7 @@ static void get_sectorsize(struct scsi_cd *cd) | |||
624 | unsigned char *buffer; | 624 | unsigned char *buffer; |
625 | int the_result, retries = 3; | 625 | int the_result, retries = 3; |
626 | int sector_size; | 626 | int sector_size; |
627 | request_queue_t *queue; | 627 | struct request_queue *queue; |
628 | 628 | ||
629 | buffer = kmalloc(512, GFP_KERNEL | GFP_DMA); | 629 | buffer = kmalloc(512, GFP_KERNEL | GFP_DMA); |
630 | if (!buffer) | 630 | if (!buffer) |
diff --git a/drivers/serial/68328serial.c b/drivers/serial/68328serial.c index cad426c9711e..aad4012bbb30 100644 --- a/drivers/serial/68328serial.c +++ b/drivers/serial/68328serial.c | |||
@@ -33,7 +33,6 @@ | |||
33 | #include <linux/keyboard.h> | 33 | #include <linux/keyboard.h> |
34 | #include <linux/init.h> | 34 | #include <linux/init.h> |
35 | #include <linux/pm.h> | 35 | #include <linux/pm.h> |
36 | #include <linux/pm_legacy.h> | ||
37 | #include <linux/bitops.h> | 36 | #include <linux/bitops.h> |
38 | #include <linux/delay.h> | 37 | #include <linux/delay.h> |
39 | 38 | ||
@@ -401,9 +400,9 @@ irqreturn_t rs_interrupt(int irq, void *dev_id) | |||
401 | return IRQ_HANDLED; | 400 | return IRQ_HANDLED; |
402 | } | 401 | } |
403 | 402 | ||
404 | static void do_softint(void *private) | 403 | static void do_softint(struct work_struct *work) |
405 | { | 404 | { |
406 | struct m68k_serial *info = (struct m68k_serial *) private; | 405 | struct m68k_serial *info = container_of(work, struct m68k_serial, tqueue); |
407 | struct tty_struct *tty; | 406 | struct tty_struct *tty; |
408 | 407 | ||
409 | tty = info->tty; | 408 | tty = info->tty; |
@@ -425,9 +424,9 @@ static void do_softint(void *private) | |||
425 | * do_serial_hangup() -> tty->hangup() -> rs_hangup() | 424 | * do_serial_hangup() -> tty->hangup() -> rs_hangup() |
426 | * | 425 | * |
427 | */ | 426 | */ |
428 | static void do_serial_hangup(void *private) | 427 | static void do_serial_hangup(struct work_struct *work) |
429 | { | 428 | { |
430 | struct m68k_serial *info = (struct m68k_serial *) private; | 429 | struct m68k_serial *info = container_of(work, struct m68k_serial, tqueue_hangup); |
431 | struct tty_struct *tty; | 430 | struct tty_struct *tty; |
432 | 431 | ||
433 | tty = info->tty; | 432 | tty = info->tty; |
@@ -1324,59 +1323,6 @@ static void show_serial_version(void) | |||
1324 | printk("MC68328 serial driver version 1.00\n"); | 1323 | printk("MC68328 serial driver version 1.00\n"); |
1325 | } | 1324 | } |
1326 | 1325 | ||
1327 | #ifdef CONFIG_PM_LEGACY | ||
1328 | /* Serial Power management | ||
1329 | * The console (currently fixed at line 0) is a special case for power | ||
1330 | * management because the kernel is so chatty. The console will be | ||
1331 | * explicitly disabled my our power manager as the last minute, so we won't | ||
1332 | * mess with it here. | ||
1333 | */ | ||
1334 | static struct pm_dev *serial_pm[NR_PORTS]; | ||
1335 | |||
1336 | static int serial_pm_callback(struct pm_dev *dev, pm_request_t request, void *data) | ||
1337 | { | ||
1338 | struct m68k_serial *info = (struct m68k_serial *)dev->data; | ||
1339 | |||
1340 | if(info == NULL) | ||
1341 | return -1; | ||
1342 | |||
1343 | /* special case for line 0 - pm restores it */ | ||
1344 | if(info->line == 0) | ||
1345 | return 0; | ||
1346 | |||
1347 | switch (request) { | ||
1348 | case PM_SUSPEND: | ||
1349 | shutdown(info); | ||
1350 | break; | ||
1351 | |||
1352 | case PM_RESUME: | ||
1353 | startup(info); | ||
1354 | break; | ||
1355 | } | ||
1356 | return 0; | ||
1357 | } | ||
1358 | |||
1359 | void shutdown_console(void) | ||
1360 | { | ||
1361 | struct m68k_serial *info = &m68k_soft[0]; | ||
1362 | |||
1363 | /* HACK: wait a bit for any pending printk's to be dumped */ | ||
1364 | { | ||
1365 | int i = 10000; | ||
1366 | while(i--); | ||
1367 | } | ||
1368 | |||
1369 | shutdown(info); | ||
1370 | } | ||
1371 | |||
1372 | void startup_console(void) | ||
1373 | { | ||
1374 | struct m68k_serial *info = &m68k_soft[0]; | ||
1375 | startup(info); | ||
1376 | } | ||
1377 | #endif /* CONFIG_PM_LEGACY */ | ||
1378 | |||
1379 | |||
1380 | static const struct tty_operations rs_ops = { | 1326 | static const struct tty_operations rs_ops = { |
1381 | .open = rs_open, | 1327 | .open = rs_open, |
1382 | .close = rs_close, | 1328 | .close = rs_close, |
@@ -1444,8 +1390,8 @@ rs68328_init(void) | |||
1444 | info->event = 0; | 1390 | info->event = 0; |
1445 | info->count = 0; | 1391 | info->count = 0; |
1446 | info->blocked_open = 0; | 1392 | info->blocked_open = 0; |
1447 | INIT_WORK(&info->tqueue, do_softint, info); | 1393 | INIT_WORK(&info->tqueue, do_softint); |
1448 | INIT_WORK(&info->tqueue_hangup, do_serial_hangup, info); | 1394 | INIT_WORK(&info->tqueue_hangup, do_serial_hangup); |
1449 | init_waitqueue_head(&info->open_wait); | 1395 | init_waitqueue_head(&info->open_wait); |
1450 | init_waitqueue_head(&info->close_wait); | 1396 | init_waitqueue_head(&info->close_wait); |
1451 | info->line = i; | 1397 | info->line = i; |
@@ -1467,11 +1413,6 @@ rs68328_init(void) | |||
1467 | IRQ_FLG_STD, | 1413 | IRQ_FLG_STD, |
1468 | "M68328_UART", NULL)) | 1414 | "M68328_UART", NULL)) |
1469 | panic("Unable to attach 68328 serial interrupt\n"); | 1415 | panic("Unable to attach 68328 serial interrupt\n"); |
1470 | #ifdef CONFIG_PM_LEGACY | ||
1471 | serial_pm[i] = pm_register(PM_SYS_DEV, PM_SYS_COM, serial_pm_callback); | ||
1472 | if (serial_pm[i]) | ||
1473 | serial_pm[i]->data = info; | ||
1474 | #endif | ||
1475 | } | 1416 | } |
1476 | local_irq_restore(flags); | 1417 | local_irq_restore(flags); |
1477 | return 0; | 1418 | return 0; |
diff --git a/drivers/spi/spi_s3c24xx.c b/drivers/spi/spi_s3c24xx.c index 7071ff8da63e..5cf48123e0ef 100644 --- a/drivers/spi/spi_s3c24xx.c +++ b/drivers/spi/spi_s3c24xx.c | |||
@@ -28,7 +28,7 @@ | |||
28 | #include <asm/hardware.h> | 28 | #include <asm/hardware.h> |
29 | 29 | ||
30 | #include <asm/arch/regs-gpio.h> | 30 | #include <asm/arch/regs-gpio.h> |
31 | #include <asm/arch/regs-spi.h> | 31 | #include <asm/plat-s3c24xx/regs-spi.h> |
32 | #include <asm/arch/spi.h> | 32 | #include <asm/arch/spi.h> |
33 | 33 | ||
34 | struct s3c24xx_spi { | 34 | struct s3c24xx_spi { |
diff --git a/drivers/video/chipsfb.c b/drivers/video/chipsfb.c index f48e8c534c87..6796ba62c3c6 100644 --- a/drivers/video/chipsfb.c +++ b/drivers/video/chipsfb.c | |||
@@ -24,6 +24,7 @@ | |||
24 | #include <linux/delay.h> | 24 | #include <linux/delay.h> |
25 | #include <linux/interrupt.h> | 25 | #include <linux/interrupt.h> |
26 | #include <linux/fb.h> | 26 | #include <linux/fb.h> |
27 | #include <linux/pm.h> | ||
27 | #include <linux/init.h> | 28 | #include <linux/init.h> |
28 | #include <linux/pci.h> | 29 | #include <linux/pci.h> |
29 | #include <linux/console.h> | 30 | #include <linux/console.h> |
@@ -458,7 +459,7 @@ static int chipsfb_pci_suspend(struct pci_dev *pdev, pm_message_t state) | |||
458 | 459 | ||
459 | if (state.event == pdev->dev.power.power_state.event) | 460 | if (state.event == pdev->dev.power.power_state.event) |
460 | return 0; | 461 | return 0; |
461 | if (state.event != PM_SUSPEND_MEM) | 462 | if (state.event != PM_EVENT_SUSPEND) |
462 | goto done; | 463 | goto done; |
463 | 464 | ||
464 | acquire_console_sem(); | 465 | acquire_console_sem(); |
diff --git a/drivers/video/tgafb.c b/drivers/video/tgafb.c index 89facb73edfc..d292a37ec7d6 100644 --- a/drivers/video/tgafb.c +++ b/drivers/video/tgafb.c | |||
@@ -849,7 +849,7 @@ tgafb_clut_imageblit(struct fb_info *info, const struct fb_image *image) | |||
849 | u32 *palette = ((u32 *)info->pseudo_palette); | 849 | u32 *palette = ((u32 *)info->pseudo_palette); |
850 | unsigned long pos, line_length, i, j; | 850 | unsigned long pos, line_length, i, j; |
851 | const unsigned char *data; | 851 | const unsigned char *data; |
852 | void *regs_base, *fb_base; | 852 | void __iomem *regs_base, *fb_base; |
853 | 853 | ||
854 | dx = image->dx; | 854 | dx = image->dx; |
855 | dy = image->dy; | 855 | dy = image->dy; |
diff --git a/drivers/w1/masters/ds1wm.c b/drivers/w1/masters/ds1wm.c index 763bc73e5070..4b696641ce33 100644 --- a/drivers/w1/masters/ds1wm.c +++ b/drivers/w1/masters/ds1wm.c | |||
@@ -85,7 +85,7 @@ static struct { | |||
85 | }; | 85 | }; |
86 | 86 | ||
87 | struct ds1wm_data { | 87 | struct ds1wm_data { |
88 | void *map; | 88 | void __iomem *map; |
89 | int bus_shift; /* # of shifts to calc register offsets */ | 89 | int bus_shift; /* # of shifts to calc register offsets */ |
90 | struct platform_device *pdev; | 90 | struct platform_device *pdev; |
91 | struct ds1wm_platform_data *pdata; | 91 | struct ds1wm_platform_data *pdata; |
diff --git a/drivers/xen/xenbus/xenbus_xs.c b/drivers/xen/xenbus/xenbus_xs.c index 9e943fbce81b..227d53b12a5c 100644 --- a/drivers/xen/xenbus/xenbus_xs.c +++ b/drivers/xen/xenbus/xenbus_xs.c | |||
@@ -782,8 +782,8 @@ static int process_msg(void) | |||
782 | msg->u.watch.vec = split(body, msg->hdr.len, | 782 | msg->u.watch.vec = split(body, msg->hdr.len, |
783 | &msg->u.watch.vec_size); | 783 | &msg->u.watch.vec_size); |
784 | if (IS_ERR(msg->u.watch.vec)) { | 784 | if (IS_ERR(msg->u.watch.vec)) { |
785 | kfree(msg); | ||
786 | err = PTR_ERR(msg->u.watch.vec); | 785 | err = PTR_ERR(msg->u.watch.vec); |
786 | kfree(msg); | ||
787 | goto out; | 787 | goto out; |
788 | } | 788 | } |
789 | 789 | ||
@@ -230,7 +230,7 @@ void bio_put(struct bio *bio) | |||
230 | } | 230 | } |
231 | } | 231 | } |
232 | 232 | ||
233 | inline int bio_phys_segments(request_queue_t *q, struct bio *bio) | 233 | inline int bio_phys_segments(struct request_queue *q, struct bio *bio) |
234 | { | 234 | { |
235 | if (unlikely(!bio_flagged(bio, BIO_SEG_VALID))) | 235 | if (unlikely(!bio_flagged(bio, BIO_SEG_VALID))) |
236 | blk_recount_segments(q, bio); | 236 | blk_recount_segments(q, bio); |
@@ -238,7 +238,7 @@ inline int bio_phys_segments(request_queue_t *q, struct bio *bio) | |||
238 | return bio->bi_phys_segments; | 238 | return bio->bi_phys_segments; |
239 | } | 239 | } |
240 | 240 | ||
241 | inline int bio_hw_segments(request_queue_t *q, struct bio *bio) | 241 | inline int bio_hw_segments(struct request_queue *q, struct bio *bio) |
242 | { | 242 | { |
243 | if (unlikely(!bio_flagged(bio, BIO_SEG_VALID))) | 243 | if (unlikely(!bio_flagged(bio, BIO_SEG_VALID))) |
244 | blk_recount_segments(q, bio); | 244 | blk_recount_segments(q, bio); |
@@ -257,7 +257,7 @@ inline int bio_hw_segments(request_queue_t *q, struct bio *bio) | |||
257 | */ | 257 | */ |
258 | void __bio_clone(struct bio *bio, struct bio *bio_src) | 258 | void __bio_clone(struct bio *bio, struct bio *bio_src) |
259 | { | 259 | { |
260 | request_queue_t *q = bdev_get_queue(bio_src->bi_bdev); | 260 | struct request_queue *q = bdev_get_queue(bio_src->bi_bdev); |
261 | 261 | ||
262 | memcpy(bio->bi_io_vec, bio_src->bi_io_vec, | 262 | memcpy(bio->bi_io_vec, bio_src->bi_io_vec, |
263 | bio_src->bi_max_vecs * sizeof(struct bio_vec)); | 263 | bio_src->bi_max_vecs * sizeof(struct bio_vec)); |
@@ -303,7 +303,7 @@ struct bio *bio_clone(struct bio *bio, gfp_t gfp_mask) | |||
303 | */ | 303 | */ |
304 | int bio_get_nr_vecs(struct block_device *bdev) | 304 | int bio_get_nr_vecs(struct block_device *bdev) |
305 | { | 305 | { |
306 | request_queue_t *q = bdev_get_queue(bdev); | 306 | struct request_queue *q = bdev_get_queue(bdev); |
307 | int nr_pages; | 307 | int nr_pages; |
308 | 308 | ||
309 | nr_pages = ((q->max_sectors << 9) + PAGE_SIZE - 1) >> PAGE_SHIFT; | 309 | nr_pages = ((q->max_sectors << 9) + PAGE_SIZE - 1) >> PAGE_SHIFT; |
@@ -315,7 +315,7 @@ int bio_get_nr_vecs(struct block_device *bdev) | |||
315 | return nr_pages; | 315 | return nr_pages; |
316 | } | 316 | } |
317 | 317 | ||
318 | static int __bio_add_page(request_queue_t *q, struct bio *bio, struct page | 318 | static int __bio_add_page(struct request_queue *q, struct bio *bio, struct page |
319 | *page, unsigned int len, unsigned int offset, | 319 | *page, unsigned int len, unsigned int offset, |
320 | unsigned short max_sectors) | 320 | unsigned short max_sectors) |
321 | { | 321 | { |
@@ -425,7 +425,7 @@ static int __bio_add_page(request_queue_t *q, struct bio *bio, struct page | |||
425 | * smaller than PAGE_SIZE, so it is always possible to add a single | 425 | * smaller than PAGE_SIZE, so it is always possible to add a single |
426 | * page to an empty bio. This should only be used by REQ_PC bios. | 426 | * page to an empty bio. This should only be used by REQ_PC bios. |
427 | */ | 427 | */ |
428 | int bio_add_pc_page(request_queue_t *q, struct bio *bio, struct page *page, | 428 | int bio_add_pc_page(struct request_queue *q, struct bio *bio, struct page *page, |
429 | unsigned int len, unsigned int offset) | 429 | unsigned int len, unsigned int offset) |
430 | { | 430 | { |
431 | return __bio_add_page(q, bio, page, len, offset, q->max_hw_sectors); | 431 | return __bio_add_page(q, bio, page, len, offset, q->max_hw_sectors); |
@@ -523,7 +523,7 @@ int bio_uncopy_user(struct bio *bio) | |||
523 | * to/from kernel pages as necessary. Must be paired with | 523 | * to/from kernel pages as necessary. Must be paired with |
524 | * call bio_uncopy_user() on io completion. | 524 | * call bio_uncopy_user() on io completion. |
525 | */ | 525 | */ |
526 | struct bio *bio_copy_user(request_queue_t *q, unsigned long uaddr, | 526 | struct bio *bio_copy_user(struct request_queue *q, unsigned long uaddr, |
527 | unsigned int len, int write_to_vm) | 527 | unsigned int len, int write_to_vm) |
528 | { | 528 | { |
529 | unsigned long end = (uaddr + len + PAGE_SIZE - 1) >> PAGE_SHIFT; | 529 | unsigned long end = (uaddr + len + PAGE_SIZE - 1) >> PAGE_SHIFT; |
@@ -600,7 +600,7 @@ out_bmd: | |||
600 | return ERR_PTR(ret); | 600 | return ERR_PTR(ret); |
601 | } | 601 | } |
602 | 602 | ||
603 | static struct bio *__bio_map_user_iov(request_queue_t *q, | 603 | static struct bio *__bio_map_user_iov(struct request_queue *q, |
604 | struct block_device *bdev, | 604 | struct block_device *bdev, |
605 | struct sg_iovec *iov, int iov_count, | 605 | struct sg_iovec *iov, int iov_count, |
606 | int write_to_vm) | 606 | int write_to_vm) |
@@ -712,7 +712,7 @@ static struct bio *__bio_map_user_iov(request_queue_t *q, | |||
712 | 712 | ||
713 | /** | 713 | /** |
714 | * bio_map_user - map user address into bio | 714 | * bio_map_user - map user address into bio |
715 | * @q: the request_queue_t for the bio | 715 | * @q: the struct request_queue for the bio |
716 | * @bdev: destination block device | 716 | * @bdev: destination block device |
717 | * @uaddr: start of user address | 717 | * @uaddr: start of user address |
718 | * @len: length in bytes | 718 | * @len: length in bytes |
@@ -721,7 +721,7 @@ static struct bio *__bio_map_user_iov(request_queue_t *q, | |||
721 | * Map the user space address into a bio suitable for io to a block | 721 | * Map the user space address into a bio suitable for io to a block |
722 | * device. Returns an error pointer in case of error. | 722 | * device. Returns an error pointer in case of error. |
723 | */ | 723 | */ |
724 | struct bio *bio_map_user(request_queue_t *q, struct block_device *bdev, | 724 | struct bio *bio_map_user(struct request_queue *q, struct block_device *bdev, |
725 | unsigned long uaddr, unsigned int len, int write_to_vm) | 725 | unsigned long uaddr, unsigned int len, int write_to_vm) |
726 | { | 726 | { |
727 | struct sg_iovec iov; | 727 | struct sg_iovec iov; |
@@ -734,7 +734,7 @@ struct bio *bio_map_user(request_queue_t *q, struct block_device *bdev, | |||
734 | 734 | ||
735 | /** | 735 | /** |
736 | * bio_map_user_iov - map user sg_iovec table into bio | 736 | * bio_map_user_iov - map user sg_iovec table into bio |
737 | * @q: the request_queue_t for the bio | 737 | * @q: the struct request_queue for the bio |
738 | * @bdev: destination block device | 738 | * @bdev: destination block device |
739 | * @iov: the iovec. | 739 | * @iov: the iovec. |
740 | * @iov_count: number of elements in the iovec | 740 | * @iov_count: number of elements in the iovec |
@@ -743,7 +743,7 @@ struct bio *bio_map_user(request_queue_t *q, struct block_device *bdev, | |||
743 | * Map the user space address into a bio suitable for io to a block | 743 | * Map the user space address into a bio suitable for io to a block |
744 | * device. Returns an error pointer in case of error. | 744 | * device. Returns an error pointer in case of error. |
745 | */ | 745 | */ |
746 | struct bio *bio_map_user_iov(request_queue_t *q, struct block_device *bdev, | 746 | struct bio *bio_map_user_iov(struct request_queue *q, struct block_device *bdev, |
747 | struct sg_iovec *iov, int iov_count, | 747 | struct sg_iovec *iov, int iov_count, |
748 | int write_to_vm) | 748 | int write_to_vm) |
749 | { | 749 | { |
@@ -808,7 +808,7 @@ static int bio_map_kern_endio(struct bio *bio, unsigned int bytes_done, int err) | |||
808 | } | 808 | } |
809 | 809 | ||
810 | 810 | ||
811 | static struct bio *__bio_map_kern(request_queue_t *q, void *data, | 811 | static struct bio *__bio_map_kern(struct request_queue *q, void *data, |
812 | unsigned int len, gfp_t gfp_mask) | 812 | unsigned int len, gfp_t gfp_mask) |
813 | { | 813 | { |
814 | unsigned long kaddr = (unsigned long)data; | 814 | unsigned long kaddr = (unsigned long)data; |
@@ -847,7 +847,7 @@ static struct bio *__bio_map_kern(request_queue_t *q, void *data, | |||
847 | 847 | ||
848 | /** | 848 | /** |
849 | * bio_map_kern - map kernel address into bio | 849 | * bio_map_kern - map kernel address into bio |
850 | * @q: the request_queue_t for the bio | 850 | * @q: the struct request_queue for the bio |
851 | * @data: pointer to buffer to map | 851 | * @data: pointer to buffer to map |
852 | * @len: length in bytes | 852 | * @len: length in bytes |
853 | * @gfp_mask: allocation flags for bio allocation | 853 | * @gfp_mask: allocation flags for bio allocation |
@@ -855,7 +855,7 @@ static struct bio *__bio_map_kern(request_queue_t *q, void *data, | |||
855 | * Map the kernel address into a bio suitable for io to a block | 855 | * Map the kernel address into a bio suitable for io to a block |
856 | * device. Returns an error pointer in case of error. | 856 | * device. Returns an error pointer in case of error. |
857 | */ | 857 | */ |
858 | struct bio *bio_map_kern(request_queue_t *q, void *data, unsigned int len, | 858 | struct bio *bio_map_kern(struct request_queue *q, void *data, unsigned int len, |
859 | gfp_t gfp_mask) | 859 | gfp_t gfp_mask) |
860 | { | 860 | { |
861 | struct bio *bio; | 861 | struct bio *bio; |
diff --git a/fs/ext2/super.c b/fs/ext2/super.c index 68579a0ed3f0..639a32c3c9c1 100644 --- a/fs/ext2/super.c +++ b/fs/ext2/super.c | |||
@@ -580,7 +580,7 @@ static int ext2_check_descriptors (struct super_block * sb) | |||
580 | return 0; | 580 | return 0; |
581 | } | 581 | } |
582 | if (le32_to_cpu(gdp->bg_inode_table) < first_block || | 582 | if (le32_to_cpu(gdp->bg_inode_table) < first_block || |
583 | le32_to_cpu(gdp->bg_inode_table) + sbi->s_itb_per_group > | 583 | le32_to_cpu(gdp->bg_inode_table) + sbi->s_itb_per_group - 1 > |
584 | last_block) | 584 | last_block) |
585 | { | 585 | { |
586 | ext2_error (sb, "ext2_check_descriptors", | 586 | ext2_error (sb, "ext2_check_descriptors", |
diff --git a/fs/ext3/super.c b/fs/ext3/super.c index f0614e3f1fe8..22cfdd61c060 100644 --- a/fs/ext3/super.c +++ b/fs/ext3/super.c | |||
@@ -1221,7 +1221,7 @@ static int ext3_check_descriptors (struct super_block * sb) | |||
1221 | return 0; | 1221 | return 0; |
1222 | } | 1222 | } |
1223 | if (le32_to_cpu(gdp->bg_inode_table) < first_block || | 1223 | if (le32_to_cpu(gdp->bg_inode_table) < first_block || |
1224 | le32_to_cpu(gdp->bg_inode_table) + sbi->s_itb_per_group > | 1224 | le32_to_cpu(gdp->bg_inode_table) + sbi->s_itb_per_group - 1 > |
1225 | last_block) | 1225 | last_block) |
1226 | { | 1226 | { |
1227 | ext3_error (sb, "ext3_check_descriptors", | 1227 | ext3_error (sb, "ext3_check_descriptors", |
diff --git a/fs/ext4/super.c b/fs/ext4/super.c index 75adbb64e028..4550b83ab1c9 100644 --- a/fs/ext4/super.c +++ b/fs/ext4/super.c | |||
@@ -1283,7 +1283,7 @@ static int ext4_check_descriptors (struct super_block * sb) | |||
1283 | } | 1283 | } |
1284 | inode_table = ext4_inode_table(sb, gdp); | 1284 | inode_table = ext4_inode_table(sb, gdp); |
1285 | if (inode_table < first_block || | 1285 | if (inode_table < first_block || |
1286 | inode_table + sbi->s_itb_per_group > last_block) | 1286 | inode_table + sbi->s_itb_per_group - 1 > last_block) |
1287 | { | 1287 | { |
1288 | ext4_error (sb, "ext4_check_descriptors", | 1288 | ext4_error (sb, "ext4_check_descriptors", |
1289 | "Inode table for group %d" | 1289 | "Inode table for group %d" |
diff --git a/fs/lockd/svclock.c b/fs/lockd/svclock.c index b3efa4536cc5..a21e4bc5444b 100644 --- a/fs/lockd/svclock.c +++ b/fs/lockd/svclock.c | |||
@@ -335,10 +335,10 @@ static void nlmsvc_freegrantargs(struct nlm_rqst *call) | |||
335 | /* | 335 | /* |
336 | * Deferred lock request handling for non-blocking lock | 336 | * Deferred lock request handling for non-blocking lock |
337 | */ | 337 | */ |
338 | static u32 | 338 | static __be32 |
339 | nlmsvc_defer_lock_rqst(struct svc_rqst *rqstp, struct nlm_block *block) | 339 | nlmsvc_defer_lock_rqst(struct svc_rqst *rqstp, struct nlm_block *block) |
340 | { | 340 | { |
341 | u32 status = nlm_lck_denied_nolocks; | 341 | __be32 status = nlm_lck_denied_nolocks; |
342 | 342 | ||
343 | block->b_flags |= B_QUEUED; | 343 | block->b_flags |= B_QUEUED; |
344 | 344 | ||
@@ -352,7 +352,7 @@ nlmsvc_defer_lock_rqst(struct svc_rqst *rqstp, struct nlm_block *block) | |||
352 | status = nlm_drop_reply; | 352 | status = nlm_drop_reply; |
353 | } | 353 | } |
354 | dprintk("lockd: nlmsvc_defer_lock_rqst block %p flags %d status %d\n", | 354 | dprintk("lockd: nlmsvc_defer_lock_rqst block %p flags %d status %d\n", |
355 | block, block->b_flags, status); | 355 | block, block->b_flags, ntohl(status)); |
356 | 356 | ||
357 | return status; | 357 | return status; |
358 | } | 358 | } |
diff --git a/fs/nfsd/nfs4xdr.c b/fs/nfsd/nfs4xdr.c index b3d55c6747fd..8ef0964179bc 100644 --- a/fs/nfsd/nfs4xdr.c +++ b/fs/nfsd/nfs4xdr.c | |||
@@ -2450,7 +2450,7 @@ nfsd4_encode_rename(struct nfsd4_compoundres *resp, __be32 nfserr, struct nfsd4_ | |||
2450 | } | 2450 | } |
2451 | 2451 | ||
2452 | static void | 2452 | static void |
2453 | nfsd4_encode_secinfo(struct nfsd4_compoundres *resp, int nfserr, | 2453 | nfsd4_encode_secinfo(struct nfsd4_compoundres *resp, __be32 nfserr, |
2454 | struct nfsd4_secinfo *secinfo) | 2454 | struct nfsd4_secinfo *secinfo) |
2455 | { | 2455 | { |
2456 | int i = 0; | 2456 | int i = 0; |
diff --git a/fs/ocfs2/file.c b/fs/ocfs2/file.c index 5727cd18302a..c4034f693e7b 100644 --- a/fs/ocfs2/file.c +++ b/fs/ocfs2/file.c | |||
@@ -2153,7 +2153,7 @@ static int ocfs2_splice_write_actor(struct pipe_inode_info *pipe, | |||
2153 | src = buf->ops->map(pipe, buf, 1); | 2153 | src = buf->ops->map(pipe, buf, 1); |
2154 | dst = kmap_atomic(page, KM_USER1); | 2154 | dst = kmap_atomic(page, KM_USER1); |
2155 | memcpy(dst + offset, src + buf->offset, count); | 2155 | memcpy(dst + offset, src + buf->offset, count); |
2156 | kunmap_atomic(page, KM_USER1); | 2156 | kunmap_atomic(dst, KM_USER1); |
2157 | buf->ops->unmap(pipe, buf, src); | 2157 | buf->ops->unmap(pipe, buf, src); |
2158 | 2158 | ||
2159 | copied = ocfs2_write_end(file, file->f_mapping, sd->pos, count, count, | 2159 | copied = ocfs2_write_end(file, file->f_mapping, sd->pos, count, count, |
diff --git a/fs/reiserfs/stree.c b/fs/reiserfs/stree.c index b6f12593c39d..981027d1187b 100644 --- a/fs/reiserfs/stree.c +++ b/fs/reiserfs/stree.c | |||
@@ -1042,7 +1042,8 @@ static char prepare_for_delete_or_cut(struct reiserfs_transaction_handle *th, st | |||
1042 | pos = I_UNFM_NUM(&s_ih); | 1042 | pos = I_UNFM_NUM(&s_ih); |
1043 | 1043 | ||
1044 | while (le_ih_k_offset (&s_ih) + (pos - 1) * blk_size > n_new_file_length) { | 1044 | while (le_ih_k_offset (&s_ih) + (pos - 1) * blk_size > n_new_file_length) { |
1045 | __u32 *unfm, block; | 1045 | __le32 *unfm; |
1046 | __u32 block; | ||
1046 | 1047 | ||
1047 | /* Each unformatted block deletion may involve one additional | 1048 | /* Each unformatted block deletion may involve one additional |
1048 | * bitmap block into the transaction, thereby the initial | 1049 | * bitmap block into the transaction, thereby the initial |
@@ -1052,7 +1053,7 @@ static char prepare_for_delete_or_cut(struct reiserfs_transaction_handle *th, st | |||
1052 | break; | 1053 | break; |
1053 | } | 1054 | } |
1054 | 1055 | ||
1055 | unfm = (__u32 *)B_I_PITEM(p_s_bh, &s_ih) + pos - 1; | 1056 | unfm = (__le32 *)B_I_PITEM(p_s_bh, &s_ih) + pos - 1; |
1056 | block = get_block_num(unfm, 0); | 1057 | block = get_block_num(unfm, 0); |
1057 | 1058 | ||
1058 | if (block != 0) { | 1059 | if (block != 0) { |
diff --git a/fs/signalfd.c b/fs/signalfd.c index 3b07f26d984d..7b941abbcde0 100644 --- a/fs/signalfd.c +++ b/fs/signalfd.c | |||
@@ -320,7 +320,7 @@ asmlinkage long sys_signalfd(int ufd, sigset_t __user *user_mask, size_t sizemas | |||
320 | 320 | ||
321 | if (sizemask != sizeof(sigset_t) || | 321 | if (sizemask != sizeof(sigset_t) || |
322 | copy_from_user(&sigmask, user_mask, sizeof(sigmask))) | 322 | copy_from_user(&sigmask, user_mask, sizeof(sigmask))) |
323 | return error = -EINVAL; | 323 | return -EINVAL; |
324 | sigdelsetmask(&sigmask, sigmask(SIGKILL) | sigmask(SIGSTOP)); | 324 | sigdelsetmask(&sigmask, sigmask(SIGKILL) | sigmask(SIGSTOP)); |
325 | signotset(&sigmask); | 325 | signotset(&sigmask); |
326 | 326 | ||
diff --git a/fs/timerfd.c b/fs/timerfd.c index af9eca5c0230..61983f3b107c 100644 --- a/fs/timerfd.c +++ b/fs/timerfd.c | |||
@@ -95,7 +95,7 @@ static ssize_t timerfd_read(struct file *file, char __user *buf, size_t count, | |||
95 | { | 95 | { |
96 | struct timerfd_ctx *ctx = file->private_data; | 96 | struct timerfd_ctx *ctx = file->private_data; |
97 | ssize_t res; | 97 | ssize_t res; |
98 | u32 ticks = 0; | 98 | u64 ticks = 0; |
99 | DECLARE_WAITQUEUE(wait, current); | 99 | DECLARE_WAITQUEUE(wait, current); |
100 | 100 | ||
101 | if (count < sizeof(ticks)) | 101 | if (count < sizeof(ticks)) |
@@ -130,7 +130,7 @@ static ssize_t timerfd_read(struct file *file, char __user *buf, size_t count, | |||
130 | * callback to avoid DoS attacks specifying a very | 130 | * callback to avoid DoS attacks specifying a very |
131 | * short timer period. | 131 | * short timer period. |
132 | */ | 132 | */ |
133 | ticks = (u32) | 133 | ticks = (u64) |
134 | hrtimer_forward(&ctx->tmr, | 134 | hrtimer_forward(&ctx->tmr, |
135 | hrtimer_cb_get_time(&ctx->tmr), | 135 | hrtimer_cb_get_time(&ctx->tmr), |
136 | ctx->tintv); | 136 | ctx->tintv); |
@@ -140,7 +140,7 @@ static ssize_t timerfd_read(struct file *file, char __user *buf, size_t count, | |||
140 | } | 140 | } |
141 | spin_unlock_irq(&ctx->wqh.lock); | 141 | spin_unlock_irq(&ctx->wqh.lock); |
142 | if (ticks) | 142 | if (ticks) |
143 | res = put_user(ticks, buf) ? -EFAULT: sizeof(ticks); | 143 | res = put_user(ticks, (u64 __user *) buf) ? -EFAULT: sizeof(ticks); |
144 | return res; | 144 | return res; |
145 | } | 145 | } |
146 | 146 | ||
diff --git a/fs/xfs/linux-2.6/xfs_ioctl32.c b/fs/xfs/linux-2.6/xfs_ioctl32.c index 141cf15067c2..42319d75aaab 100644 --- a/fs/xfs/linux-2.6/xfs_ioctl32.c +++ b/fs/xfs/linux-2.6/xfs_ioctl32.c | |||
@@ -139,7 +139,7 @@ STATIC int xfs_inumbers_fmt_compat( | |||
139 | long count, | 139 | long count, |
140 | long *written) | 140 | long *written) |
141 | { | 141 | { |
142 | compat_xfs_inogrp_t *p32 = ubuffer; | 142 | compat_xfs_inogrp_t __user *p32 = ubuffer; |
143 | long i; | 143 | long i; |
144 | 144 | ||
145 | for (i = 0; i < count; i++) { | 145 | for (i = 0; i < count; i++) { |
@@ -444,7 +444,7 @@ xfs_compat_ioctl( | |||
444 | case XFS_IOC_FSINUMBERS_32: | 444 | case XFS_IOC_FSINUMBERS_32: |
445 | cmd = _NATIVE_IOC(cmd, struct xfs_fsop_bulkreq); | 445 | cmd = _NATIVE_IOC(cmd, struct xfs_fsop_bulkreq); |
446 | return xfs_ioc_bulkstat_compat(XFS_BHVTOI(VNHEAD(vp))->i_mount, | 446 | return xfs_ioc_bulkstat_compat(XFS_BHVTOI(VNHEAD(vp))->i_mount, |
447 | cmd, (void*)arg); | 447 | cmd, (void __user*)arg); |
448 | case XFS_IOC_FD_TO_HANDLE_32: | 448 | case XFS_IOC_FD_TO_HANDLE_32: |
449 | case XFS_IOC_PATH_TO_HANDLE_32: | 449 | case XFS_IOC_PATH_TO_HANDLE_32: |
450 | case XFS_IOC_PATH_TO_FSHANDLE_32: | 450 | case XFS_IOC_PATH_TO_FSHANDLE_32: |
diff --git a/include/acpi/acpi_bus.h b/include/acpi/acpi_bus.h index 5e3dcf3299bf..533ef40f7ccf 100644 --- a/include/acpi/acpi_bus.h +++ b/include/acpi/acpi_bus.h | |||
@@ -131,7 +131,7 @@ struct acpi_device_ops { | |||
131 | struct acpi_driver { | 131 | struct acpi_driver { |
132 | char name[80]; | 132 | char name[80]; |
133 | char class[80]; | 133 | char class[80]; |
134 | char *ids; /* Supported Hardware IDs */ | 134 | const struct acpi_device_id *ids; /* Supported Hardware IDs */ |
135 | struct acpi_device_ops ops; | 135 | struct acpi_device_ops ops; |
136 | struct device_driver drv; | 136 | struct device_driver drv; |
137 | struct module *owner; | 137 | struct module *owner; |
@@ -341,7 +341,8 @@ int acpi_bus_add(struct acpi_device **child, struct acpi_device *parent, | |||
341 | int acpi_bus_trim(struct acpi_device *start, int rmdevice); | 341 | int acpi_bus_trim(struct acpi_device *start, int rmdevice); |
342 | int acpi_bus_start(struct acpi_device *device); | 342 | int acpi_bus_start(struct acpi_device *device); |
343 | acpi_status acpi_bus_get_ejd(acpi_handle handle, acpi_handle * ejd); | 343 | acpi_status acpi_bus_get_ejd(acpi_handle handle, acpi_handle * ejd); |
344 | int acpi_match_ids(struct acpi_device *device, char *ids); | 344 | int acpi_match_device_ids(struct acpi_device *device, |
345 | const struct acpi_device_id *ids); | ||
345 | int acpi_create_dir(struct acpi_device *); | 346 | int acpi_create_dir(struct acpi_device *); |
346 | void acpi_remove_dir(struct acpi_device *); | 347 | void acpi_remove_dir(struct acpi_device *); |
347 | 348 | ||
@@ -365,6 +366,8 @@ acpi_handle acpi_get_child(acpi_handle, acpi_integer); | |||
365 | acpi_handle acpi_get_pci_rootbridge_handle(unsigned int, unsigned int); | 366 | acpi_handle acpi_get_pci_rootbridge_handle(unsigned int, unsigned int); |
366 | #define DEVICE_ACPI_HANDLE(dev) ((acpi_handle)((dev)->archdata.acpi_handle)) | 367 | #define DEVICE_ACPI_HANDLE(dev) ((acpi_handle)((dev)->archdata.acpi_handle)) |
367 | 368 | ||
369 | int acpi_pm_device_sleep_state(struct device *, int, int *); | ||
370 | |||
368 | #endif /* CONFIG_ACPI */ | 371 | #endif /* CONFIG_ACPI */ |
369 | 372 | ||
370 | #endif /*__ACPI_BUS_H__*/ | 373 | #endif /*__ACPI_BUS_H__*/ |
diff --git a/include/acpi/acpi_drivers.h b/include/acpi/acpi_drivers.h index 553515912c0b..f85f77a538aa 100644 --- a/include/acpi/acpi_drivers.h +++ b/include/acpi/acpi_drivers.h | |||
@@ -34,16 +34,21 @@ | |||
34 | #define ACPI_BUS_COMPONENT 0x00010000 | 34 | #define ACPI_BUS_COMPONENT 0x00010000 |
35 | #define ACPI_SYSTEM_COMPONENT 0x02000000 | 35 | #define ACPI_SYSTEM_COMPONENT 0x02000000 |
36 | 36 | ||
37 | /* _HID definitions */ | 37 | /* |
38 | * _HID definitions | ||
39 | * HIDs must conform to ACPI spec(6.1.4) | ||
40 | * Linux specific HIDs do not apply to this and begin with LNX: | ||
41 | */ | ||
38 | 42 | ||
39 | #define ACPI_POWER_HID "power_resource" | 43 | #define ACPI_POWER_HID "LNXPOWER" |
40 | #define ACPI_PROCESSOR_HID "ACPI0007" | 44 | #define ACPI_PROCESSOR_HID "ACPI0007" |
41 | #define ACPI_SYSTEM_HID "acpi_system" | 45 | #define ACPI_SYSTEM_HID "LNXSYSTM" |
42 | #define ACPI_THERMAL_HID "thermal" | 46 | #define ACPI_THERMAL_HID "LNXTHERM" |
43 | #define ACPI_BUTTON_HID_POWERF "button_power" | 47 | #define ACPI_BUTTON_HID_POWERF "LNXPWRBN" |
44 | #define ACPI_BUTTON_HID_SLEEPF "button_sleep" | 48 | #define ACPI_BUTTON_HID_SLEEPF "LNXSLPBN" |
45 | #define ACPI_VIDEO_HID "video" | 49 | #define ACPI_VIDEO_HID "LNXVIDEO" |
46 | #define ACPI_BAY_HID "bay" | 50 | #define ACPI_BAY_HID "LNXIOBAY" |
51 | |||
47 | /* -------------------------------------------------------------------------- | 52 | /* -------------------------------------------------------------------------- |
48 | PCI | 53 | PCI |
49 | -------------------------------------------------------------------------- */ | 54 | -------------------------------------------------------------------------- */ |
@@ -142,10 +147,6 @@ static inline void unregister_hotplug_dock_device(acpi_handle handle) | |||
142 | /*-------------------------------------------------------------------------- | 147 | /*-------------------------------------------------------------------------- |
143 | Suspend/Resume | 148 | Suspend/Resume |
144 | -------------------------------------------------------------------------- */ | 149 | -------------------------------------------------------------------------- */ |
145 | #ifdef CONFIG_ACPI_SLEEP | ||
146 | extern int acpi_sleep_init(void); | 150 | extern int acpi_sleep_init(void); |
147 | #else | ||
148 | #define acpi_sleep_init() do {} while (0) | ||
149 | #endif | ||
150 | 151 | ||
151 | #endif /*__ACPI_DRIVERS_H__*/ | 152 | #endif /*__ACPI_DRIVERS_H__*/ |
diff --git a/include/acpi/actypes.h b/include/acpi/actypes.h index fe8abc276437..e73a38939120 100644 --- a/include/acpi/actypes.h +++ b/include/acpi/actypes.h | |||
@@ -809,7 +809,7 @@ acpi_status(*acpi_walk_callback) (acpi_handle obj_handle, | |||
809 | 809 | ||
810 | /* Common string version of device HIDs and UIDs */ | 810 | /* Common string version of device HIDs and UIDs */ |
811 | 811 | ||
812 | struct acpi_device_id { | 812 | struct acpica_device_id { |
813 | char value[ACPI_DEVICE_ID_LENGTH]; | 813 | char value[ACPI_DEVICE_ID_LENGTH]; |
814 | }; | 814 | }; |
815 | 815 | ||
@@ -859,8 +859,8 @@ struct acpi_device_info { | |||
859 | u32 valid; /* Indicates which fields below are valid */ | 859 | u32 valid; /* Indicates which fields below are valid */ |
860 | u32 current_status; /* _STA value */ | 860 | u32 current_status; /* _STA value */ |
861 | acpi_integer address; /* _ADR value if any */ | 861 | acpi_integer address; /* _ADR value if any */ |
862 | struct acpi_device_id hardware_id; /* _HID value if any */ | 862 | struct acpica_device_id hardware_id; /* _HID value if any */ |
863 | struct acpi_device_id unique_id; /* _UID value if any */ | 863 | struct acpica_device_id unique_id; /* _UID value if any */ |
864 | u8 highest_dstates[4]; /* _sx_d values: 0xFF indicates not valid */ | 864 | u8 highest_dstates[4]; /* _sx_d values: 0xFF indicates not valid */ |
865 | struct acpi_compatible_id_list compatibility_id; /* List of _CIDs if any */ | 865 | struct acpi_compatible_id_list compatibility_id; /* List of _CIDs if any */ |
866 | }; | 866 | }; |
diff --git a/include/acpi/acutils.h b/include/acpi/acutils.h index a87ef1c8d46b..a2918547c73f 100644 --- a/include/acpi/acutils.h +++ b/include/acpi/acutils.h | |||
@@ -354,7 +354,7 @@ acpi_ut_evaluate_numeric_object(char *object_name, | |||
354 | 354 | ||
355 | acpi_status | 355 | acpi_status |
356 | acpi_ut_execute_HID(struct acpi_namespace_node *device_node, | 356 | acpi_ut_execute_HID(struct acpi_namespace_node *device_node, |
357 | struct acpi_device_id *hid); | 357 | struct acpica_device_id *hid); |
358 | 358 | ||
359 | acpi_status | 359 | acpi_status |
360 | acpi_ut_execute_CID(struct acpi_namespace_node *device_node, | 360 | acpi_ut_execute_CID(struct acpi_namespace_node *device_node, |
@@ -366,7 +366,7 @@ acpi_ut_execute_STA(struct acpi_namespace_node *device_node, | |||
366 | 366 | ||
367 | acpi_status | 367 | acpi_status |
368 | acpi_ut_execute_UID(struct acpi_namespace_node *device_node, | 368 | acpi_ut_execute_UID(struct acpi_namespace_node *device_node, |
369 | struct acpi_device_id *uid); | 369 | struct acpica_device_id *uid); |
370 | 370 | ||
371 | acpi_status | 371 | acpi_status |
372 | acpi_ut_execute_sxds(struct acpi_namespace_node *device_node, u8 * highest); | 372 | acpi_ut_execute_sxds(struct acpi_namespace_node *device_node, u8 * highest); |
diff --git a/include/asm-alpha/bitops.h b/include/asm-alpha/bitops.h index 3a0cbeb03fa1..9e71201000d5 100644 --- a/include/asm-alpha/bitops.h +++ b/include/asm-alpha/bitops.h | |||
@@ -324,7 +324,7 @@ static inline int fls64(unsigned long x) | |||
324 | { | 324 | { |
325 | unsigned long t, a, r; | 325 | unsigned long t, a, r; |
326 | 326 | ||
327 | t = __kernel_cmpbge (x, 0x0101010101010101); | 327 | t = __kernel_cmpbge (x, 0x0101010101010101UL); |
328 | a = __flsm1_tab[t]; | 328 | a = __flsm1_tab[t]; |
329 | t = __kernel_extbl (x, a); | 329 | t = __kernel_extbl (x, a); |
330 | r = a*8 + __flsm1_tab[t] + (x != 0); | 330 | r = a*8 + __flsm1_tab[t] + (x != 0); |
diff --git a/include/asm-arm/arch-omap/mailbox.h b/include/asm-arm/arch-omap/mailbox.h index 4bf0909461f2..7cbed9332e16 100644 --- a/include/asm-arm/arch-omap/mailbox.h +++ b/include/asm-arm/arch-omap/mailbox.h | |||
@@ -37,7 +37,7 @@ struct omap_mbox_ops { | |||
37 | 37 | ||
38 | struct omap_mbox_queue { | 38 | struct omap_mbox_queue { |
39 | spinlock_t lock; | 39 | spinlock_t lock; |
40 | request_queue_t *queue; | 40 | struct request_queue *queue; |
41 | struct work_struct work; | 41 | struct work_struct work; |
42 | int (*callback)(void *); | 42 | int (*callback)(void *); |
43 | struct omap_mbox *mbox; | 43 | struct omap_mbox *mbox; |
diff --git a/include/asm-arm/unaligned.h b/include/asm-arm/unaligned.h index 795b9e5b9e6a..8431f6eed5c6 100644 --- a/include/asm-arm/unaligned.h +++ b/include/asm-arm/unaligned.h | |||
@@ -60,24 +60,24 @@ extern int __bug_unaligned_x(const void *ptr); | |||
60 | __get_unaligned_4_be((__p+4))) | 60 | __get_unaligned_4_be((__p+4))) |
61 | 61 | ||
62 | #define __get_unaligned_le(ptr) \ | 62 | #define __get_unaligned_le(ptr) \ |
63 | ({ \ | 63 | ((__force typeof(*(ptr)))({ \ |
64 | const __u8 *__p = (const __u8 *)(ptr); \ | 64 | const __u8 *__p = (const __u8 *)(ptr); \ |
65 | __builtin_choose_expr(sizeof(*(ptr)) == 1, *__p, \ | 65 | __builtin_choose_expr(sizeof(*(ptr)) == 1, *__p, \ |
66 | __builtin_choose_expr(sizeof(*(ptr)) == 2, __get_unaligned_2_le(__p), \ | 66 | __builtin_choose_expr(sizeof(*(ptr)) == 2, __get_unaligned_2_le(__p), \ |
67 | __builtin_choose_expr(sizeof(*(ptr)) == 4, __get_unaligned_4_le(__p), \ | 67 | __builtin_choose_expr(sizeof(*(ptr)) == 4, __get_unaligned_4_le(__p), \ |
68 | __builtin_choose_expr(sizeof(*(ptr)) == 8, __get_unaligned_8_le(__p), \ | 68 | __builtin_choose_expr(sizeof(*(ptr)) == 8, __get_unaligned_8_le(__p), \ |
69 | (void)__bug_unaligned_x(__p))))); \ | 69 | (void)__bug_unaligned_x(__p))))); \ |
70 | }) | 70 | })) |
71 | 71 | ||
72 | #define __get_unaligned_be(ptr) \ | 72 | #define __get_unaligned_be(ptr) \ |
73 | ({ \ | 73 | ((__force typeof(*(ptr)))({ \ |
74 | const __u8 *__p = (const __u8 *)(ptr); \ | 74 | const __u8 *__p = (const __u8 *)(ptr); \ |
75 | __builtin_choose_expr(sizeof(*(ptr)) == 1, *__p, \ | 75 | __builtin_choose_expr(sizeof(*(ptr)) == 1, *__p, \ |
76 | __builtin_choose_expr(sizeof(*(ptr)) == 2, __get_unaligned_2_be(__p), \ | 76 | __builtin_choose_expr(sizeof(*(ptr)) == 2, __get_unaligned_2_be(__p), \ |
77 | __builtin_choose_expr(sizeof(*(ptr)) == 4, __get_unaligned_4_be(__p), \ | 77 | __builtin_choose_expr(sizeof(*(ptr)) == 4, __get_unaligned_4_be(__p), \ |
78 | __builtin_choose_expr(sizeof(*(ptr)) == 8, __get_unaligned_8_be(__p), \ | 78 | __builtin_choose_expr(sizeof(*(ptr)) == 8, __get_unaligned_8_be(__p), \ |
79 | (void)__bug_unaligned_x(__p))))); \ | 79 | (void)__bug_unaligned_x(__p))))); \ |
80 | }) | 80 | })) |
81 | 81 | ||
82 | 82 | ||
83 | static inline void __put_unaligned_2_le(__u32 __v, register __u8 *__p) | 83 | static inline void __put_unaligned_2_le(__u32 __v, register __u8 *__p) |
@@ -131,15 +131,16 @@ static inline void __put_unaligned_8_be(const unsigned long long __v, register _ | |||
131 | */ | 131 | */ |
132 | #define __put_unaligned_le(val,ptr) \ | 132 | #define __put_unaligned_le(val,ptr) \ |
133 | ({ \ | 133 | ({ \ |
134 | (void)sizeof(*(ptr) = (val)); \ | ||
134 | switch (sizeof(*(ptr))) { \ | 135 | switch (sizeof(*(ptr))) { \ |
135 | case 1: \ | 136 | case 1: \ |
136 | *(ptr) = (val); \ | 137 | *(ptr) = (val); \ |
137 | break; \ | 138 | break; \ |
138 | case 2: __put_unaligned_2_le((val),(__u8 *)(ptr)); \ | 139 | case 2: __put_unaligned_2_le((__force u16)(val),(__u8 *)(ptr)); \ |
139 | break; \ | 140 | break; \ |
140 | case 4: __put_unaligned_4_le((val),(__u8 *)(ptr)); \ | 141 | case 4: __put_unaligned_4_le((__force u32)(val),(__u8 *)(ptr)); \ |
141 | break; \ | 142 | break; \ |
142 | case 8: __put_unaligned_8_le((val),(__u8 *)(ptr)); \ | 143 | case 8: __put_unaligned_8_le((__force u64)(val),(__u8 *)(ptr)); \ |
143 | break; \ | 144 | break; \ |
144 | default: __bug_unaligned_x(ptr); \ | 145 | default: __bug_unaligned_x(ptr); \ |
145 | break; \ | 146 | break; \ |
@@ -149,15 +150,16 @@ static inline void __put_unaligned_8_be(const unsigned long long __v, register _ | |||
149 | 150 | ||
150 | #define __put_unaligned_be(val,ptr) \ | 151 | #define __put_unaligned_be(val,ptr) \ |
151 | ({ \ | 152 | ({ \ |
153 | (void)sizeof(*(ptr) = (val)); \ | ||
152 | switch (sizeof(*(ptr))) { \ | 154 | switch (sizeof(*(ptr))) { \ |
153 | case 1: \ | 155 | case 1: \ |
154 | *(ptr) = (val); \ | 156 | *(ptr) = (val); \ |
155 | break; \ | 157 | break; \ |
156 | case 2: __put_unaligned_2_be((val),(__u8 *)(ptr)); \ | 158 | case 2: __put_unaligned_2_be((__force u16)(val),(__u8 *)(ptr)); \ |
157 | break; \ | 159 | break; \ |
158 | case 4: __put_unaligned_4_be((val),(__u8 *)(ptr)); \ | 160 | case 4: __put_unaligned_4_be((__force u32)(val),(__u8 *)(ptr)); \ |
159 | break; \ | 161 | break; \ |
160 | case 8: __put_unaligned_8_be((val),(__u8 *)(ptr)); \ | 162 | case 8: __put_unaligned_8_be((__force u64)(val),(__u8 *)(ptr)); \ |
161 | break; \ | 163 | break; \ |
162 | default: __bug_unaligned_x(ptr); \ | 164 | default: __bug_unaligned_x(ptr); \ |
163 | break; \ | 165 | break; \ |
diff --git a/include/asm-i386/acpi.h b/include/asm-i386/acpi.h index 449f3f272e07..125179adf044 100644 --- a/include/asm-i386/acpi.h +++ b/include/asm-i386/acpi.h | |||
@@ -121,19 +121,6 @@ static inline void acpi_disable_pci(void) | |||
121 | } | 121 | } |
122 | extern int acpi_irq_balance_set(char *str); | 122 | extern int acpi_irq_balance_set(char *str); |
123 | 123 | ||
124 | #else /* !CONFIG_ACPI */ | ||
125 | |||
126 | #define acpi_lapic 0 | ||
127 | #define acpi_ioapic 0 | ||
128 | static inline void acpi_noirq_set(void) { } | ||
129 | static inline void acpi_disable_pci(void) { } | ||
130 | static inline void disable_acpi(void) { } | ||
131 | |||
132 | #endif /* !CONFIG_ACPI */ | ||
133 | |||
134 | |||
135 | #ifdef CONFIG_ACPI_SLEEP | ||
136 | |||
137 | /* routines for saving/restoring kernel state */ | 124 | /* routines for saving/restoring kernel state */ |
138 | extern int acpi_save_state_mem(void); | 125 | extern int acpi_save_state_mem(void); |
139 | extern void acpi_restore_state_mem(void); | 126 | extern void acpi_restore_state_mem(void); |
@@ -143,7 +130,15 @@ extern unsigned long acpi_wakeup_address; | |||
143 | /* early initialization routine */ | 130 | /* early initialization routine */ |
144 | extern void acpi_reserve_bootmem(void); | 131 | extern void acpi_reserve_bootmem(void); |
145 | 132 | ||
146 | #endif /*CONFIG_ACPI_SLEEP*/ | 133 | #else /* !CONFIG_ACPI */ |
134 | |||
135 | #define acpi_lapic 0 | ||
136 | #define acpi_ioapic 0 | ||
137 | static inline void acpi_noirq_set(void) { } | ||
138 | static inline void acpi_disable_pci(void) { } | ||
139 | static inline void disable_acpi(void) { } | ||
140 | |||
141 | #endif /* !CONFIG_ACPI */ | ||
147 | 142 | ||
148 | #define ARCH_HAS_POWER_INIT 1 | 143 | #define ARCH_HAS_POWER_INIT 1 |
149 | 144 | ||
diff --git a/include/asm-i386/bootparam.h b/include/asm-i386/bootparam.h index 427d8652bfde..b91b01783e4b 100644 --- a/include/asm-i386/bootparam.h +++ b/include/asm-i386/bootparam.h | |||
@@ -4,8 +4,9 @@ | |||
4 | #include <linux/types.h> | 4 | #include <linux/types.h> |
5 | #include <linux/screen_info.h> | 5 | #include <linux/screen_info.h> |
6 | #include <linux/apm_bios.h> | 6 | #include <linux/apm_bios.h> |
7 | #include <asm/e820.h> | ||
8 | #include <linux/edd.h> | 7 | #include <linux/edd.h> |
8 | #include <asm/e820.h> | ||
9 | #include <asm/ist.h> | ||
9 | #include <video/edid.h> | 10 | #include <video/edid.h> |
10 | 11 | ||
11 | struct setup_header { | 12 | struct setup_header { |
@@ -48,9 +49,9 @@ struct efi_info { | |||
48 | u32 _pad1; | 49 | u32 _pad1; |
49 | u32 efi_systab; | 50 | u32 efi_systab; |
50 | u32 efi_memdesc_size; | 51 | u32 efi_memdesc_size; |
51 | u32 efi_memdec_version; | 52 | u32 efi_memdesc_version; |
52 | u32 efi_memmap; | 53 | u32 efi_memmap; |
53 | u32 fi_memmap_size; | 54 | u32 efi_memmap_size; |
54 | u32 _pad2[2]; | 55 | u32 _pad2[2]; |
55 | }; | 56 | }; |
56 | 57 | ||
@@ -59,7 +60,7 @@ struct boot_params { | |||
59 | struct screen_info screen_info; /* 0x000 */ | 60 | struct screen_info screen_info; /* 0x000 */ |
60 | struct apm_bios_info apm_bios_info; /* 0x040 */ | 61 | struct apm_bios_info apm_bios_info; /* 0x040 */ |
61 | u8 _pad2[12]; /* 0x054 */ | 62 | u8 _pad2[12]; /* 0x054 */ |
62 | u32 speedstep_info[4]; /* 0x060 */ | 63 | struct ist_info ist_info; /* 0x060 */ |
63 | u8 _pad3[16]; /* 0x070 */ | 64 | u8 _pad3[16]; /* 0x070 */ |
64 | u8 hd0_info[16]; /* obsolete! */ /* 0x080 */ | 65 | u8 hd0_info[16]; /* obsolete! */ /* 0x080 */ |
65 | u8 hd1_info[16]; /* obsolete! */ /* 0x090 */ | 66 | u8 hd1_info[16]; /* obsolete! */ /* 0x090 */ |
diff --git a/include/asm-i386/ist.h b/include/asm-i386/ist.h index d13d1e68afa9..ef2003ebc6f9 100644 --- a/include/asm-i386/ist.h +++ b/include/asm-i386/ist.h | |||
@@ -19,11 +19,13 @@ | |||
19 | 19 | ||
20 | #ifdef __KERNEL__ | 20 | #ifdef __KERNEL__ |
21 | 21 | ||
22 | #include <linux/types.h> | ||
23 | |||
22 | struct ist_info { | 24 | struct ist_info { |
23 | unsigned long signature; | 25 | u32 signature; |
24 | unsigned long command; | 26 | u32 command; |
25 | unsigned long event; | 27 | u32 event; |
26 | unsigned long perf_level; | 28 | u32 perf_level; |
27 | }; | 29 | }; |
28 | 30 | ||
29 | extern struct ist_info ist_info; | 31 | extern struct ist_info ist_info; |
diff --git a/include/asm-i386/suspend.h b/include/asm-i386/suspend.h index 8dbaafe611ff..a2520732ffd6 100644 --- a/include/asm-i386/suspend.h +++ b/include/asm-i386/suspend.h | |||
@@ -21,7 +21,7 @@ struct saved_context { | |||
21 | unsigned long return_address; | 21 | unsigned long return_address; |
22 | } __attribute__((packed)); | 22 | } __attribute__((packed)); |
23 | 23 | ||
24 | #ifdef CONFIG_ACPI_SLEEP | 24 | #ifdef CONFIG_ACPI |
25 | extern unsigned long saved_eip; | 25 | extern unsigned long saved_eip; |
26 | extern unsigned long saved_esp; | 26 | extern unsigned long saved_esp; |
27 | extern unsigned long saved_ebp; | 27 | extern unsigned long saved_ebp; |
diff --git a/include/asm-ia64/acpi.h b/include/asm-ia64/acpi.h index 5b526357d178..49730ffbbae4 100644 --- a/include/asm-ia64/acpi.h +++ b/include/asm-ia64/acpi.h | |||
@@ -100,6 +100,11 @@ const char *acpi_get_sysname (void); | |||
100 | int acpi_request_vector (u32 int_type); | 100 | int acpi_request_vector (u32 int_type); |
101 | int acpi_gsi_to_irq (u32 gsi, unsigned int *irq); | 101 | int acpi_gsi_to_irq (u32 gsi, unsigned int *irq); |
102 | 102 | ||
103 | /* routines for saving/restoring kernel state */ | ||
104 | extern int acpi_save_state_mem(void); | ||
105 | extern void acpi_restore_state_mem(void); | ||
106 | extern unsigned long acpi_wakeup_address; | ||
107 | |||
103 | /* | 108 | /* |
104 | * Record the cpei override flag and current logical cpu. This is | 109 | * Record the cpei override flag and current logical cpu. This is |
105 | * useful for CPU removal. | 110 | * useful for CPU removal. |
diff --git a/include/asm-ia64/ia32.h b/include/asm-ia64/ia32.h index 5ff8d74c3e00..2390ee145aa1 100644 --- a/include/asm-ia64/ia32.h +++ b/include/asm-ia64/ia32.h | |||
@@ -27,11 +27,12 @@ extern int ia32_clone_tls (struct task_struct *child, struct pt_regs *childregs) | |||
27 | extern int ia32_setup_frame1 (int sig, struct k_sigaction *ka, siginfo_t *info, | 27 | extern int ia32_setup_frame1 (int sig, struct k_sigaction *ka, siginfo_t *info, |
28 | sigset_t *set, struct pt_regs *regs); | 28 | sigset_t *set, struct pt_regs *regs); |
29 | #if PAGE_SHIFT > IA32_PAGE_SHIFT | 29 | #if PAGE_SHIFT > IA32_PAGE_SHIFT |
30 | extern int ia32_copy_partial_page_list (struct task_struct *, unsigned long); | 30 | extern int ia32_copy_ia64_partial_page_list(struct task_struct *, |
31 | extern void ia32_drop_partial_page_list (struct task_struct *); | 31 | unsigned long); |
32 | extern void ia32_drop_ia64_partial_page_list(struct task_struct *); | ||
32 | #else | 33 | #else |
33 | # define ia32_copy_partial_page_list(a1, a2) 0 | 34 | # define ia32_copy_ia64_partial_page_list(a1, a2) 0 |
34 | # define ia32_drop_partial_page_list(a1) do { ; } while (0) | 35 | # define ia32_drop_ia64_partial_page_list(a1) do { ; } while (0) |
35 | #endif | 36 | #endif |
36 | 37 | ||
37 | #endif /* !__ASSEMBLY__ */ | 38 | #endif /* !__ASSEMBLY__ */ |
diff --git a/include/asm-ia64/machvec.h b/include/asm-ia64/machvec.h index ca33eb181ff2..5cf8bf1e805e 100644 --- a/include/asm-ia64/machvec.h +++ b/include/asm-ia64/machvec.h | |||
@@ -275,6 +275,7 @@ struct ia64_machine_vector { | |||
275 | 275 | ||
276 | extern struct ia64_machine_vector ia64_mv; | 276 | extern struct ia64_machine_vector ia64_mv; |
277 | extern void machvec_init (const char *name); | 277 | extern void machvec_init (const char *name); |
278 | extern void machvec_init_from_cmdline(const char *cmdline); | ||
278 | 279 | ||
279 | # else | 280 | # else |
280 | # error Unknown configuration. Update asm-ia64/machvec.h. | 281 | # error Unknown configuration. Update asm-ia64/machvec.h. |
diff --git a/include/asm-ia64/processor.h b/include/asm-ia64/processor.h index 6251c76437d2..be3b0ae43270 100644 --- a/include/asm-ia64/processor.h +++ b/include/asm-ia64/processor.h | |||
@@ -220,7 +220,7 @@ struct desc_struct { | |||
220 | 220 | ||
221 | #define TLS_SIZE (GDT_ENTRY_TLS_ENTRIES * 8) | 221 | #define TLS_SIZE (GDT_ENTRY_TLS_ENTRIES * 8) |
222 | 222 | ||
223 | struct partial_page_list; | 223 | struct ia64_partial_page_list; |
224 | #endif | 224 | #endif |
225 | 225 | ||
226 | struct thread_struct { | 226 | struct thread_struct { |
@@ -242,7 +242,7 @@ struct thread_struct { | |||
242 | __u64 fdr; /* IA32 fp except. data reg */ | 242 | __u64 fdr; /* IA32 fp except. data reg */ |
243 | __u64 old_k1; /* old value of ar.k1 */ | 243 | __u64 old_k1; /* old value of ar.k1 */ |
244 | __u64 old_iob; /* old IOBase value */ | 244 | __u64 old_iob; /* old IOBase value */ |
245 | struct partial_page_list *ppl; /* partial page list for 4K page size issue */ | 245 | struct ia64_partial_page_list *ppl; /* partial page list for 4K page size issue */ |
246 | /* cached TLS descriptors. */ | 246 | /* cached TLS descriptors. */ |
247 | struct desc_struct tls_array[GDT_ENTRY_TLS_ENTRIES]; | 247 | struct desc_struct tls_array[GDT_ENTRY_TLS_ENTRIES]; |
248 | 248 | ||
diff --git a/include/asm-ia64/smp.h b/include/asm-ia64/smp.h index c60024989ebd..6314b29e8c4d 100644 --- a/include/asm-ia64/smp.h +++ b/include/asm-ia64/smp.h | |||
@@ -116,7 +116,6 @@ max_xtp (void) | |||
116 | extern int __cpu_disable (void); | 116 | extern int __cpu_disable (void); |
117 | extern void __cpu_die (unsigned int cpu); | 117 | extern void __cpu_die (unsigned int cpu); |
118 | extern void cpu_die (void) __attribute__ ((noreturn)); | 118 | extern void cpu_die (void) __attribute__ ((noreturn)); |
119 | extern int __cpu_up (unsigned int cpu); | ||
120 | extern void __init smp_build_cpu_map(void); | 119 | extern void __init smp_build_cpu_map(void); |
121 | 120 | ||
122 | extern void __init init_smp_config (void); | 121 | extern void __init init_smp_config (void); |
diff --git a/include/asm-m68k/raw_io.h b/include/asm-m68k/raw_io.h index 91c623f0994c..d9eb9834ccc8 100644 --- a/include/asm-m68k/raw_io.h +++ b/include/asm-m68k/raw_io.h | |||
@@ -36,15 +36,15 @@ extern void __iounmap(void *addr, unsigned long size); | |||
36 | #define in_be32(addr) \ | 36 | #define in_be32(addr) \ |
37 | ({ u32 __v = (*(__force volatile u32 *) (addr)); __v; }) | 37 | ({ u32 __v = (*(__force volatile u32 *) (addr)); __v; }) |
38 | #define in_le16(addr) \ | 38 | #define in_le16(addr) \ |
39 | ({ u16 __v = le16_to_cpu(*(__force volatile u16 *) (addr)); __v; }) | 39 | ({ u16 __v = le16_to_cpu(*(__force volatile __le16 *) (addr)); __v; }) |
40 | #define in_le32(addr) \ | 40 | #define in_le32(addr) \ |
41 | ({ u32 __v = le32_to_cpu(*(__force volatile u32 *) (addr)); __v; }) | 41 | ({ u32 __v = le32_to_cpu(*(__force volatile __le32 *) (addr)); __v; }) |
42 | 42 | ||
43 | #define out_8(addr,b) (void)((*(__force volatile u8 *) (addr)) = (b)) | 43 | #define out_8(addr,b) (void)((*(__force volatile u8 *) (addr)) = (b)) |
44 | #define out_be16(addr,w) (void)((*(__force volatile u16 *) (addr)) = (w)) | 44 | #define out_be16(addr,w) (void)((*(__force volatile u16 *) (addr)) = (w)) |
45 | #define out_be32(addr,l) (void)((*(__force volatile u32 *) (addr)) = (l)) | 45 | #define out_be32(addr,l) (void)((*(__force volatile u32 *) (addr)) = (l)) |
46 | #define out_le16(addr,w) (void)((*(__force volatile u16 *) (addr)) = cpu_to_le16(w)) | 46 | #define out_le16(addr,w) (void)((*(__force volatile __le16 *) (addr)) = cpu_to_le16(w)) |
47 | #define out_le32(addr,l) (void)((*(__force volatile u32 *) (addr)) = cpu_to_le32(l)) | 47 | #define out_le32(addr,l) (void)((*(__force volatile __le32 *) (addr)) = cpu_to_le32(l)) |
48 | 48 | ||
49 | #define raw_inb in_8 | 49 | #define raw_inb in_8 |
50 | #define raw_inw in_be16 | 50 | #define raw_inw in_be16 |
diff --git a/include/asm-m68knommu/hardirq.h b/include/asm-m68knommu/hardirq.h index 980075bab792..bfad28149a49 100644 --- a/include/asm-m68knommu/hardirq.h +++ b/include/asm-m68knommu/hardirq.h | |||
@@ -22,4 +22,6 @@ typedef struct { | |||
22 | # error HARDIRQ_BITS is too low! | 22 | # error HARDIRQ_BITS is too low! |
23 | #endif | 23 | #endif |
24 | 24 | ||
25 | void ack_bad_irq(unsigned int irq); | ||
26 | |||
25 | #endif /* __M68K_HARDIRQ_H */ | 27 | #endif /* __M68K_HARDIRQ_H */ |
diff --git a/include/asm-m68knommu/hw_irq.h b/include/asm-m68knommu/hw_irq.h new file mode 100644 index 000000000000..f3ec9e5ae049 --- /dev/null +++ b/include/asm-m68knommu/hw_irq.h | |||
@@ -0,0 +1,4 @@ | |||
1 | #ifndef __M68KNOMMU_HW_IRQ_H__ | ||
2 | #define __M68KNOMMU_HW_IRQ_H__ | ||
3 | |||
4 | #endif /* __M68KNOMMU_HW_IRQ_H__ */ | ||
diff --git a/include/asm-m68knommu/machdep.h b/include/asm-m68knommu/machdep.h index 6ce28f8e0ead..2b75a300df4f 100644 --- a/include/asm-m68knommu/machdep.h +++ b/include/asm-m68knommu/machdep.h | |||
@@ -48,6 +48,5 @@ extern char *mach_sysrq_xlate; | |||
48 | 48 | ||
49 | extern void config_BSP(char *command, int len); | 49 | extern void config_BSP(char *command, int len); |
50 | extern void (*mach_tick)(void); | 50 | extern void (*mach_tick)(void); |
51 | extern void (*mach_trap_init)(void); | ||
52 | 51 | ||
53 | #endif /* _M68KNOMMU_MACHDEP_H */ | 52 | #endif /* _M68KNOMMU_MACHDEP_H */ |
diff --git a/include/asm-m68knommu/mcfdma.h b/include/asm-m68knommu/mcfdma.h index ea729e81a6be..705c52c79cd8 100644 --- a/include/asm-m68knommu/mcfdma.h +++ b/include/asm-m68knommu/mcfdma.h | |||
@@ -133,7 +133,7 @@ | |||
133 | #define MCFDMA_DIR_ASCEN 0x0800 /* Address Sequence Complete (Completion) interrupt enable */ | 133 | #define MCFDMA_DIR_ASCEN 0x0800 /* Address Sequence Complete (Completion) interrupt enable */ |
134 | #define MCFDMA_DIR_TEEN 0x0200 /* Transfer Error interrupt enable */ | 134 | #define MCFDMA_DIR_TEEN 0x0200 /* Transfer Error interrupt enable */ |
135 | #define MCFDMA_DIR_TCEN 0x0100 /* Transfer Complete (a bus transfer, that is) interrupt enable */ | 135 | #define MCFDMA_DIR_TCEN 0x0100 /* Transfer Complete (a bus transfer, that is) interrupt enable */ |
136 | #define MCFDMA_DIR_INV 0x1000 /* Invalid Combination */ | 136 | #define MCFDMA_DIR_INV 0x0010 /* Invalid Combination */ |
137 | #define MCFDMA_DIR_ASC 0x0008 /* Address Sequence Complete (DMA Completion) */ | 137 | #define MCFDMA_DIR_ASC 0x0008 /* Address Sequence Complete (DMA Completion) */ |
138 | #define MCFDMA_DIR_TE 0x0002 /* Transfer Error */ | 138 | #define MCFDMA_DIR_TE 0x0002 /* Transfer Error */ |
139 | #define MCFDMA_DIR_TC 0x0001 /* Transfer Complete */ | 139 | #define MCFDMA_DIR_TC 0x0001 /* Transfer Complete */ |
diff --git a/include/asm-m68knommu/system.h b/include/asm-m68knommu/system.h index 5e5ed18bb78f..5da43a5d12a3 100644 --- a/include/asm-m68knommu/system.h +++ b/include/asm-m68knommu/system.h | |||
@@ -296,7 +296,7 @@ cmpxchg(volatile int *p, int old, int new) | |||
296 | ({ \ | 296 | ({ \ |
297 | unsigned char volatile *reset; \ | 297 | unsigned char volatile *reset; \ |
298 | asm("move.w #0x2700, %sr"); \ | 298 | asm("move.w #0x2700, %sr"); \ |
299 | reset = ((volatile unsigned short *)(MCF_IPSBAR + 0x110000)); \ | 299 | reset = ((volatile unsigned char *)(MCF_IPSBAR + 0x110000)); \ |
300 | while(1) \ | 300 | while(1) \ |
301 | *reset |= (0x01 << 7);\ | 301 | *reset |= (0x01 << 7);\ |
302 | }) | 302 | }) |
@@ -318,7 +318,7 @@ cmpxchg(volatile int *p, int old, int new) | |||
318 | ({ \ | 318 | ({ \ |
319 | unsigned char volatile *reset; \ | 319 | unsigned char volatile *reset; \ |
320 | asm("move.w #0x2700, %sr"); \ | 320 | asm("move.w #0x2700, %sr"); \ |
321 | reset = ((volatile unsigned short *)(MCF_IPSBAR + 0xA0000)); \ | 321 | reset = ((volatile unsigned char *)(MCF_IPSBAR + 0xA0000)); \ |
322 | while(1) \ | 322 | while(1) \ |
323 | *reset |= 0x80; \ | 323 | *reset |= 0x80; \ |
324 | }) | 324 | }) |
diff --git a/include/asm-m68knommu/timex.h b/include/asm-m68knommu/timex.h index 85069998db52..109050f3fe91 100644 --- a/include/asm-m68knommu/timex.h +++ b/include/asm-m68knommu/timex.h | |||
@@ -1 +1,23 @@ | |||
1 | #include <asm-m68k/timex.h> | 1 | /* |
2 | * linux/include/asm-m68knommu/timex.h | ||
3 | * | ||
4 | * m68knommu architecture timex specifications | ||
5 | */ | ||
6 | #ifndef _ASM_M68KNOMMU_TIMEX_H | ||
7 | #define _ASM_M68KNOMMU_TIMEX_H | ||
8 | |||
9 | #ifdef CONFIG_COLDFIRE | ||
10 | #include <asm/coldfire.h> | ||
11 | #define CLOCK_TICK_RATE MCF_CLK | ||
12 | #else | ||
13 | #define CLOCK_TICK_RATE 1193180 /* Underlying HZ */ | ||
14 | #endif | ||
15 | |||
16 | typedef unsigned long cycles_t; | ||
17 | |||
18 | static inline cycles_t get_cycles(void) | ||
19 | { | ||
20 | return 0; | ||
21 | } | ||
22 | |||
23 | #endif | ||
diff --git a/include/asm-mips/edac.h b/include/asm-mips/edac.h new file mode 100644 index 000000000000..83719eee2d13 --- /dev/null +++ b/include/asm-mips/edac.h | |||
@@ -0,0 +1,35 @@ | |||
1 | #ifndef ASM_EDAC_H | ||
2 | #define ASM_EDAC_H | ||
3 | |||
4 | /* ECC atomic, DMA, SMP and interrupt safe scrub function */ | ||
5 | |||
6 | static inline void atomic_scrub(void *va, u32 size) | ||
7 | { | ||
8 | unsigned long *virt_addr = va; | ||
9 | unsigned long temp; | ||
10 | u32 i; | ||
11 | |||
12 | for (i = 0; i < size / sizeof(unsigned long); i++, virt_addr++) { | ||
13 | |||
14 | /* | ||
15 | * Very carefully read and write to memory atomically | ||
16 | * so we are interrupt, DMA and SMP safe. | ||
17 | * | ||
18 | * Intel: asm("lock; addl $0, %0"::"m"(*virt_addr)); | ||
19 | */ | ||
20 | |||
21 | __asm__ __volatile__ ( | ||
22 | " .set mips3 \n" | ||
23 | "1: ll %0, %1 # atomic_add \n" | ||
24 | " ll %0, %1 # atomic_add \n" | ||
25 | " addu %0, $0 \n" | ||
26 | " sc %0, %1 \n" | ||
27 | " beqz %0, 1b \n" | ||
28 | " .set mips0 \n" | ||
29 | : "=&r" (temp), "=m" (*virt_addr) | ||
30 | : "m" (*virt_addr)); | ||
31 | |||
32 | } | ||
33 | } | ||
34 | |||
35 | #endif | ||
diff --git a/include/asm-mips/war.h b/include/asm-mips/war.h index 9de52a5b0f3d..2883ccc69ed0 100644 --- a/include/asm-mips/war.h +++ b/include/asm-mips/war.h | |||
@@ -188,7 +188,6 @@ | |||
188 | #define ICACHE_REFILLS_WORKAROUND_WAR 1 | 188 | #define ICACHE_REFILLS_WORKAROUND_WAR 1 |
189 | #endif | 189 | #endif |
190 | 190 | ||
191 | |||
192 | /* | 191 | /* |
193 | * On the R10000 upto version 2.6 (not sure about 2.7) there is a bug that | 192 | * On the R10000 upto version 2.6 (not sure about 2.7) there is a bug that |
194 | * may cause ll / sc and lld / scd sequences to execute non-atomically. | 193 | * may cause ll / sc and lld / scd sequences to execute non-atomically. |
diff --git a/include/asm-powerpc/bug.h b/include/asm-powerpc/bug.h index f6fa39474846..a248b8bd4d7c 100644 --- a/include/asm-powerpc/bug.h +++ b/include/asm-powerpc/bug.h | |||
@@ -79,7 +79,7 @@ | |||
79 | _EMIT_BUG_ENTRY \ | 79 | _EMIT_BUG_ENTRY \ |
80 | : : "i" (__FILE__), "i" (__LINE__), "i" (0), \ | 80 | : : "i" (__FILE__), "i" (__LINE__), "i" (0), \ |
81 | "i" (sizeof(struct bug_entry)), \ | 81 | "i" (sizeof(struct bug_entry)), \ |
82 | "r" ((long)(x))); \ | 82 | "r" ((__force long)(x))); \ |
83 | } \ | 83 | } \ |
84 | } while (0) | 84 | } while (0) |
85 | 85 | ||
diff --git a/include/asm-powerpc/page.h b/include/asm-powerpc/page.h index 10c51f457d48..236a9210e5fc 100644 --- a/include/asm-powerpc/page.h +++ b/include/asm-powerpc/page.h | |||
@@ -190,7 +190,6 @@ extern void copy_user_page(void *to, void *from, unsigned long vaddr, | |||
190 | extern int page_is_ram(unsigned long pfn); | 190 | extern int page_is_ram(unsigned long pfn); |
191 | 191 | ||
192 | struct vm_area_struct; | 192 | struct vm_area_struct; |
193 | extern const char *arch_vma_name(struct vm_area_struct *vma); | ||
194 | 193 | ||
195 | #include <asm-generic/memory_model.h> | 194 | #include <asm-generic/memory_model.h> |
196 | #endif /* __ASSEMBLY__ */ | 195 | #endif /* __ASSEMBLY__ */ |
diff --git a/include/asm-powerpc/pci-bridge.h b/include/asm-powerpc/pci-bridge.h index e72c2a60853c..e909769b6410 100644 --- a/include/asm-powerpc/pci-bridge.h +++ b/include/asm-powerpc/pci-bridge.h | |||
@@ -45,10 +45,17 @@ struct pci_controller { | |||
45 | * on Freescale PCI-e controllers since they used the PCI_PRIMARY_BUS | 45 | * on Freescale PCI-e controllers since they used the PCI_PRIMARY_BUS |
46 | * to determine which bus number to match on when generating type0 | 46 | * to determine which bus number to match on when generating type0 |
47 | * config cycles | 47 | * config cycles |
48 | * NO_PCIE_LINK - the Freescale PCI-e controllers have issues with | ||
49 | * hanging if we don't have link and try to do config cycles to | ||
50 | * anything but the PHB. Only allow talking to the PHB if this is | ||
51 | * set. | ||
52 | * BIG_ENDIAN - cfg_addr is a big endian register | ||
48 | */ | 53 | */ |
49 | #define PPC_INDIRECT_TYPE_SET_CFG_TYPE (0x00000001) | 54 | #define PPC_INDIRECT_TYPE_SET_CFG_TYPE (0x00000001) |
50 | #define PPC_INDIRECT_TYPE_EXT_REG (0x00000002) | 55 | #define PPC_INDIRECT_TYPE_EXT_REG (0x00000002) |
51 | #define PPC_INDIRECT_TYPE_SURPRESS_PRIMARY_BUS (0x00000004) | 56 | #define PPC_INDIRECT_TYPE_SURPRESS_PRIMARY_BUS (0x00000004) |
57 | #define PPC_INDIRECT_TYPE_NO_PCIE_LINK (0x00000008) | ||
58 | #define PPC_INDIRECT_TYPE_BIG_ENDIAN (0x00000010) | ||
52 | u32 indirect_type; | 59 | u32 indirect_type; |
53 | 60 | ||
54 | /* Currently, we limit ourselves to 1 IO range and 3 mem | 61 | /* Currently, we limit ourselves to 1 IO range and 3 mem |
@@ -64,6 +71,14 @@ static inline struct pci_controller *pci_bus_to_host(struct pci_bus *bus) | |||
64 | return bus->sysdata; | 71 | return bus->sysdata; |
65 | } | 72 | } |
66 | 73 | ||
74 | static inline int isa_vaddr_is_ioport(void __iomem *address) | ||
75 | { | ||
76 | /* No specific ISA handling on ppc32 at this stage, it | ||
77 | * all goes through PCI | ||
78 | */ | ||
79 | return 0; | ||
80 | } | ||
81 | |||
67 | /* These are used for config access before all the PCI probing | 82 | /* These are used for config access before all the PCI probing |
68 | has been done. */ | 83 | has been done. */ |
69 | int early_read_config_byte(struct pci_controller *hose, int bus, int dev_fn, | 84 | int early_read_config_byte(struct pci_controller *hose, int bus, int dev_fn, |
@@ -79,11 +94,14 @@ int early_write_config_word(struct pci_controller *hose, int bus, int dev_fn, | |||
79 | int early_write_config_dword(struct pci_controller *hose, int bus, int dev_fn, | 94 | int early_write_config_dword(struct pci_controller *hose, int bus, int dev_fn, |
80 | int where, u32 val); | 95 | int where, u32 val); |
81 | 96 | ||
82 | extern void setup_indirect_pci_nomap(struct pci_controller* hose, | 97 | extern int early_find_capability(struct pci_controller *hose, int bus, |
83 | void __iomem *cfg_addr, void __iomem *cfg_data); | 98 | int dev_fn, int cap); |
99 | |||
84 | extern void setup_indirect_pci(struct pci_controller* hose, | 100 | extern void setup_indirect_pci(struct pci_controller* hose, |
85 | u32 cfg_addr, u32 cfg_data); | 101 | u32 cfg_addr, u32 cfg_data, u32 flags); |
86 | extern void setup_grackle(struct pci_controller *hose); | 102 | extern void setup_grackle(struct pci_controller *hose); |
103 | extern void __init update_bridge_resource(struct pci_dev *dev, | ||
104 | struct resource *res); | ||
87 | 105 | ||
88 | #else | 106 | #else |
89 | 107 | ||
@@ -231,6 +249,13 @@ extern void pcibios_free_controller(struct pci_controller *phb); | |||
231 | 249 | ||
232 | extern void isa_bridge_find_early(struct pci_controller *hose); | 250 | extern void isa_bridge_find_early(struct pci_controller *hose); |
233 | 251 | ||
252 | static inline int isa_vaddr_is_ioport(void __iomem *address) | ||
253 | { | ||
254 | /* Check if address hits the reserved legacy IO range */ | ||
255 | unsigned long ea = (unsigned long)address; | ||
256 | return ea >= ISA_IO_BASE && ea < ISA_IO_END; | ||
257 | } | ||
258 | |||
234 | extern int pcibios_unmap_io_space(struct pci_bus *bus); | 259 | extern int pcibios_unmap_io_space(struct pci_bus *bus); |
235 | extern int pcibios_map_io_space(struct pci_bus *bus); | 260 | extern int pcibios_map_io_space(struct pci_bus *bus); |
236 | 261 | ||
@@ -261,11 +286,16 @@ extern struct pci_controller * | |||
261 | pcibios_alloc_controller(struct device_node *dev); | 286 | pcibios_alloc_controller(struct device_node *dev); |
262 | #ifdef CONFIG_PCI | 287 | #ifdef CONFIG_PCI |
263 | extern unsigned long pci_address_to_pio(phys_addr_t address); | 288 | extern unsigned long pci_address_to_pio(phys_addr_t address); |
289 | extern int pcibios_vaddr_is_ioport(void __iomem *address); | ||
264 | #else | 290 | #else |
265 | static inline unsigned long pci_address_to_pio(phys_addr_t address) | 291 | static inline unsigned long pci_address_to_pio(phys_addr_t address) |
266 | { | 292 | { |
267 | return (unsigned long)-1; | 293 | return (unsigned long)-1; |
268 | } | 294 | } |
295 | static inline int pcibios_vaddr_is_ioport(void __iomem *address) | ||
296 | { | ||
297 | return 0; | ||
298 | } | ||
269 | #endif | 299 | #endif |
270 | 300 | ||
271 | 301 | ||
diff --git a/include/asm-powerpc/ppc_asm.h b/include/asm-powerpc/ppc_asm.h index fa083d8e4663..65325721446d 100644 --- a/include/asm-powerpc/ppc_asm.h +++ b/include/asm-powerpc/ppc_asm.h | |||
@@ -181,6 +181,18 @@ name: \ | |||
181 | .type GLUE(.,name),@function; \ | 181 | .type GLUE(.,name),@function; \ |
182 | GLUE(.,name): | 182 | GLUE(.,name): |
183 | 183 | ||
184 | #define _INIT_STATIC(name) \ | ||
185 | .section ".text.init.refok"; \ | ||
186 | .align 2 ; \ | ||
187 | .section ".opd","aw"; \ | ||
188 | name: \ | ||
189 | .quad GLUE(.,name); \ | ||
190 | .quad .TOC.@tocbase; \ | ||
191 | .quad 0; \ | ||
192 | .previous; \ | ||
193 | .type GLUE(.,name),@function; \ | ||
194 | GLUE(.,name): | ||
195 | |||
184 | #else /* 32-bit */ | 196 | #else /* 32-bit */ |
185 | 197 | ||
186 | #define _GLOBAL(n) \ | 198 | #define _GLOBAL(n) \ |
diff --git a/include/asm-powerpc/vio.h b/include/asm-powerpc/vio.h index 0117b544ecbc..3a0975e2adad 100644 --- a/include/asm-powerpc/vio.h +++ b/include/asm-powerpc/vio.h | |||
@@ -80,6 +80,11 @@ extern const void *vio_get_attribute(struct vio_dev *vdev, char *which, | |||
80 | extern struct vio_dev *vio_find_node(struct device_node *vnode); | 80 | extern struct vio_dev *vio_find_node(struct device_node *vnode); |
81 | extern int vio_enable_interrupts(struct vio_dev *dev); | 81 | extern int vio_enable_interrupts(struct vio_dev *dev); |
82 | extern int vio_disable_interrupts(struct vio_dev *dev); | 82 | extern int vio_disable_interrupts(struct vio_dev *dev); |
83 | #else | ||
84 | static inline int vio_enable_interrupts(struct vio_dev *dev) | ||
85 | { | ||
86 | return 0; | ||
87 | } | ||
83 | #endif | 88 | #endif |
84 | 89 | ||
85 | static inline struct vio_driver *to_vio_driver(struct device_driver *drv) | 90 | static inline struct vio_driver *to_vio_driver(struct device_driver *drv) |
diff --git a/include/asm-x86_64/acpi.h b/include/asm-x86_64/acpi.h index 1da8f49c0fe2..98173357dd89 100644 --- a/include/asm-x86_64/acpi.h +++ b/include/asm-x86_64/acpi.h | |||
@@ -108,6 +108,15 @@ static inline void acpi_disable_pci(void) | |||
108 | } | 108 | } |
109 | extern int acpi_irq_balance_set(char *str); | 109 | extern int acpi_irq_balance_set(char *str); |
110 | 110 | ||
111 | /* routines for saving/restoring kernel state */ | ||
112 | extern int acpi_save_state_mem(void); | ||
113 | extern void acpi_restore_state_mem(void); | ||
114 | |||
115 | extern unsigned long acpi_wakeup_address; | ||
116 | |||
117 | /* early initialization routine */ | ||
118 | extern void acpi_reserve_bootmem(void); | ||
119 | |||
111 | #else /* !CONFIG_ACPI */ | 120 | #else /* !CONFIG_ACPI */ |
112 | 121 | ||
113 | #define acpi_lapic 0 | 122 | #define acpi_lapic 0 |
@@ -121,19 +130,6 @@ extern int acpi_numa; | |||
121 | extern int acpi_scan_nodes(unsigned long start, unsigned long end); | 130 | extern int acpi_scan_nodes(unsigned long start, unsigned long end); |
122 | #define NR_NODE_MEMBLKS (MAX_NUMNODES*2) | 131 | #define NR_NODE_MEMBLKS (MAX_NUMNODES*2) |
123 | 132 | ||
124 | #ifdef CONFIG_ACPI_SLEEP | ||
125 | |||
126 | /* routines for saving/restoring kernel state */ | ||
127 | extern int acpi_save_state_mem(void); | ||
128 | extern void acpi_restore_state_mem(void); | ||
129 | |||
130 | extern unsigned long acpi_wakeup_address; | ||
131 | |||
132 | /* early initialization routine */ | ||
133 | extern void acpi_reserve_bootmem(void); | ||
134 | |||
135 | #endif /*CONFIG_ACPI_SLEEP*/ | ||
136 | |||
137 | extern int acpi_disabled; | 133 | extern int acpi_disabled; |
138 | extern int acpi_pci_disabled; | 134 | extern int acpi_pci_disabled; |
139 | 135 | ||
diff --git a/include/asm-x86_64/ist.h b/include/asm-x86_64/ist.h new file mode 100644 index 000000000000..338857ecbc68 --- /dev/null +++ b/include/asm-x86_64/ist.h | |||
@@ -0,0 +1 @@ | |||
#include <asm-i386/ist.h> | |||
diff --git a/include/asm-x86_64/suspend.h b/include/asm-x86_64/suspend.h index 9c3f8de90d2d..b897e8cb55fb 100644 --- a/include/asm-x86_64/suspend.h +++ b/include/asm-x86_64/suspend.h | |||
@@ -44,7 +44,6 @@ extern unsigned long saved_context_eflags; | |||
44 | 44 | ||
45 | extern void fix_processor_context(void); | 45 | extern void fix_processor_context(void); |
46 | 46 | ||
47 | #ifdef CONFIG_ACPI_SLEEP | ||
48 | extern unsigned long saved_rip; | 47 | extern unsigned long saved_rip; |
49 | extern unsigned long saved_rsp; | 48 | extern unsigned long saved_rsp; |
50 | extern unsigned long saved_rbp; | 49 | extern unsigned long saved_rbp; |
@@ -54,4 +53,3 @@ extern unsigned long saved_rdi; | |||
54 | 53 | ||
55 | /* routines for saving/restoring kernel state */ | 54 | /* routines for saving/restoring kernel state */ |
56 | extern int acpi_save_state_mem(void); | 55 | extern int acpi_save_state_mem(void); |
57 | #endif | ||
diff --git a/include/asm-x86_64/uaccess.h b/include/asm-x86_64/uaccess.h index 9df30b939c4e..f4ce8768ad44 100644 --- a/include/asm-x86_64/uaccess.h +++ b/include/asm-x86_64/uaccess.h | |||
@@ -100,7 +100,7 @@ struct exception_table_entry | |||
100 | case 8: __get_user_x(8,__ret_gu,__val_gu,ptr); break; \ | 100 | case 8: __get_user_x(8,__ret_gu,__val_gu,ptr); break; \ |
101 | default: __get_user_bad(); break; \ | 101 | default: __get_user_bad(); break; \ |
102 | } \ | 102 | } \ |
103 | (x) = (typeof(*(ptr)))__val_gu; \ | 103 | (x) = (__force typeof(*(ptr)))__val_gu; \ |
104 | __ret_gu; \ | 104 | __ret_gu; \ |
105 | }) | 105 | }) |
106 | 106 | ||
@@ -192,7 +192,7 @@ struct __large_struct { unsigned long buf[100]; }; | |||
192 | int __gu_err; \ | 192 | int __gu_err; \ |
193 | unsigned long __gu_val; \ | 193 | unsigned long __gu_val; \ |
194 | __get_user_size(__gu_val,(ptr),(size),__gu_err); \ | 194 | __get_user_size(__gu_val,(ptr),(size),__gu_err); \ |
195 | (x) = (typeof(*(ptr)))__gu_val; \ | 195 | (x) = (__force typeof(*(ptr)))__gu_val; \ |
196 | __gu_err; \ | 196 | __gu_err; \ |
197 | }) | 197 | }) |
198 | 198 | ||
diff --git a/include/linux/acpi.h b/include/linux/acpi.h index d5680cd7746a..bf5e0009de75 100644 --- a/include/linux/acpi.h +++ b/include/linux/acpi.h | |||
@@ -33,6 +33,7 @@ | |||
33 | #endif | 33 | #endif |
34 | 34 | ||
35 | #include <linux/list.h> | 35 | #include <linux/list.h> |
36 | #include <linux/mod_devicetable.h> | ||
36 | 37 | ||
37 | #include <acpi/acpi.h> | 38 | #include <acpi/acpi.h> |
38 | #include <acpi/acpi_bus.h> | 39 | #include <acpi/acpi_bus.h> |
diff --git a/include/linux/apm_bios.h b/include/linux/apm_bios.h index 290aef326812..5f921c84827a 100644 --- a/include/linux/apm_bios.h +++ b/include/linux/apm_bios.h | |||
@@ -21,20 +21,22 @@ typedef unsigned short apm_eventinfo_t; | |||
21 | 21 | ||
22 | #ifdef __KERNEL__ | 22 | #ifdef __KERNEL__ |
23 | 23 | ||
24 | #include <linux/types.h> | ||
25 | |||
24 | #define APM_CS (GDT_ENTRY_APMBIOS_BASE * 8) | 26 | #define APM_CS (GDT_ENTRY_APMBIOS_BASE * 8) |
25 | #define APM_CS_16 (APM_CS + 8) | 27 | #define APM_CS_16 (APM_CS + 8) |
26 | #define APM_DS (APM_CS_16 + 8) | 28 | #define APM_DS (APM_CS_16 + 8) |
27 | 29 | ||
28 | struct apm_bios_info { | 30 | struct apm_bios_info { |
29 | unsigned short version; | 31 | u16 version; |
30 | unsigned short cseg; | 32 | u16 cseg; |
31 | unsigned long offset; | 33 | u32 offset; |
32 | unsigned short cseg_16; | 34 | u16 cseg_16; |
33 | unsigned short dseg; | 35 | u16 dseg; |
34 | unsigned short flags; | 36 | u16 flags; |
35 | unsigned short cseg_len; | 37 | u16 cseg_len; |
36 | unsigned short cseg_16_len; | 38 | u16 cseg_16_len; |
37 | unsigned short dseg_len; | 39 | u16 dseg_len; |
38 | }; | 40 | }; |
39 | 41 | ||
40 | /* Results of APM Installation Check */ | 42 | /* Results of APM Installation Check */ |
diff --git a/include/linux/blkdev.h b/include/linux/blkdev.h index 695e34964cb7..a1c96d9ee720 100644 --- a/include/linux/blkdev.h +++ b/include/linux/blkdev.h | |||
@@ -37,7 +37,7 @@ | |||
37 | struct scsi_ioctl_command; | 37 | struct scsi_ioctl_command; |
38 | 38 | ||
39 | struct request_queue; | 39 | struct request_queue; |
40 | typedef struct request_queue request_queue_t; | 40 | typedef struct request_queue request_queue_t __deprecated; |
41 | struct elevator_queue; | 41 | struct elevator_queue; |
42 | typedef struct elevator_queue elevator_t; | 42 | typedef struct elevator_queue elevator_t; |
43 | struct request_pm_state; | 43 | struct request_pm_state; |
@@ -233,7 +233,7 @@ struct request { | |||
233 | struct list_head queuelist; | 233 | struct list_head queuelist; |
234 | struct list_head donelist; | 234 | struct list_head donelist; |
235 | 235 | ||
236 | request_queue_t *q; | 236 | struct request_queue *q; |
237 | 237 | ||
238 | unsigned int cmd_flags; | 238 | unsigned int cmd_flags; |
239 | enum rq_cmd_type_bits cmd_type; | 239 | enum rq_cmd_type_bits cmd_type; |
@@ -337,15 +337,15 @@ struct request_pm_state | |||
337 | 337 | ||
338 | #include <linux/elevator.h> | 338 | #include <linux/elevator.h> |
339 | 339 | ||
340 | typedef void (request_fn_proc) (request_queue_t *q); | 340 | typedef void (request_fn_proc) (struct request_queue *q); |
341 | typedef int (make_request_fn) (request_queue_t *q, struct bio *bio); | 341 | typedef int (make_request_fn) (struct request_queue *q, struct bio *bio); |
342 | typedef int (prep_rq_fn) (request_queue_t *, struct request *); | 342 | typedef int (prep_rq_fn) (struct request_queue *, struct request *); |
343 | typedef void (unplug_fn) (request_queue_t *); | 343 | typedef void (unplug_fn) (struct request_queue *); |
344 | 344 | ||
345 | struct bio_vec; | 345 | struct bio_vec; |
346 | typedef int (merge_bvec_fn) (request_queue_t *, struct bio *, struct bio_vec *); | 346 | typedef int (merge_bvec_fn) (struct request_queue *, struct bio *, struct bio_vec *); |
347 | typedef int (issue_flush_fn) (request_queue_t *, struct gendisk *, sector_t *); | 347 | typedef int (issue_flush_fn) (struct request_queue *, struct gendisk *, sector_t *); |
348 | typedef void (prepare_flush_fn) (request_queue_t *, struct request *); | 348 | typedef void (prepare_flush_fn) (struct request_queue *, struct request *); |
349 | typedef void (softirq_done_fn)(struct request *); | 349 | typedef void (softirq_done_fn)(struct request *); |
350 | 350 | ||
351 | enum blk_queue_state { | 351 | enum blk_queue_state { |
@@ -626,13 +626,13 @@ extern unsigned long blk_max_low_pfn, blk_max_pfn; | |||
626 | 626 | ||
627 | #ifdef CONFIG_BOUNCE | 627 | #ifdef CONFIG_BOUNCE |
628 | extern int init_emergency_isa_pool(void); | 628 | extern int init_emergency_isa_pool(void); |
629 | extern void blk_queue_bounce(request_queue_t *q, struct bio **bio); | 629 | extern void blk_queue_bounce(struct request_queue *q, struct bio **bio); |
630 | #else | 630 | #else |
631 | static inline int init_emergency_isa_pool(void) | 631 | static inline int init_emergency_isa_pool(void) |
632 | { | 632 | { |
633 | return 0; | 633 | return 0; |
634 | } | 634 | } |
635 | static inline void blk_queue_bounce(request_queue_t *q, struct bio **bio) | 635 | static inline void blk_queue_bounce(struct request_queue *q, struct bio **bio) |
636 | { | 636 | { |
637 | } | 637 | } |
638 | #endif /* CONFIG_MMU */ | 638 | #endif /* CONFIG_MMU */ |
@@ -646,14 +646,14 @@ extern void blk_unregister_queue(struct gendisk *disk); | |||
646 | extern void register_disk(struct gendisk *dev); | 646 | extern void register_disk(struct gendisk *dev); |
647 | extern void generic_make_request(struct bio *bio); | 647 | extern void generic_make_request(struct bio *bio); |
648 | extern void blk_put_request(struct request *); | 648 | extern void blk_put_request(struct request *); |
649 | extern void __blk_put_request(request_queue_t *, struct request *); | 649 | extern void __blk_put_request(struct request_queue *, struct request *); |
650 | extern void blk_end_sync_rq(struct request *rq, int error); | 650 | extern void blk_end_sync_rq(struct request *rq, int error); |
651 | extern struct request *blk_get_request(request_queue_t *, int, gfp_t); | 651 | extern struct request *blk_get_request(struct request_queue *, int, gfp_t); |
652 | extern void blk_insert_request(request_queue_t *, struct request *, int, void *); | 652 | extern void blk_insert_request(struct request_queue *, struct request *, int, void *); |
653 | extern void blk_requeue_request(request_queue_t *, struct request *); | 653 | extern void blk_requeue_request(struct request_queue *, struct request *); |
654 | extern void blk_plug_device(request_queue_t *); | 654 | extern void blk_plug_device(struct request_queue *); |
655 | extern int blk_remove_plug(request_queue_t *); | 655 | extern int blk_remove_plug(struct request_queue *); |
656 | extern void blk_recount_segments(request_queue_t *, struct bio *); | 656 | extern void blk_recount_segments(struct request_queue *, struct bio *); |
657 | extern int scsi_cmd_ioctl(struct file *, struct request_queue *, | 657 | extern int scsi_cmd_ioctl(struct file *, struct request_queue *, |
658 | struct gendisk *, unsigned int, void __user *); | 658 | struct gendisk *, unsigned int, void __user *); |
659 | extern int sg_scsi_ioctl(struct file *, struct request_queue *, | 659 | extern int sg_scsi_ioctl(struct file *, struct request_queue *, |
@@ -662,14 +662,15 @@ extern int sg_scsi_ioctl(struct file *, struct request_queue *, | |||
662 | /* | 662 | /* |
663 | * Temporary export, until SCSI gets fixed up. | 663 | * Temporary export, until SCSI gets fixed up. |
664 | */ | 664 | */ |
665 | extern int ll_back_merge_fn(request_queue_t *, struct request *, struct bio *); | 665 | extern int ll_back_merge_fn(struct request_queue *, struct request *, |
666 | struct bio *); | ||
666 | 667 | ||
667 | /* | 668 | /* |
668 | * A queue has just exitted congestion. Note this in the global counter of | 669 | * A queue has just exitted congestion. Note this in the global counter of |
669 | * congested queues, and wake up anyone who was waiting for requests to be | 670 | * congested queues, and wake up anyone who was waiting for requests to be |
670 | * put back. | 671 | * put back. |
671 | */ | 672 | */ |
672 | static inline void blk_clear_queue_congested(request_queue_t *q, int rw) | 673 | static inline void blk_clear_queue_congested(struct request_queue *q, int rw) |
673 | { | 674 | { |
674 | clear_bdi_congested(&q->backing_dev_info, rw); | 675 | clear_bdi_congested(&q->backing_dev_info, rw); |
675 | } | 676 | } |
@@ -678,29 +679,29 @@ static inline void blk_clear_queue_congested(request_queue_t *q, int rw) | |||
678 | * A queue has just entered congestion. Flag that in the queue's VM-visible | 679 | * A queue has just entered congestion. Flag that in the queue's VM-visible |
679 | * state flags and increment the global gounter of congested queues. | 680 | * state flags and increment the global gounter of congested queues. |
680 | */ | 681 | */ |
681 | static inline void blk_set_queue_congested(request_queue_t *q, int rw) | 682 | static inline void blk_set_queue_congested(struct request_queue *q, int rw) |
682 | { | 683 | { |
683 | set_bdi_congested(&q->backing_dev_info, rw); | 684 | set_bdi_congested(&q->backing_dev_info, rw); |
684 | } | 685 | } |
685 | 686 | ||
686 | extern void blk_start_queue(request_queue_t *q); | 687 | extern void blk_start_queue(struct request_queue *q); |
687 | extern void blk_stop_queue(request_queue_t *q); | 688 | extern void blk_stop_queue(struct request_queue *q); |
688 | extern void blk_sync_queue(struct request_queue *q); | 689 | extern void blk_sync_queue(struct request_queue *q); |
689 | extern void __blk_stop_queue(request_queue_t *q); | 690 | extern void __blk_stop_queue(struct request_queue *q); |
690 | extern void blk_run_queue(request_queue_t *); | 691 | extern void blk_run_queue(struct request_queue *); |
691 | extern void blk_start_queueing(request_queue_t *); | 692 | extern void blk_start_queueing(struct request_queue *); |
692 | extern int blk_rq_map_user(request_queue_t *, struct request *, void __user *, unsigned long); | 693 | extern int blk_rq_map_user(struct request_queue *, struct request *, void __user *, unsigned long); |
693 | extern int blk_rq_unmap_user(struct bio *); | 694 | extern int blk_rq_unmap_user(struct bio *); |
694 | extern int blk_rq_map_kern(request_queue_t *, struct request *, void *, unsigned int, gfp_t); | 695 | extern int blk_rq_map_kern(struct request_queue *, struct request *, void *, unsigned int, gfp_t); |
695 | extern int blk_rq_map_user_iov(request_queue_t *, struct request *, | 696 | extern int blk_rq_map_user_iov(struct request_queue *, struct request *, |
696 | struct sg_iovec *, int, unsigned int); | 697 | struct sg_iovec *, int, unsigned int); |
697 | extern int blk_execute_rq(request_queue_t *, struct gendisk *, | 698 | extern int blk_execute_rq(struct request_queue *, struct gendisk *, |
698 | struct request *, int); | 699 | struct request *, int); |
699 | extern void blk_execute_rq_nowait(request_queue_t *, struct gendisk *, | 700 | extern void blk_execute_rq_nowait(struct request_queue *, struct gendisk *, |
700 | struct request *, int, rq_end_io_fn *); | 701 | struct request *, int, rq_end_io_fn *); |
701 | extern int blk_verify_command(unsigned char *, int); | 702 | extern int blk_verify_command(unsigned char *, int); |
702 | 703 | ||
703 | static inline request_queue_t *bdev_get_queue(struct block_device *bdev) | 704 | static inline struct request_queue *bdev_get_queue(struct block_device *bdev) |
704 | { | 705 | { |
705 | return bdev->bd_disk->queue; | 706 | return bdev->bd_disk->queue; |
706 | } | 707 | } |
@@ -749,41 +750,41 @@ static inline void blkdev_dequeue_request(struct request *req) | |||
749 | /* | 750 | /* |
750 | * Access functions for manipulating queue properties | 751 | * Access functions for manipulating queue properties |
751 | */ | 752 | */ |
752 | extern request_queue_t *blk_init_queue_node(request_fn_proc *rfn, | 753 | extern struct request_queue *blk_init_queue_node(request_fn_proc *rfn, |
753 | spinlock_t *lock, int node_id); | 754 | spinlock_t *lock, int node_id); |
754 | extern request_queue_t *blk_init_queue(request_fn_proc *, spinlock_t *); | 755 | extern struct request_queue *blk_init_queue(request_fn_proc *, spinlock_t *); |
755 | extern void blk_cleanup_queue(request_queue_t *); | 756 | extern void blk_cleanup_queue(struct request_queue *); |
756 | extern void blk_queue_make_request(request_queue_t *, make_request_fn *); | 757 | extern void blk_queue_make_request(struct request_queue *, make_request_fn *); |
757 | extern void blk_queue_bounce_limit(request_queue_t *, u64); | 758 | extern void blk_queue_bounce_limit(struct request_queue *, u64); |
758 | extern void blk_queue_max_sectors(request_queue_t *, unsigned int); | 759 | extern void blk_queue_max_sectors(struct request_queue *, unsigned int); |
759 | extern void blk_queue_max_phys_segments(request_queue_t *, unsigned short); | 760 | extern void blk_queue_max_phys_segments(struct request_queue *, unsigned short); |
760 | extern void blk_queue_max_hw_segments(request_queue_t *, unsigned short); | 761 | extern void blk_queue_max_hw_segments(struct request_queue *, unsigned short); |
761 | extern void blk_queue_max_segment_size(request_queue_t *, unsigned int); | 762 | extern void blk_queue_max_segment_size(struct request_queue *, unsigned int); |
762 | extern void blk_queue_hardsect_size(request_queue_t *, unsigned short); | 763 | extern void blk_queue_hardsect_size(struct request_queue *, unsigned short); |
763 | extern void blk_queue_stack_limits(request_queue_t *t, request_queue_t *b); | 764 | extern void blk_queue_stack_limits(struct request_queue *t, struct request_queue *b); |
764 | extern void blk_queue_segment_boundary(request_queue_t *, unsigned long); | 765 | extern void blk_queue_segment_boundary(struct request_queue *, unsigned long); |
765 | extern void blk_queue_prep_rq(request_queue_t *, prep_rq_fn *pfn); | 766 | extern void blk_queue_prep_rq(struct request_queue *, prep_rq_fn *pfn); |
766 | extern void blk_queue_merge_bvec(request_queue_t *, merge_bvec_fn *); | 767 | extern void blk_queue_merge_bvec(struct request_queue *, merge_bvec_fn *); |
767 | extern void blk_queue_dma_alignment(request_queue_t *, int); | 768 | extern void blk_queue_dma_alignment(struct request_queue *, int); |
768 | extern void blk_queue_softirq_done(request_queue_t *, softirq_done_fn *); | 769 | extern void blk_queue_softirq_done(struct request_queue *, softirq_done_fn *); |
769 | extern struct backing_dev_info *blk_get_backing_dev_info(struct block_device *bdev); | 770 | extern struct backing_dev_info *blk_get_backing_dev_info(struct block_device *bdev); |
770 | extern int blk_queue_ordered(request_queue_t *, unsigned, prepare_flush_fn *); | 771 | extern int blk_queue_ordered(struct request_queue *, unsigned, prepare_flush_fn *); |
771 | extern void blk_queue_issue_flush_fn(request_queue_t *, issue_flush_fn *); | 772 | extern void blk_queue_issue_flush_fn(struct request_queue *, issue_flush_fn *); |
772 | extern int blk_do_ordered(request_queue_t *, struct request **); | 773 | extern int blk_do_ordered(struct request_queue *, struct request **); |
773 | extern unsigned blk_ordered_cur_seq(request_queue_t *); | 774 | extern unsigned blk_ordered_cur_seq(struct request_queue *); |
774 | extern unsigned blk_ordered_req_seq(struct request *); | 775 | extern unsigned blk_ordered_req_seq(struct request *); |
775 | extern void blk_ordered_complete_seq(request_queue_t *, unsigned, int); | 776 | extern void blk_ordered_complete_seq(struct request_queue *, unsigned, int); |
776 | 777 | ||
777 | extern int blk_rq_map_sg(request_queue_t *, struct request *, struct scatterlist *); | 778 | extern int blk_rq_map_sg(struct request_queue *, struct request *, struct scatterlist *); |
778 | extern void blk_dump_rq_flags(struct request *, char *); | 779 | extern void blk_dump_rq_flags(struct request *, char *); |
779 | extern void generic_unplug_device(request_queue_t *); | 780 | extern void generic_unplug_device(struct request_queue *); |
780 | extern void __generic_unplug_device(request_queue_t *); | 781 | extern void __generic_unplug_device(struct request_queue *); |
781 | extern long nr_blockdev_pages(void); | 782 | extern long nr_blockdev_pages(void); |
782 | 783 | ||
783 | int blk_get_queue(request_queue_t *); | 784 | int blk_get_queue(struct request_queue *); |
784 | request_queue_t *blk_alloc_queue(gfp_t); | 785 | struct request_queue *blk_alloc_queue(gfp_t); |
785 | request_queue_t *blk_alloc_queue_node(gfp_t, int); | 786 | struct request_queue *blk_alloc_queue_node(gfp_t, int); |
786 | extern void blk_put_queue(request_queue_t *); | 787 | extern void blk_put_queue(struct request_queue *); |
787 | 788 | ||
788 | /* | 789 | /* |
789 | * tag stuff | 790 | * tag stuff |
@@ -791,13 +792,13 @@ extern void blk_put_queue(request_queue_t *); | |||
791 | #define blk_queue_tag_depth(q) ((q)->queue_tags->busy) | 792 | #define blk_queue_tag_depth(q) ((q)->queue_tags->busy) |
792 | #define blk_queue_tag_queue(q) ((q)->queue_tags->busy < (q)->queue_tags->max_depth) | 793 | #define blk_queue_tag_queue(q) ((q)->queue_tags->busy < (q)->queue_tags->max_depth) |
793 | #define blk_rq_tagged(rq) ((rq)->cmd_flags & REQ_QUEUED) | 794 | #define blk_rq_tagged(rq) ((rq)->cmd_flags & REQ_QUEUED) |
794 | extern int blk_queue_start_tag(request_queue_t *, struct request *); | 795 | extern int blk_queue_start_tag(struct request_queue *, struct request *); |
795 | extern struct request *blk_queue_find_tag(request_queue_t *, int); | 796 | extern struct request *blk_queue_find_tag(struct request_queue *, int); |
796 | extern void blk_queue_end_tag(request_queue_t *, struct request *); | 797 | extern void blk_queue_end_tag(struct request_queue *, struct request *); |
797 | extern int blk_queue_init_tags(request_queue_t *, int, struct blk_queue_tag *); | 798 | extern int blk_queue_init_tags(struct request_queue *, int, struct blk_queue_tag *); |
798 | extern void blk_queue_free_tags(request_queue_t *); | 799 | extern void blk_queue_free_tags(struct request_queue *); |
799 | extern int blk_queue_resize_tags(request_queue_t *, int); | 800 | extern int blk_queue_resize_tags(struct request_queue *, int); |
800 | extern void blk_queue_invalidate_tags(request_queue_t *); | 801 | extern void blk_queue_invalidate_tags(struct request_queue *); |
801 | extern struct blk_queue_tag *blk_init_tags(int); | 802 | extern struct blk_queue_tag *blk_init_tags(int); |
802 | extern void blk_free_tags(struct blk_queue_tag *); | 803 | extern void blk_free_tags(struct blk_queue_tag *); |
803 | 804 | ||
@@ -809,7 +810,7 @@ static inline struct request *blk_map_queue_find_tag(struct blk_queue_tag *bqt, | |||
809 | return bqt->tag_index[tag]; | 810 | return bqt->tag_index[tag]; |
810 | } | 811 | } |
811 | 812 | ||
812 | extern void blk_rq_bio_prep(request_queue_t *, struct request *, struct bio *); | 813 | extern void blk_rq_bio_prep(struct request_queue *, struct request *, struct bio *); |
813 | extern int blkdev_issue_flush(struct block_device *, sector_t *); | 814 | extern int blkdev_issue_flush(struct block_device *, sector_t *); |
814 | 815 | ||
815 | #define MAX_PHYS_SEGMENTS 128 | 816 | #define MAX_PHYS_SEGMENTS 128 |
@@ -821,7 +822,7 @@ extern int blkdev_issue_flush(struct block_device *, sector_t *); | |||
821 | 822 | ||
822 | #define blkdev_entry_to_request(entry) list_entry((entry), struct request, queuelist) | 823 | #define blkdev_entry_to_request(entry) list_entry((entry), struct request, queuelist) |
823 | 824 | ||
824 | static inline int queue_hardsect_size(request_queue_t *q) | 825 | static inline int queue_hardsect_size(struct request_queue *q) |
825 | { | 826 | { |
826 | int retval = 512; | 827 | int retval = 512; |
827 | 828 | ||
@@ -836,7 +837,7 @@ static inline int bdev_hardsect_size(struct block_device *bdev) | |||
836 | return queue_hardsect_size(bdev_get_queue(bdev)); | 837 | return queue_hardsect_size(bdev_get_queue(bdev)); |
837 | } | 838 | } |
838 | 839 | ||
839 | static inline int queue_dma_alignment(request_queue_t *q) | 840 | static inline int queue_dma_alignment(struct request_queue *q) |
840 | { | 841 | { |
841 | int retval = 511; | 842 | int retval = 511; |
842 | 843 | ||
diff --git a/include/linux/blktrace_api.h b/include/linux/blktrace_api.h index 3680ff9a30ed..90874a5d7d78 100644 --- a/include/linux/blktrace_api.h +++ b/include/linux/blktrace_api.h | |||
@@ -144,7 +144,7 @@ struct blk_user_trace_setup { | |||
144 | 144 | ||
145 | #if defined(CONFIG_BLK_DEV_IO_TRACE) | 145 | #if defined(CONFIG_BLK_DEV_IO_TRACE) |
146 | extern int blk_trace_ioctl(struct block_device *, unsigned, char __user *); | 146 | extern int blk_trace_ioctl(struct block_device *, unsigned, char __user *); |
147 | extern void blk_trace_shutdown(request_queue_t *); | 147 | extern void blk_trace_shutdown(struct request_queue *); |
148 | extern void __blk_add_trace(struct blk_trace *, sector_t, int, int, u32, int, int, void *); | 148 | extern void __blk_add_trace(struct blk_trace *, sector_t, int, int, u32, int, int, void *); |
149 | 149 | ||
150 | /** | 150 | /** |
diff --git a/include/linux/compiler.h b/include/linux/compiler.h index 12a1291855e2..86f9a3a6137d 100644 --- a/include/linux/compiler.h +++ b/include/linux/compiler.h | |||
@@ -15,8 +15,8 @@ | |||
15 | # define __acquire(x) __context__(x,1) | 15 | # define __acquire(x) __context__(x,1) |
16 | # define __release(x) __context__(x,-1) | 16 | # define __release(x) __context__(x,-1) |
17 | # define __cond_lock(x,c) ((c) ? ({ __acquire(x); 1; }) : 0) | 17 | # define __cond_lock(x,c) ((c) ? ({ __acquire(x); 1; }) : 0) |
18 | extern void __chk_user_ptr(const void __user *); | 18 | extern void __chk_user_ptr(const volatile void __user *); |
19 | extern void __chk_io_ptr(const void __iomem *); | 19 | extern void __chk_io_ptr(const volatile void __iomem *); |
20 | #else | 20 | #else |
21 | # define __user | 21 | # define __user |
22 | # define __kernel | 22 | # define __kernel |
diff --git a/include/linux/device.h b/include/linux/device.h index d9f0a57f5a2f..3a38d1f70cb7 100644 --- a/include/linux/device.h +++ b/include/linux/device.h | |||
@@ -551,6 +551,9 @@ extern void put_device(struct device * dev); | |||
551 | /* drivers/base/power/shutdown.c */ | 551 | /* drivers/base/power/shutdown.c */ |
552 | extern void device_shutdown(void); | 552 | extern void device_shutdown(void); |
553 | 553 | ||
554 | /* drivers/base/sys.c */ | ||
555 | extern void sysdev_shutdown(void); | ||
556 | |||
554 | 557 | ||
555 | /* drivers/base/firmware.c */ | 558 | /* drivers/base/firmware.c */ |
556 | extern int __must_check firmware_register(struct kset *); | 559 | extern int __must_check firmware_register(struct kset *); |
diff --git a/include/linux/elevator.h b/include/linux/elevator.h index e88fcbc77f8f..e8f42133a616 100644 --- a/include/linux/elevator.h +++ b/include/linux/elevator.h | |||
@@ -5,29 +5,29 @@ | |||
5 | 5 | ||
6 | #ifdef CONFIG_BLOCK | 6 | #ifdef CONFIG_BLOCK |
7 | 7 | ||
8 | typedef int (elevator_merge_fn) (request_queue_t *, struct request **, | 8 | typedef int (elevator_merge_fn) (struct request_queue *, struct request **, |
9 | struct bio *); | 9 | struct bio *); |
10 | 10 | ||
11 | typedef void (elevator_merge_req_fn) (request_queue_t *, struct request *, struct request *); | 11 | typedef void (elevator_merge_req_fn) (struct request_queue *, struct request *, struct request *); |
12 | 12 | ||
13 | typedef void (elevator_merged_fn) (request_queue_t *, struct request *, int); | 13 | typedef void (elevator_merged_fn) (struct request_queue *, struct request *, int); |
14 | 14 | ||
15 | typedef int (elevator_allow_merge_fn) (request_queue_t *, struct request *, struct bio *); | 15 | typedef int (elevator_allow_merge_fn) (struct request_queue *, struct request *, struct bio *); |
16 | 16 | ||
17 | typedef int (elevator_dispatch_fn) (request_queue_t *, int); | 17 | typedef int (elevator_dispatch_fn) (struct request_queue *, int); |
18 | 18 | ||
19 | typedef void (elevator_add_req_fn) (request_queue_t *, struct request *); | 19 | typedef void (elevator_add_req_fn) (struct request_queue *, struct request *); |
20 | typedef int (elevator_queue_empty_fn) (request_queue_t *); | 20 | typedef int (elevator_queue_empty_fn) (struct request_queue *); |
21 | typedef struct request *(elevator_request_list_fn) (request_queue_t *, struct request *); | 21 | typedef struct request *(elevator_request_list_fn) (struct request_queue *, struct request *); |
22 | typedef void (elevator_completed_req_fn) (request_queue_t *, struct request *); | 22 | typedef void (elevator_completed_req_fn) (struct request_queue *, struct request *); |
23 | typedef int (elevator_may_queue_fn) (request_queue_t *, int); | 23 | typedef int (elevator_may_queue_fn) (struct request_queue *, int); |
24 | 24 | ||
25 | typedef int (elevator_set_req_fn) (request_queue_t *, struct request *, gfp_t); | 25 | typedef int (elevator_set_req_fn) (struct request_queue *, struct request *, gfp_t); |
26 | typedef void (elevator_put_req_fn) (struct request *); | 26 | typedef void (elevator_put_req_fn) (struct request *); |
27 | typedef void (elevator_activate_req_fn) (request_queue_t *, struct request *); | 27 | typedef void (elevator_activate_req_fn) (struct request_queue *, struct request *); |
28 | typedef void (elevator_deactivate_req_fn) (request_queue_t *, struct request *); | 28 | typedef void (elevator_deactivate_req_fn) (struct request_queue *, struct request *); |
29 | 29 | ||
30 | typedef void *(elevator_init_fn) (request_queue_t *); | 30 | typedef void *(elevator_init_fn) (struct request_queue *); |
31 | typedef void (elevator_exit_fn) (elevator_t *); | 31 | typedef void (elevator_exit_fn) (elevator_t *); |
32 | 32 | ||
33 | struct elevator_ops | 33 | struct elevator_ops |
@@ -94,27 +94,27 @@ struct elevator_queue | |||
94 | /* | 94 | /* |
95 | * block elevator interface | 95 | * block elevator interface |
96 | */ | 96 | */ |
97 | extern void elv_dispatch_sort(request_queue_t *, struct request *); | 97 | extern void elv_dispatch_sort(struct request_queue *, struct request *); |
98 | extern void elv_dispatch_add_tail(request_queue_t *, struct request *); | 98 | extern void elv_dispatch_add_tail(struct request_queue *, struct request *); |
99 | extern void elv_add_request(request_queue_t *, struct request *, int, int); | 99 | extern void elv_add_request(struct request_queue *, struct request *, int, int); |
100 | extern void __elv_add_request(request_queue_t *, struct request *, int, int); | 100 | extern void __elv_add_request(struct request_queue *, struct request *, int, int); |
101 | extern void elv_insert(request_queue_t *, struct request *, int); | 101 | extern void elv_insert(struct request_queue *, struct request *, int); |
102 | extern int elv_merge(request_queue_t *, struct request **, struct bio *); | 102 | extern int elv_merge(struct request_queue *, struct request **, struct bio *); |
103 | extern void elv_merge_requests(request_queue_t *, struct request *, | 103 | extern void elv_merge_requests(struct request_queue *, struct request *, |
104 | struct request *); | 104 | struct request *); |
105 | extern void elv_merged_request(request_queue_t *, struct request *, int); | 105 | extern void elv_merged_request(struct request_queue *, struct request *, int); |
106 | extern void elv_dequeue_request(request_queue_t *, struct request *); | 106 | extern void elv_dequeue_request(struct request_queue *, struct request *); |
107 | extern void elv_requeue_request(request_queue_t *, struct request *); | 107 | extern void elv_requeue_request(struct request_queue *, struct request *); |
108 | extern int elv_queue_empty(request_queue_t *); | 108 | extern int elv_queue_empty(struct request_queue *); |
109 | extern struct request *elv_next_request(struct request_queue *q); | 109 | extern struct request *elv_next_request(struct request_queue *q); |
110 | extern struct request *elv_former_request(request_queue_t *, struct request *); | 110 | extern struct request *elv_former_request(struct request_queue *, struct request *); |
111 | extern struct request *elv_latter_request(request_queue_t *, struct request *); | 111 | extern struct request *elv_latter_request(struct request_queue *, struct request *); |
112 | extern int elv_register_queue(request_queue_t *q); | 112 | extern int elv_register_queue(struct request_queue *q); |
113 | extern void elv_unregister_queue(request_queue_t *q); | 113 | extern void elv_unregister_queue(struct request_queue *q); |
114 | extern int elv_may_queue(request_queue_t *, int); | 114 | extern int elv_may_queue(struct request_queue *, int); |
115 | extern void elv_completed_request(request_queue_t *, struct request *); | 115 | extern void elv_completed_request(struct request_queue *, struct request *); |
116 | extern int elv_set_request(request_queue_t *, struct request *, gfp_t); | 116 | extern int elv_set_request(struct request_queue *, struct request *, gfp_t); |
117 | extern void elv_put_request(request_queue_t *, struct request *); | 117 | extern void elv_put_request(struct request_queue *, struct request *); |
118 | 118 | ||
119 | /* | 119 | /* |
120 | * io scheduler registration | 120 | * io scheduler registration |
@@ -125,18 +125,18 @@ extern void elv_unregister(struct elevator_type *); | |||
125 | /* | 125 | /* |
126 | * io scheduler sysfs switching | 126 | * io scheduler sysfs switching |
127 | */ | 127 | */ |
128 | extern ssize_t elv_iosched_show(request_queue_t *, char *); | 128 | extern ssize_t elv_iosched_show(struct request_queue *, char *); |
129 | extern ssize_t elv_iosched_store(request_queue_t *, const char *, size_t); | 129 | extern ssize_t elv_iosched_store(struct request_queue *, const char *, size_t); |
130 | 130 | ||
131 | extern int elevator_init(request_queue_t *, char *); | 131 | extern int elevator_init(struct request_queue *, char *); |
132 | extern void elevator_exit(elevator_t *); | 132 | extern void elevator_exit(elevator_t *); |
133 | extern int elv_rq_merge_ok(struct request *, struct bio *); | 133 | extern int elv_rq_merge_ok(struct request *, struct bio *); |
134 | 134 | ||
135 | /* | 135 | /* |
136 | * Helper functions. | 136 | * Helper functions. |
137 | */ | 137 | */ |
138 | extern struct request *elv_rb_former_request(request_queue_t *, struct request *); | 138 | extern struct request *elv_rb_former_request(struct request_queue *, struct request *); |
139 | extern struct request *elv_rb_latter_request(request_queue_t *, struct request *); | 139 | extern struct request *elv_rb_latter_request(struct request_queue *, struct request *); |
140 | 140 | ||
141 | /* | 141 | /* |
142 | * rb support functions. | 142 | * rb support functions. |
diff --git a/include/linux/ide.h b/include/linux/ide.h index 5f5daad8bc54..d71d0121b7f9 100644 --- a/include/linux/ide.h +++ b/include/linux/ide.h | |||
@@ -555,7 +555,7 @@ typedef struct ide_drive_s { | |||
555 | char name[4]; /* drive name, such as "hda" */ | 555 | char name[4]; /* drive name, such as "hda" */ |
556 | char driver_req[10]; /* requests specific driver */ | 556 | char driver_req[10]; /* requests specific driver */ |
557 | 557 | ||
558 | request_queue_t *queue; /* request queue */ | 558 | struct request_queue *queue; /* request queue */ |
559 | 559 | ||
560 | struct request *rq; /* current request */ | 560 | struct request *rq; /* current request */ |
561 | struct ide_drive_s *next; /* circular list of hwgroup drives */ | 561 | struct ide_drive_s *next; /* circular list of hwgroup drives */ |
@@ -1206,7 +1206,7 @@ extern void ide_stall_queue(ide_drive_t *drive, unsigned long timeout); | |||
1206 | extern int ide_spin_wait_hwgroup(ide_drive_t *); | 1206 | extern int ide_spin_wait_hwgroup(ide_drive_t *); |
1207 | extern void ide_timer_expiry(unsigned long); | 1207 | extern void ide_timer_expiry(unsigned long); |
1208 | extern irqreturn_t ide_intr(int irq, void *dev_id); | 1208 | extern irqreturn_t ide_intr(int irq, void *dev_id); |
1209 | extern void do_ide_request(request_queue_t *); | 1209 | extern void do_ide_request(struct request_queue *); |
1210 | 1210 | ||
1211 | void ide_init_disk(struct gendisk *, ide_drive_t *); | 1211 | void ide_init_disk(struct gendisk *, ide_drive_t *); |
1212 | 1212 | ||
diff --git a/include/linux/lguest.h b/include/linux/lguest.h index 500aace21ca7..e76c151c7129 100644 --- a/include/linux/lguest.h +++ b/include/linux/lguest.h | |||
@@ -27,18 +27,38 @@ | |||
27 | #define LG_CLOCK_MIN_DELTA 100UL | 27 | #define LG_CLOCK_MIN_DELTA 100UL |
28 | #define LG_CLOCK_MAX_DELTA ULONG_MAX | 28 | #define LG_CLOCK_MAX_DELTA ULONG_MAX |
29 | 29 | ||
30 | /*G:031 First, how does our Guest contact the Host to ask for privileged | ||
31 | * operations? There are two ways: the direct way is to make a "hypercall", | ||
32 | * to make requests of the Host Itself. | ||
33 | * | ||
34 | * Our hypercall mechanism uses the highest unused trap code (traps 32 and | ||
35 | * above are used by real hardware interrupts). Seventeen hypercalls are | ||
36 | * available: the hypercall number is put in the %eax register, and the | ||
37 | * arguments (when required) are placed in %edx, %ebx and %ecx. If a return | ||
38 | * value makes sense, it's returned in %eax. | ||
39 | * | ||
40 | * Grossly invalid calls result in Sudden Death at the hands of the vengeful | ||
41 | * Host, rather than returning failure. This reflects Winston Churchill's | ||
42 | * definition of a gentleman: "someone who is only rude intentionally". */ | ||
30 | #define LGUEST_TRAP_ENTRY 0x1F | 43 | #define LGUEST_TRAP_ENTRY 0x1F |
31 | 44 | ||
32 | static inline unsigned long | 45 | static inline unsigned long |
33 | hcall(unsigned long call, | 46 | hcall(unsigned long call, |
34 | unsigned long arg1, unsigned long arg2, unsigned long arg3) | 47 | unsigned long arg1, unsigned long arg2, unsigned long arg3) |
35 | { | 48 | { |
49 | /* "int" is the Intel instruction to trigger a trap. */ | ||
36 | asm volatile("int $" __stringify(LGUEST_TRAP_ENTRY) | 50 | asm volatile("int $" __stringify(LGUEST_TRAP_ENTRY) |
51 | /* The call is in %eax (aka "a"), and can be replaced */ | ||
37 | : "=a"(call) | 52 | : "=a"(call) |
53 | /* The other arguments are in %eax, %edx, %ebx & %ecx */ | ||
38 | : "a"(call), "d"(arg1), "b"(arg2), "c"(arg3) | 54 | : "a"(call), "d"(arg1), "b"(arg2), "c"(arg3) |
55 | /* "memory" means this might write somewhere in memory. | ||
56 | * This isn't true for all calls, but it's safe to tell | ||
57 | * gcc that it might happen so it doesn't get clever. */ | ||
39 | : "memory"); | 58 | : "memory"); |
40 | return call; | 59 | return call; |
41 | } | 60 | } |
61 | /*:*/ | ||
42 | 62 | ||
43 | void async_hcall(unsigned long call, | 63 | void async_hcall(unsigned long call, |
44 | unsigned long arg1, unsigned long arg2, unsigned long arg3); | 64 | unsigned long arg1, unsigned long arg2, unsigned long arg3); |
@@ -52,31 +72,40 @@ struct hcall_ring | |||
52 | u32 eax, edx, ebx, ecx; | 72 | u32 eax, edx, ebx, ecx; |
53 | }; | 73 | }; |
54 | 74 | ||
55 | /* All the good stuff happens here: guest registers it with LGUEST_INIT */ | 75 | /*G:032 The second method of communicating with the Host is to via "struct |
76 | * lguest_data". The Guest's very first hypercall is to tell the Host where | ||
77 | * this is, and then the Guest and Host both publish information in it. :*/ | ||
56 | struct lguest_data | 78 | struct lguest_data |
57 | { | 79 | { |
58 | /* Fields which change during running: */ | 80 | /* 512 == enabled (same as eflags in normal hardware). The Guest |
59 | /* 512 == enabled (same as eflags) */ | 81 | * changes interrupts so often that a hypercall is too slow. */ |
60 | unsigned int irq_enabled; | 82 | unsigned int irq_enabled; |
61 | /* Interrupts blocked by guest. */ | 83 | /* Fine-grained interrupt disabling by the Guest */ |
62 | DECLARE_BITMAP(blocked_interrupts, LGUEST_IRQS); | 84 | DECLARE_BITMAP(blocked_interrupts, LGUEST_IRQS); |
63 | 85 | ||
64 | /* Virtual address of page fault. */ | 86 | /* The Host writes the virtual address of the last page fault here, |
87 | * which saves the Guest a hypercall. CR2 is the native register where | ||
88 | * this address would normally be found. */ | ||
65 | unsigned long cr2; | 89 | unsigned long cr2; |
66 | 90 | ||
67 | /* Async hypercall ring. 0xFF == done, 0 == pending. */ | 91 | /* Async hypercall ring. Instead of directly making hypercalls, we can |
92 | * place them in here for processing the next time the Host wants. | ||
93 | * This batching can be quite efficient. */ | ||
94 | |||
95 | /* 0xFF == done (set by Host), 0 == pending (set by Guest). */ | ||
68 | u8 hcall_status[LHCALL_RING_SIZE]; | 96 | u8 hcall_status[LHCALL_RING_SIZE]; |
97 | /* The actual registers for the hypercalls. */ | ||
69 | struct hcall_ring hcalls[LHCALL_RING_SIZE]; | 98 | struct hcall_ring hcalls[LHCALL_RING_SIZE]; |
70 | 99 | ||
71 | /* Fields initialized by the hypervisor at boot: */ | 100 | /* Fields initialized by the Host at boot: */ |
72 | /* Memory not to try to access */ | 101 | /* Memory not to try to access */ |
73 | unsigned long reserve_mem; | 102 | unsigned long reserve_mem; |
74 | /* ID of this guest (used by network driver to set ethernet address) */ | 103 | /* ID of this Guest (used by network driver to set ethernet address) */ |
75 | u16 guestid; | 104 | u16 guestid; |
76 | /* KHz for the TSC clock. */ | 105 | /* KHz for the TSC clock. */ |
77 | u32 tsc_khz; | 106 | u32 tsc_khz; |
78 | 107 | ||
79 | /* Fields initialized by the guest at boot: */ | 108 | /* Fields initialized by the Guest at boot: */ |
80 | /* Instruction range to suppress interrupts even if enabled */ | 109 | /* Instruction range to suppress interrupts even if enabled */ |
81 | unsigned long noirq_start, noirq_end; | 110 | unsigned long noirq_start, noirq_end; |
82 | }; | 111 | }; |
diff --git a/include/linux/lguest_bus.h b/include/linux/lguest_bus.h index c9b4e05fee49..d27853ddc644 100644 --- a/include/linux/lguest_bus.h +++ b/include/linux/lguest_bus.h | |||
@@ -15,11 +15,14 @@ struct lguest_device { | |||
15 | void *private; | 15 | void *private; |
16 | }; | 16 | }; |
17 | 17 | ||
18 | /* By convention, each device can use irq index+1 if it wants to. */ | 18 | /*D:380 Since interrupt numbers are arbitrary, we use a convention: each device |
19 | * can use the interrupt number corresponding to its index. The +1 is because | ||
20 | * interrupt 0 is not usable (it's actually the timer interrupt). */ | ||
19 | static inline int lgdev_irq(const struct lguest_device *dev) | 21 | static inline int lgdev_irq(const struct lguest_device *dev) |
20 | { | 22 | { |
21 | return dev->index + 1; | 23 | return dev->index + 1; |
22 | } | 24 | } |
25 | /*:*/ | ||
23 | 26 | ||
24 | /* dma args must not be vmalloced! */ | 27 | /* dma args must not be vmalloced! */ |
25 | void lguest_send_dma(unsigned long key, struct lguest_dma *dma); | 28 | void lguest_send_dma(unsigned long key, struct lguest_dma *dma); |
diff --git a/include/linux/lguest_launcher.h b/include/linux/lguest_launcher.h index 0ba414a40c80..641670579446 100644 --- a/include/linux/lguest_launcher.h +++ b/include/linux/lguest_launcher.h | |||
@@ -9,14 +9,45 @@ | |||
9 | /* How many devices? Assume each one wants up to two dma arrays per device. */ | 9 | /* How many devices? Assume each one wants up to two dma arrays per device. */ |
10 | #define LGUEST_MAX_DEVICES (LGUEST_MAX_DMA/2) | 10 | #define LGUEST_MAX_DEVICES (LGUEST_MAX_DMA/2) |
11 | 11 | ||
12 | /*D:200 | ||
13 | * Lguest I/O | ||
14 | * | ||
15 | * The lguest I/O mechanism is the only way Guests can talk to devices. There | ||
16 | * are two hypercalls involved: SEND_DMA for output and BIND_DMA for input. In | ||
17 | * each case, "struct lguest_dma" describes the buffer: this contains 16 | ||
18 | * addr/len pairs, and if there are fewer buffer elements the len array is | ||
19 | * terminated with a 0. | ||
20 | * | ||
21 | * I/O is organized by keys: BIND_DMA attaches buffers to a particular key, and | ||
22 | * SEND_DMA transfers to buffers bound to particular key. By convention, keys | ||
23 | * correspond to a physical address within the device's page. This means that | ||
24 | * devices will never accidentally end up with the same keys, and allows the | ||
25 | * Host use The Futex Trick (as we'll see later in our journey). | ||
26 | * | ||
27 | * SEND_DMA simply indicates a key to send to, and the physical address of the | ||
28 | * "struct lguest_dma" to send. The Host will write the number of bytes | ||
29 | * transferred into the "struct lguest_dma"'s used_len member. | ||
30 | * | ||
31 | * BIND_DMA indicates a key to bind to, a pointer to an array of "struct | ||
32 | * lguest_dma"s ready for receiving, the size of that array, and an interrupt | ||
33 | * to trigger when data is received. The Host will only allow transfers into | ||
34 | * buffers with a used_len of zero: it then sets used_len to the number of | ||
35 | * bytes transferred and triggers the interrupt for the Guest to process the | ||
36 | * new input. */ | ||
12 | struct lguest_dma | 37 | struct lguest_dma |
13 | { | 38 | { |
14 | /* 0 if free to be used, filled by hypervisor. */ | 39 | /* 0 if free to be used, filled by the Host. */ |
15 | u32 used_len; | 40 | u32 used_len; |
16 | unsigned long addr[LGUEST_MAX_DMA_SECTIONS]; | 41 | unsigned long addr[LGUEST_MAX_DMA_SECTIONS]; |
17 | u16 len[LGUEST_MAX_DMA_SECTIONS]; | 42 | u16 len[LGUEST_MAX_DMA_SECTIONS]; |
18 | }; | 43 | }; |
44 | /*:*/ | ||
19 | 45 | ||
46 | /*D:460 This is the layout of a block device memory page. The Launcher sets up | ||
47 | * the num_sectors initially to tell the Guest the size of the disk. The Guest | ||
48 | * puts the type, sector and length of the request in the first three fields, | ||
49 | * then DMAs to the Host. The Host processes the request, sets up the result, | ||
50 | * then DMAs back to the Guest. */ | ||
20 | struct lguest_block_page | 51 | struct lguest_block_page |
21 | { | 52 | { |
22 | /* 0 is a read, 1 is a write. */ | 53 | /* 0 is a read, 1 is a write. */ |
@@ -28,27 +59,47 @@ struct lguest_block_page | |||
28 | u32 num_sectors; /* Disk length = num_sectors * 512 */ | 59 | u32 num_sectors; /* Disk length = num_sectors * 512 */ |
29 | }; | 60 | }; |
30 | 61 | ||
31 | /* There is a shared page of these. */ | 62 | /*D:520 The network device is basically a memory page where all the Guests on |
63 | * the network publish their MAC (ethernet) addresses: it's an array of "struct | ||
64 | * lguest_net": */ | ||
32 | struct lguest_net | 65 | struct lguest_net |
33 | { | 66 | { |
34 | /* Simply the mac address (with multicast bit meaning promisc). */ | 67 | /* Simply the mac address (with multicast bit meaning promisc). */ |
35 | unsigned char mac[6]; | 68 | unsigned char mac[6]; |
36 | }; | 69 | }; |
70 | /*:*/ | ||
37 | 71 | ||
38 | /* Where the Host expects the Guest to SEND_DMA console output to. */ | 72 | /* Where the Host expects the Guest to SEND_DMA console output to. */ |
39 | #define LGUEST_CONSOLE_DMA_KEY 0 | 73 | #define LGUEST_CONSOLE_DMA_KEY 0 |
40 | 74 | ||
41 | /* We have a page of these descriptors in the lguest_device page. */ | 75 | /*D:010 |
76 | * Drivers | ||
77 | * | ||
78 | * The Guest needs devices to do anything useful. Since we don't let it touch | ||
79 | * real devices (think of the damage it could do!) we provide virtual devices. | ||
80 | * We could emulate a PCI bus with various devices on it, but that is a fairly | ||
81 | * complex burden for the Host and suboptimal for the Guest, so we have our own | ||
82 | * "lguest" bus and simple drivers. | ||
83 | * | ||
84 | * Devices are described by an array of LGUEST_MAX_DEVICES of these structs, | ||
85 | * placed by the Launcher just above the top of physical memory: | ||
86 | */ | ||
42 | struct lguest_device_desc { | 87 | struct lguest_device_desc { |
88 | /* The device type: console, network, disk etc. */ | ||
43 | u16 type; | 89 | u16 type; |
44 | #define LGUEST_DEVICE_T_CONSOLE 1 | 90 | #define LGUEST_DEVICE_T_CONSOLE 1 |
45 | #define LGUEST_DEVICE_T_NET 2 | 91 | #define LGUEST_DEVICE_T_NET 2 |
46 | #define LGUEST_DEVICE_T_BLOCK 3 | 92 | #define LGUEST_DEVICE_T_BLOCK 3 |
47 | 93 | ||
94 | /* The specific features of this device: these depends on device type | ||
95 | * except for LGUEST_DEVICE_F_RANDOMNESS. */ | ||
48 | u16 features; | 96 | u16 features; |
49 | #define LGUEST_NET_F_NOCSUM 0x4000 /* Don't bother checksumming */ | 97 | #define LGUEST_NET_F_NOCSUM 0x4000 /* Don't bother checksumming */ |
50 | #define LGUEST_DEVICE_F_RANDOMNESS 0x8000 /* IRQ is fairly random */ | 98 | #define LGUEST_DEVICE_F_RANDOMNESS 0x8000 /* IRQ is fairly random */ |
51 | 99 | ||
100 | /* This is how the Guest reports status of the device: the Host can set | ||
101 | * LGUEST_DEVICE_S_REMOVED to indicate removal, but the rest are only | ||
102 | * ever manipulated by the Guest, and only ever set. */ | ||
52 | u16 status; | 103 | u16 status; |
53 | /* 256 and above are device specific. */ | 104 | /* 256 and above are device specific. */ |
54 | #define LGUEST_DEVICE_S_ACKNOWLEDGE 1 /* We have seen device. */ | 105 | #define LGUEST_DEVICE_S_ACKNOWLEDGE 1 /* We have seen device. */ |
@@ -58,9 +109,12 @@ struct lguest_device_desc { | |||
58 | #define LGUEST_DEVICE_S_REMOVED_ACK 16 /* Driver has been told. */ | 109 | #define LGUEST_DEVICE_S_REMOVED_ACK 16 /* Driver has been told. */ |
59 | #define LGUEST_DEVICE_S_FAILED 128 /* Something actually failed */ | 110 | #define LGUEST_DEVICE_S_FAILED 128 /* Something actually failed */ |
60 | 111 | ||
112 | /* Each device exists somewhere in Guest physical memory, over some | ||
113 | * number of pages. */ | ||
61 | u16 num_pages; | 114 | u16 num_pages; |
62 | u32 pfn; | 115 | u32 pfn; |
63 | }; | 116 | }; |
117 | /*:*/ | ||
64 | 118 | ||
65 | /* Write command first word is a request. */ | 119 | /* Write command first word is a request. */ |
66 | enum lguest_req | 120 | enum lguest_req |
diff --git a/include/linux/libata.h b/include/linux/libata.h index 9aa6c10f7bb1..41978a557318 100644 --- a/include/linux/libata.h +++ b/include/linux/libata.h | |||
@@ -216,6 +216,8 @@ enum { | |||
216 | ATA_HOST_SIMPLEX = (1 << 0), /* Host is simplex, one DMA channel per host only */ | 216 | ATA_HOST_SIMPLEX = (1 << 0), /* Host is simplex, one DMA channel per host only */ |
217 | ATA_HOST_STARTED = (1 << 1), /* Host started */ | 217 | ATA_HOST_STARTED = (1 << 1), /* Host started */ |
218 | 218 | ||
219 | /* bits 24:31 of host->flags are reserved for LLD specific flags */ | ||
220 | |||
219 | /* various lengths of time */ | 221 | /* various lengths of time */ |
220 | ATA_TMOUT_BOOT = 30 * HZ, /* heuristic */ | 222 | ATA_TMOUT_BOOT = 30 * HZ, /* heuristic */ |
221 | ATA_TMOUT_BOOT_QUICK = 7 * HZ, /* heuristic */ | 223 | ATA_TMOUT_BOOT_QUICK = 7 * HZ, /* heuristic */ |
diff --git a/include/linux/mm.h b/include/linux/mm.h index c456c3a1c28e..3e9e8fec5a41 100644 --- a/include/linux/mm.h +++ b/include/linux/mm.h | |||
@@ -1246,7 +1246,7 @@ void drop_slab(void); | |||
1246 | extern int randomize_va_space; | 1246 | extern int randomize_va_space; |
1247 | #endif | 1247 | #endif |
1248 | 1248 | ||
1249 | __attribute__((weak)) const char *arch_vma_name(struct vm_area_struct *vma); | 1249 | const char * arch_vma_name(struct vm_area_struct *vma); |
1250 | 1250 | ||
1251 | #endif /* __KERNEL__ */ | 1251 | #endif /* __KERNEL__ */ |
1252 | #endif /* _LINUX_MM_H */ | 1252 | #endif /* _LINUX_MM_H */ |
diff --git a/include/linux/mod_devicetable.h b/include/linux/mod_devicetable.h index af04a555b52c..2ada8ee316b3 100644 --- a/include/linux/mod_devicetable.h +++ b/include/linux/mod_devicetable.h | |||
@@ -159,6 +159,12 @@ struct ap_device_id { | |||
159 | 159 | ||
160 | #define AP_DEVICE_ID_MATCH_DEVICE_TYPE 0x01 | 160 | #define AP_DEVICE_ID_MATCH_DEVICE_TYPE 0x01 |
161 | 161 | ||
162 | #define ACPI_ID_LEN 9 | ||
163 | |||
164 | struct acpi_device_id { | ||
165 | __u8 id[ACPI_ID_LEN]; | ||
166 | kernel_ulong_t driver_data; | ||
167 | }; | ||
162 | 168 | ||
163 | #define PNP_ID_LEN 8 | 169 | #define PNP_ID_LEN 8 |
164 | #define PNP_MAX_DEVICES 8 | 170 | #define PNP_MAX_DEVICES 8 |
diff --git a/include/linux/netfilter/xt_connlimit.h b/include/linux/netfilter/xt_connlimit.h index 90ae8b474cb8..37e933c9987d 100644 --- a/include/linux/netfilter/xt_connlimit.h +++ b/include/linux/netfilter/xt_connlimit.h | |||
@@ -5,8 +5,8 @@ struct xt_connlimit_data; | |||
5 | 5 | ||
6 | struct xt_connlimit_info { | 6 | struct xt_connlimit_info { |
7 | union { | 7 | union { |
8 | u_int32_t v4_mask; | 8 | __be32 v4_mask; |
9 | u_int32_t v6_mask[4]; | 9 | __be32 v6_mask[4]; |
10 | }; | 10 | }; |
11 | unsigned int limit, inverse; | 11 | unsigned int limit, inverse; |
12 | 12 | ||
diff --git a/include/linux/pci.h b/include/linux/pci.h index 5e84f2e8d54c..d8f8a3a96644 100644 --- a/include/linux/pci.h +++ b/include/linux/pci.h | |||
@@ -534,6 +534,7 @@ static inline int pci_write_config_dword(struct pci_dev *dev, int where, u32 val | |||
534 | 534 | ||
535 | int __must_check pci_enable_device(struct pci_dev *dev); | 535 | int __must_check pci_enable_device(struct pci_dev *dev); |
536 | int __must_check pci_enable_device_bars(struct pci_dev *dev, int mask); | 536 | int __must_check pci_enable_device_bars(struct pci_dev *dev, int mask); |
537 | int __must_check __pci_reenable_device(struct pci_dev *); | ||
537 | int __must_check pcim_enable_device(struct pci_dev *pdev); | 538 | int __must_check pcim_enable_device(struct pci_dev *pdev); |
538 | void pcim_pin_device(struct pci_dev *pdev); | 539 | void pcim_pin_device(struct pci_dev *pdev); |
539 | 540 | ||
diff --git a/include/linux/pci_ids.h b/include/linux/pci_ids.h index 0befd9513f85..69d68117bdac 100644 --- a/include/linux/pci_ids.h +++ b/include/linux/pci_ids.h | |||
@@ -1225,6 +1225,10 @@ | |||
1225 | #define PCI_DEVICE_ID_NVIDIA_NVENET_25 0x054D | 1225 | #define PCI_DEVICE_ID_NVIDIA_NVENET_25 0x054D |
1226 | #define PCI_DEVICE_ID_NVIDIA_NVENET_26 0x054E | 1226 | #define PCI_DEVICE_ID_NVIDIA_NVENET_26 0x054E |
1227 | #define PCI_DEVICE_ID_NVIDIA_NVENET_27 0x054F | 1227 | #define PCI_DEVICE_ID_NVIDIA_NVENET_27 0x054F |
1228 | #define PCI_DEVICE_ID_NVIDIA_NVENET_28 0x07DC | ||
1229 | #define PCI_DEVICE_ID_NVIDIA_NVENET_29 0x07DD | ||
1230 | #define PCI_DEVICE_ID_NVIDIA_NVENET_30 0x07DE | ||
1231 | #define PCI_DEVICE_ID_NVIDIA_NVENET_31 0x07DF | ||
1228 | #define PCI_DEVICE_ID_NVIDIA_NFORCE_MCP67_IDE 0x0560 | 1232 | #define PCI_DEVICE_ID_NVIDIA_NFORCE_MCP67_IDE 0x0560 |
1229 | #define PCI_DEVICE_ID_NVIDIA_NFORCE_MCP73_IDE 0x056C | 1233 | #define PCI_DEVICE_ID_NVIDIA_NFORCE_MCP73_IDE 0x056C |
1230 | #define PCI_DEVICE_ID_NVIDIA_NFORCE_MCP77_IDE 0x0759 | 1234 | #define PCI_DEVICE_ID_NVIDIA_NFORCE_MCP77_IDE 0x0759 |
@@ -2077,6 +2081,23 @@ | |||
2077 | #define PCI_VENDOR_ID_TDI 0x192E | 2081 | #define PCI_VENDOR_ID_TDI 0x192E |
2078 | #define PCI_DEVICE_ID_TDI_EHCI 0x0101 | 2082 | #define PCI_DEVICE_ID_TDI_EHCI 0x0101 |
2079 | 2083 | ||
2084 | #define PCI_VENDOR_ID_FREESCALE 0x1957 | ||
2085 | #define PCI_DEVICE_ID_MPC8548E 0x0012 | ||
2086 | #define PCI_DEVICE_ID_MPC8548 0x0013 | ||
2087 | #define PCI_DEVICE_ID_MPC8543E 0x0014 | ||
2088 | #define PCI_DEVICE_ID_MPC8543 0x0015 | ||
2089 | #define PCI_DEVICE_ID_MPC8547E 0x0018 | ||
2090 | #define PCI_DEVICE_ID_MPC8545E 0x0019 | ||
2091 | #define PCI_DEVICE_ID_MPC8545 0x001a | ||
2092 | #define PCI_DEVICE_ID_MPC8568E 0x0020 | ||
2093 | #define PCI_DEVICE_ID_MPC8568 0x0021 | ||
2094 | #define PCI_DEVICE_ID_MPC8567E 0x0022 | ||
2095 | #define PCI_DEVICE_ID_MPC8567 0x0023 | ||
2096 | #define PCI_DEVICE_ID_MPC8544E 0x0030 | ||
2097 | #define PCI_DEVICE_ID_MPC8544 0x0031 | ||
2098 | #define PCI_DEVICE_ID_MPC8641 0x7010 | ||
2099 | #define PCI_DEVICE_ID_MPC8641D 0x7011 | ||
2100 | |||
2080 | #define PCI_VENDOR_ID_PASEMI 0x1959 | 2101 | #define PCI_VENDOR_ID_PASEMI 0x1959 |
2081 | 2102 | ||
2082 | #define PCI_VENDOR_ID_ATTANSIC 0x1969 | 2103 | #define PCI_VENDOR_ID_ATTANSIC 0x1969 |
diff --git a/include/linux/pnp.h b/include/linux/pnp.h index 2a1897e6f937..16b46aace349 100644 --- a/include/linux/pnp.h +++ b/include/linux/pnp.h | |||
@@ -1,7 +1,6 @@ | |||
1 | /* | 1 | /* |
2 | * Linux Plug and Play Support | 2 | * Linux Plug and Play Support |
3 | * Copyright by Adam Belay <ambx1@neo.rr.com> | 3 | * Copyright by Adam Belay <ambx1@neo.rr.com> |
4 | * | ||
5 | */ | 4 | */ |
6 | 5 | ||
7 | #ifndef _LINUX_PNP_H | 6 | #ifndef _LINUX_PNP_H |
@@ -23,7 +22,6 @@ | |||
23 | struct pnp_protocol; | 22 | struct pnp_protocol; |
24 | struct pnp_dev; | 23 | struct pnp_dev; |
25 | 24 | ||
26 | |||
27 | /* | 25 | /* |
28 | * Resource Management | 26 | * Resource Management |
29 | */ | 27 | */ |
@@ -73,37 +71,37 @@ struct pnp_dev; | |||
73 | #define PNP_PORT_FLAG_FIXED (1<<1) | 71 | #define PNP_PORT_FLAG_FIXED (1<<1) |
74 | 72 | ||
75 | struct pnp_port { | 73 | struct pnp_port { |
76 | unsigned short min; /* min base number */ | 74 | unsigned short min; /* min base number */ |
77 | unsigned short max; /* max base number */ | 75 | unsigned short max; /* max base number */ |
78 | unsigned char align; /* align boundary */ | 76 | unsigned char align; /* align boundary */ |
79 | unsigned char size; /* size of range */ | 77 | unsigned char size; /* size of range */ |
80 | unsigned char flags; /* port flags */ | 78 | unsigned char flags; /* port flags */ |
81 | unsigned char pad; /* pad */ | 79 | unsigned char pad; /* pad */ |
82 | struct pnp_port *next; /* next port */ | 80 | struct pnp_port *next; /* next port */ |
83 | }; | 81 | }; |
84 | 82 | ||
85 | #define PNP_IRQ_NR 256 | 83 | #define PNP_IRQ_NR 256 |
86 | struct pnp_irq { | 84 | struct pnp_irq { |
87 | DECLARE_BITMAP(map, PNP_IRQ_NR); /* bitmaks for IRQ lines */ | 85 | DECLARE_BITMAP(map, PNP_IRQ_NR); /* bitmask for IRQ lines */ |
88 | unsigned char flags; /* IRQ flags */ | 86 | unsigned char flags; /* IRQ flags */ |
89 | unsigned char pad; /* pad */ | 87 | unsigned char pad; /* pad */ |
90 | struct pnp_irq *next; /* next IRQ */ | 88 | struct pnp_irq *next; /* next IRQ */ |
91 | }; | 89 | }; |
92 | 90 | ||
93 | struct pnp_dma { | 91 | struct pnp_dma { |
94 | unsigned char map; /* bitmask for DMA channels */ | 92 | unsigned char map; /* bitmask for DMA channels */ |
95 | unsigned char flags; /* DMA flags */ | 93 | unsigned char flags; /* DMA flags */ |
96 | struct pnp_dma *next; /* next port */ | 94 | struct pnp_dma *next; /* next port */ |
97 | }; | 95 | }; |
98 | 96 | ||
99 | struct pnp_mem { | 97 | struct pnp_mem { |
100 | unsigned int min; /* min base number */ | 98 | unsigned int min; /* min base number */ |
101 | unsigned int max; /* max base number */ | 99 | unsigned int max; /* max base number */ |
102 | unsigned int align; /* align boundary */ | 100 | unsigned int align; /* align boundary */ |
103 | unsigned int size; /* size of range */ | 101 | unsigned int size; /* size of range */ |
104 | unsigned char flags; /* memory flags */ | 102 | unsigned char flags; /* memory flags */ |
105 | unsigned char pad; /* pad */ | 103 | unsigned char pad; /* pad */ |
106 | struct pnp_mem *next; /* next memory resource */ | 104 | struct pnp_mem *next; /* next memory resource */ |
107 | }; | 105 | }; |
108 | 106 | ||
109 | #define PNP_RES_PRIORITY_PREFERRED 0 | 107 | #define PNP_RES_PRIORITY_PREFERRED 0 |
@@ -127,7 +125,6 @@ struct pnp_resource_table { | |||
127 | struct resource irq_resource[PNP_MAX_IRQ]; | 125 | struct resource irq_resource[PNP_MAX_IRQ]; |
128 | }; | 126 | }; |
129 | 127 | ||
130 | |||
131 | /* | 128 | /* |
132 | * Device Managemnt | 129 | * Device Managemnt |
133 | */ | 130 | */ |
@@ -139,14 +136,14 @@ struct pnp_card { | |||
139 | struct list_head protocol_list; /* node in protocol's list of cards */ | 136 | struct list_head protocol_list; /* node in protocol's list of cards */ |
140 | struct list_head devices; /* devices attached to the card */ | 137 | struct list_head devices; /* devices attached to the card */ |
141 | 138 | ||
142 | struct pnp_protocol * protocol; | 139 | struct pnp_protocol *protocol; |
143 | struct pnp_id * id; /* contains supported EISA IDs*/ | 140 | struct pnp_id *id; /* contains supported EISA IDs */ |
144 | 141 | ||
145 | char name[PNP_NAME_LEN]; /* contains a human-readable name */ | 142 | char name[PNP_NAME_LEN]; /* contains a human-readable name */ |
146 | unsigned char pnpver; /* Plug & Play version */ | 143 | unsigned char pnpver; /* Plug & Play version */ |
147 | unsigned char productver; /* product version */ | 144 | unsigned char productver; /* product version */ |
148 | unsigned int serial; /* serial number */ | 145 | unsigned int serial; /* serial number */ |
149 | unsigned char checksum; /* if zero - checksum passed */ | 146 | unsigned char checksum; /* if zero - checksum passed */ |
150 | struct proc_dir_entry *procdir; /* directory entry in /proc/bus/isapnp */ | 147 | struct proc_dir_entry *procdir; /* directory entry in /proc/bus/isapnp */ |
151 | }; | 148 | }; |
152 | 149 | ||
@@ -159,18 +156,18 @@ struct pnp_card { | |||
159 | (card) = global_to_pnp_card((card)->global_list.next)) | 156 | (card) = global_to_pnp_card((card)->global_list.next)) |
160 | 157 | ||
161 | struct pnp_card_link { | 158 | struct pnp_card_link { |
162 | struct pnp_card * card; | 159 | struct pnp_card *card; |
163 | struct pnp_card_driver * driver; | 160 | struct pnp_card_driver *driver; |
164 | void * driver_data; | 161 | void *driver_data; |
165 | pm_message_t pm_state; | 162 | pm_message_t pm_state; |
166 | }; | 163 | }; |
167 | 164 | ||
168 | static inline void *pnp_get_card_drvdata (struct pnp_card_link *pcard) | 165 | static inline void *pnp_get_card_drvdata(struct pnp_card_link *pcard) |
169 | { | 166 | { |
170 | return pcard->driver_data; | 167 | return pcard->driver_data; |
171 | } | 168 | } |
172 | 169 | ||
173 | static inline void pnp_set_card_drvdata (struct pnp_card_link *pcard, void *data) | 170 | static inline void pnp_set_card_drvdata(struct pnp_card_link *pcard, void *data) |
174 | { | 171 | { |
175 | pcard->driver_data = data; | 172 | pcard->driver_data = data; |
176 | } | 173 | } |
@@ -186,22 +183,22 @@ struct pnp_dev { | |||
186 | struct list_head card_list; /* node in card's list of devices */ | 183 | struct list_head card_list; /* node in card's list of devices */ |
187 | struct list_head rdev_list; /* node in cards list of requested devices */ | 184 | struct list_head rdev_list; /* node in cards list of requested devices */ |
188 | 185 | ||
189 | struct pnp_protocol * protocol; | 186 | struct pnp_protocol *protocol; |
190 | struct pnp_card * card; /* card the device is attached to, none if NULL */ | 187 | struct pnp_card *card; /* card the device is attached to, none if NULL */ |
191 | struct pnp_driver * driver; | 188 | struct pnp_driver *driver; |
192 | struct pnp_card_link * card_link; | 189 | struct pnp_card_link *card_link; |
193 | 190 | ||
194 | struct pnp_id * id; /* supported EISA IDs*/ | 191 | struct pnp_id *id; /* supported EISA IDs */ |
195 | 192 | ||
196 | int active; | 193 | int active; |
197 | int capabilities; | 194 | int capabilities; |
198 | struct pnp_option * independent; | 195 | struct pnp_option *independent; |
199 | struct pnp_option * dependent; | 196 | struct pnp_option *dependent; |
200 | struct pnp_resource_table res; | 197 | struct pnp_resource_table res; |
201 | 198 | ||
202 | char name[PNP_NAME_LEN]; /* contains a human-readable name */ | 199 | char name[PNP_NAME_LEN]; /* contains a human-readable name */ |
203 | unsigned short regs; /* ISAPnP: supported registers */ | 200 | unsigned short regs; /* ISAPnP: supported registers */ |
204 | int flags; /* used by protocols */ | 201 | int flags; /* used by protocols */ |
205 | struct proc_dir_entry *procent; /* device entry in /proc/bus/isapnp */ | 202 | struct proc_dir_entry *procent; /* device entry in /proc/bus/isapnp */ |
206 | void *data; | 203 | void *data; |
207 | }; | 204 | }; |
@@ -220,19 +217,19 @@ struct pnp_dev { | |||
220 | (dev) = card_to_pnp_dev((dev)->card_list.next)) | 217 | (dev) = card_to_pnp_dev((dev)->card_list.next)) |
221 | #define pnp_dev_name(dev) (dev)->name | 218 | #define pnp_dev_name(dev) (dev)->name |
222 | 219 | ||
223 | static inline void *pnp_get_drvdata (struct pnp_dev *pdev) | 220 | static inline void *pnp_get_drvdata(struct pnp_dev *pdev) |
224 | { | 221 | { |
225 | return dev_get_drvdata(&pdev->dev); | 222 | return dev_get_drvdata(&pdev->dev); |
226 | } | 223 | } |
227 | 224 | ||
228 | static inline void pnp_set_drvdata (struct pnp_dev *pdev, void *data) | 225 | static inline void pnp_set_drvdata(struct pnp_dev *pdev, void *data) |
229 | { | 226 | { |
230 | dev_set_drvdata(&pdev->dev, data); | 227 | dev_set_drvdata(&pdev->dev, data); |
231 | } | 228 | } |
232 | 229 | ||
233 | struct pnp_fixup { | 230 | struct pnp_fixup { |
234 | char id[7]; | 231 | char id[7]; |
235 | void (*quirk_function)(struct pnp_dev *dev); /* fixup function */ | 232 | void (*quirk_function) (struct pnp_dev * dev); /* fixup function */ |
236 | }; | 233 | }; |
237 | 234 | ||
238 | /* config parameters */ | 235 | /* config parameters */ |
@@ -269,7 +266,6 @@ extern struct pnp_protocol pnpbios_protocol; | |||
269 | #define pnp_device_is_pnpbios(dev) 0 | 266 | #define pnp_device_is_pnpbios(dev) 0 |
270 | #endif | 267 | #endif |
271 | 268 | ||
272 | |||
273 | /* status */ | 269 | /* status */ |
274 | #define PNP_READY 0x0000 | 270 | #define PNP_READY 0x0000 |
275 | #define PNP_ATTACHED 0x0001 | 271 | #define PNP_ATTACHED 0x0001 |
@@ -287,17 +283,17 @@ extern struct pnp_protocol pnpbios_protocol; | |||
287 | 283 | ||
288 | struct pnp_id { | 284 | struct pnp_id { |
289 | char id[PNP_ID_LEN]; | 285 | char id[PNP_ID_LEN]; |
290 | struct pnp_id * next; | 286 | struct pnp_id *next; |
291 | }; | 287 | }; |
292 | 288 | ||
293 | struct pnp_driver { | 289 | struct pnp_driver { |
294 | char * name; | 290 | char *name; |
295 | const struct pnp_device_id *id_table; | 291 | const struct pnp_device_id *id_table; |
296 | unsigned int flags; | 292 | unsigned int flags; |
297 | int (*probe) (struct pnp_dev *dev, const struct pnp_device_id *dev_id); | 293 | int (*probe) (struct pnp_dev *dev, const struct pnp_device_id *dev_id); |
298 | void (*remove) (struct pnp_dev *dev); | 294 | void (*remove) (struct pnp_dev *dev); |
299 | int (*suspend) (struct pnp_dev *dev, pm_message_t state); | 295 | int (*suspend) (struct pnp_dev *dev, pm_message_t state); |
300 | int (*resume) (struct pnp_dev *dev); | 296 | int (*resume) (struct pnp_dev *dev); |
301 | struct device_driver driver; | 297 | struct device_driver driver; |
302 | }; | 298 | }; |
303 | 299 | ||
@@ -305,13 +301,14 @@ struct pnp_driver { | |||
305 | 301 | ||
306 | struct pnp_card_driver { | 302 | struct pnp_card_driver { |
307 | struct list_head global_list; | 303 | struct list_head global_list; |
308 | char * name; | 304 | char *name; |
309 | const struct pnp_card_device_id *id_table; | 305 | const struct pnp_card_device_id *id_table; |
310 | unsigned int flags; | 306 | unsigned int flags; |
311 | int (*probe) (struct pnp_card_link *card, const struct pnp_card_device_id *card_id); | 307 | int (*probe) (struct pnp_card_link *card, |
308 | const struct pnp_card_device_id *card_id); | ||
312 | void (*remove) (struct pnp_card_link *card); | 309 | void (*remove) (struct pnp_card_link *card); |
313 | int (*suspend) (struct pnp_card_link *card, pm_message_t state); | 310 | int (*suspend) (struct pnp_card_link *card, pm_message_t state); |
314 | int (*resume) (struct pnp_card_link *card); | 311 | int (*resume) (struct pnp_card_link *card); |
315 | struct pnp_driver link; | 312 | struct pnp_driver link; |
316 | }; | 313 | }; |
317 | 314 | ||
@@ -321,25 +318,28 @@ struct pnp_card_driver { | |||
321 | #define PNP_DRIVER_RES_DO_NOT_CHANGE 0x0001 /* do not change the state of the device */ | 318 | #define PNP_DRIVER_RES_DO_NOT_CHANGE 0x0001 /* do not change the state of the device */ |
322 | #define PNP_DRIVER_RES_DISABLE 0x0003 /* ensure the device is disabled */ | 319 | #define PNP_DRIVER_RES_DISABLE 0x0003 /* ensure the device is disabled */ |
323 | 320 | ||
324 | |||
325 | /* | 321 | /* |
326 | * Protocol Management | 322 | * Protocol Management |
327 | */ | 323 | */ |
328 | 324 | ||
329 | struct pnp_protocol { | 325 | struct pnp_protocol { |
330 | struct list_head protocol_list; | 326 | struct list_head protocol_list; |
331 | char * name; | 327 | char *name; |
332 | 328 | ||
333 | /* resource control functions */ | 329 | /* resource control functions */ |
334 | int (*get)(struct pnp_dev *dev, struct pnp_resource_table *res); | 330 | int (*get) (struct pnp_dev *dev, struct pnp_resource_table *res); |
335 | int (*set)(struct pnp_dev *dev, struct pnp_resource_table *res); | 331 | int (*set) (struct pnp_dev *dev, struct pnp_resource_table *res); |
336 | int (*disable)(struct pnp_dev *dev); | 332 | int (*disable) (struct pnp_dev *dev); |
333 | |||
334 | /* protocol specific suspend/resume */ | ||
335 | int (*suspend) (struct pnp_dev * dev, pm_message_t state); | ||
336 | int (*resume) (struct pnp_dev * dev); | ||
337 | 337 | ||
338 | /* used by pnp layer only (look but don't touch) */ | 338 | /* used by pnp layer only (look but don't touch) */ |
339 | unsigned char number; /* protocol number*/ | 339 | unsigned char number; /* protocol number */ |
340 | struct device dev; /* link to driver model */ | 340 | struct device dev; /* link to driver model */ |
341 | struct list_head cards; | 341 | struct list_head cards; |
342 | struct list_head devices; | 342 | struct list_head devices; |
343 | }; | 343 | }; |
344 | 344 | ||
345 | #define to_pnp_protocol(n) list_entry(n, struct pnp_protocol, protocol_list) | 345 | #define to_pnp_protocol(n) list_entry(n, struct pnp_protocol, protocol_list) |
@@ -352,7 +352,6 @@ struct pnp_protocol { | |||
352 | (dev) != protocol_to_pnp_dev(&(protocol)->devices); \ | 352 | (dev) != protocol_to_pnp_dev(&(protocol)->devices); \ |
353 | (dev) = protocol_to_pnp_dev((dev)->protocol_list.next)) | 353 | (dev) = protocol_to_pnp_dev((dev)->protocol_list.next)) |
354 | 354 | ||
355 | |||
356 | extern struct bus_type pnp_bus_type; | 355 | extern struct bus_type pnp_bus_type; |
357 | 356 | ||
358 | #if defined(CONFIG_PNP) | 357 | #if defined(CONFIG_PNP) |
@@ -372,21 +371,25 @@ void pnp_remove_card(struct pnp_card *card); | |||
372 | int pnp_add_card_device(struct pnp_card *card, struct pnp_dev *dev); | 371 | int pnp_add_card_device(struct pnp_card *card, struct pnp_dev *dev); |
373 | void pnp_remove_card_device(struct pnp_dev *dev); | 372 | void pnp_remove_card_device(struct pnp_dev *dev); |
374 | int pnp_add_card_id(struct pnp_id *id, struct pnp_card *card); | 373 | int pnp_add_card_id(struct pnp_id *id, struct pnp_card *card); |
375 | struct pnp_dev * pnp_request_card_device(struct pnp_card_link *clink, const char * id, struct pnp_dev * from); | 374 | struct pnp_dev *pnp_request_card_device(struct pnp_card_link *clink, |
376 | void pnp_release_card_device(struct pnp_dev * dev); | 375 | const char *id, struct pnp_dev *from); |
377 | int pnp_register_card_driver(struct pnp_card_driver * drv); | 376 | void pnp_release_card_device(struct pnp_dev *dev); |
378 | void pnp_unregister_card_driver(struct pnp_card_driver * drv); | 377 | int pnp_register_card_driver(struct pnp_card_driver *drv); |
378 | void pnp_unregister_card_driver(struct pnp_card_driver *drv); | ||
379 | extern struct list_head pnp_cards; | 379 | extern struct list_head pnp_cards; |
380 | 380 | ||
381 | /* resource management */ | 381 | /* resource management */ |
382 | struct pnp_option * pnp_register_independent_option(struct pnp_dev *dev); | 382 | struct pnp_option *pnp_register_independent_option(struct pnp_dev *dev); |
383 | struct pnp_option * pnp_register_dependent_option(struct pnp_dev *dev, int priority); | 383 | struct pnp_option *pnp_register_dependent_option(struct pnp_dev *dev, |
384 | int priority); | ||
384 | int pnp_register_irq_resource(struct pnp_option *option, struct pnp_irq *data); | 385 | int pnp_register_irq_resource(struct pnp_option *option, struct pnp_irq *data); |
385 | int pnp_register_dma_resource(struct pnp_option *option, struct pnp_dma *data); | 386 | int pnp_register_dma_resource(struct pnp_option *option, struct pnp_dma *data); |
386 | int pnp_register_port_resource(struct pnp_option *option, struct pnp_port *data); | 387 | int pnp_register_port_resource(struct pnp_option *option, |
388 | struct pnp_port *data); | ||
387 | int pnp_register_mem_resource(struct pnp_option *option, struct pnp_mem *data); | 389 | int pnp_register_mem_resource(struct pnp_option *option, struct pnp_mem *data); |
388 | void pnp_init_resource_table(struct pnp_resource_table *table); | 390 | void pnp_init_resource_table(struct pnp_resource_table *table); |
389 | int pnp_manual_config_dev(struct pnp_dev *dev, struct pnp_resource_table *res, int mode); | 391 | int pnp_manual_config_dev(struct pnp_dev *dev, struct pnp_resource_table *res, |
392 | int mode); | ||
390 | int pnp_auto_config_dev(struct pnp_dev *dev); | 393 | int pnp_auto_config_dev(struct pnp_dev *dev); |
391 | int pnp_validate_config(struct pnp_dev *dev); | 394 | int pnp_validate_config(struct pnp_dev *dev); |
392 | int pnp_start_dev(struct pnp_dev *dev); | 395 | int pnp_start_dev(struct pnp_dev *dev); |
@@ -394,11 +397,11 @@ int pnp_stop_dev(struct pnp_dev *dev); | |||
394 | int pnp_activate_dev(struct pnp_dev *dev); | 397 | int pnp_activate_dev(struct pnp_dev *dev); |
395 | int pnp_disable_dev(struct pnp_dev *dev); | 398 | int pnp_disable_dev(struct pnp_dev *dev); |
396 | void pnp_resource_change(struct resource *resource, resource_size_t start, | 399 | void pnp_resource_change(struct resource *resource, resource_size_t start, |
397 | resource_size_t size); | 400 | resource_size_t size); |
398 | 401 | ||
399 | /* protocol helpers */ | 402 | /* protocol helpers */ |
400 | int pnp_is_active(struct pnp_dev * dev); | 403 | int pnp_is_active(struct pnp_dev *dev); |
401 | int compare_pnp_id(struct pnp_id * pos, const char * id); | 404 | int compare_pnp_id(struct pnp_id *pos, const char *id); |
402 | int pnp_add_id(struct pnp_id *id, struct pnp_dev *dev); | 405 | int pnp_add_id(struct pnp_id *id, struct pnp_dev *dev); |
403 | int pnp_register_driver(struct pnp_driver *drv); | 406 | int pnp_register_driver(struct pnp_driver *drv); |
404 | void pnp_unregister_driver(struct pnp_driver *drv); | 407 | void pnp_unregister_driver(struct pnp_driver *drv); |
@@ -411,23 +414,24 @@ static inline void pnp_unregister_protocol(struct pnp_protocol *protocol) { } | |||
411 | static inline int pnp_init_device(struct pnp_dev *dev) { return -ENODEV; } | 414 | static inline int pnp_init_device(struct pnp_dev *dev) { return -ENODEV; } |
412 | static inline int pnp_add_device(struct pnp_dev *dev) { return -ENODEV; } | 415 | static inline int pnp_add_device(struct pnp_dev *dev) { return -ENODEV; } |
413 | static inline int pnp_device_attach(struct pnp_dev *pnp_dev) { return -ENODEV; } | 416 | static inline int pnp_device_attach(struct pnp_dev *pnp_dev) { return -ENODEV; } |
414 | static inline void pnp_device_detach(struct pnp_dev *pnp_dev) { ; } | 417 | static inline void pnp_device_detach(struct pnp_dev *pnp_dev) { } |
418 | |||
415 | #define pnp_platform_devices 0 | 419 | #define pnp_platform_devices 0 |
416 | 420 | ||
417 | /* multidevice card support */ | 421 | /* multidevice card support */ |
418 | static inline int pnp_add_card(struct pnp_card *card) { return -ENODEV; } | 422 | static inline int pnp_add_card(struct pnp_card *card) { return -ENODEV; } |
419 | static inline void pnp_remove_card(struct pnp_card *card) { ; } | 423 | static inline void pnp_remove_card(struct pnp_card *card) { } |
420 | static inline int pnp_add_card_device(struct pnp_card *card, struct pnp_dev *dev) { return -ENODEV; } | 424 | static inline int pnp_add_card_device(struct pnp_card *card, struct pnp_dev *dev) { return -ENODEV; } |
421 | static inline void pnp_remove_card_device(struct pnp_dev *dev) { ; } | 425 | static inline void pnp_remove_card_device(struct pnp_dev *dev) { } |
422 | static inline int pnp_add_card_id(struct pnp_id *id, struct pnp_card *card) { return -ENODEV; } | 426 | static inline int pnp_add_card_id(struct pnp_id *id, struct pnp_card *card) { return -ENODEV; } |
423 | static inline struct pnp_dev * pnp_request_card_device(struct pnp_card_link *clink, const char * id, struct pnp_dev * from) { return NULL; } | 427 | static inline struct pnp_dev *pnp_request_card_device(struct pnp_card_link *clink, const char *id, struct pnp_dev *from) { return NULL; } |
424 | static inline void pnp_release_card_device(struct pnp_dev * dev) { ; } | 428 | static inline void pnp_release_card_device(struct pnp_dev *dev) { } |
425 | static inline int pnp_register_card_driver(struct pnp_card_driver * drv) { return -ENODEV; } | 429 | static inline int pnp_register_card_driver(struct pnp_card_driver *drv) { return -ENODEV; } |
426 | static inline void pnp_unregister_card_driver(struct pnp_card_driver * drv) { ; } | 430 | static inline void pnp_unregister_card_driver(struct pnp_card_driver *drv) { } |
427 | 431 | ||
428 | /* resource management */ | 432 | /* resource management */ |
429 | static inline struct pnp_option * pnp_register_independent_option(struct pnp_dev *dev) { return NULL; } | 433 | static inline struct pnp_option *pnp_register_independent_option(struct pnp_dev *dev) { return NULL; } |
430 | static inline struct pnp_option * pnp_register_dependent_option(struct pnp_dev *dev, int priority) { return NULL; } | 434 | static inline struct pnp_option *pnp_register_dependent_option(struct pnp_dev *dev, int priority) { return NULL; } |
431 | static inline int pnp_register_irq_resource(struct pnp_option *option, struct pnp_irq *data) { return -ENODEV; } | 435 | static inline int pnp_register_irq_resource(struct pnp_option *option, struct pnp_irq *data) { return -ENODEV; } |
432 | static inline int pnp_register_dma_resource(struct pnp_option *option, struct pnp_dma *data) { return -ENODEV; } | 436 | static inline int pnp_register_dma_resource(struct pnp_option *option, struct pnp_dma *data) { return -ENODEV; } |
433 | static inline int pnp_register_port_resource(struct pnp_option *option, struct pnp_port *data) { return -ENODEV; } | 437 | static inline int pnp_register_port_resource(struct pnp_option *option, struct pnp_port *data) { return -ENODEV; } |
@@ -440,20 +444,17 @@ static inline int pnp_start_dev(struct pnp_dev *dev) { return -ENODEV; } | |||
440 | static inline int pnp_stop_dev(struct pnp_dev *dev) { return -ENODEV; } | 444 | static inline int pnp_stop_dev(struct pnp_dev *dev) { return -ENODEV; } |
441 | static inline int pnp_activate_dev(struct pnp_dev *dev) { return -ENODEV; } | 445 | static inline int pnp_activate_dev(struct pnp_dev *dev) { return -ENODEV; } |
442 | static inline int pnp_disable_dev(struct pnp_dev *dev) { return -ENODEV; } | 446 | static inline int pnp_disable_dev(struct pnp_dev *dev) { return -ENODEV; } |
443 | static inline void pnp_resource_change(struct resource *resource, | 447 | static inline void pnp_resource_change(struct resource *resource, resource_size_t start, resource_size_t size) { } |
444 | resource_size_t start, | ||
445 | resource_size_t size) { } | ||
446 | 448 | ||
447 | /* protocol helpers */ | 449 | /* protocol helpers */ |
448 | static inline int pnp_is_active(struct pnp_dev * dev) { return 0; } | 450 | static inline int pnp_is_active(struct pnp_dev *dev) { return 0; } |
449 | static inline int compare_pnp_id(struct pnp_id * pos, const char * id) { return -ENODEV; } | 451 | static inline int compare_pnp_id(struct pnp_id *pos, const char *id) { return -ENODEV; } |
450 | static inline int pnp_add_id(struct pnp_id *id, struct pnp_dev *dev) { return -ENODEV; } | 452 | static inline int pnp_add_id(struct pnp_id *id, struct pnp_dev *dev) { return -ENODEV; } |
451 | static inline int pnp_register_driver(struct pnp_driver *drv) { return -ENODEV; } | 453 | static inline int pnp_register_driver(struct pnp_driver *drv) { return -ENODEV; } |
452 | static inline void pnp_unregister_driver(struct pnp_driver *drv) { ; } | 454 | static inline void pnp_unregister_driver(struct pnp_driver *drv) { } |
453 | 455 | ||
454 | #endif /* CONFIG_PNP */ | 456 | #endif /* CONFIG_PNP */ |
455 | 457 | ||
456 | |||
457 | #define pnp_err(format, arg...) printk(KERN_ERR "pnp: " format "\n" , ## arg) | 458 | #define pnp_err(format, arg...) printk(KERN_ERR "pnp: " format "\n" , ## arg) |
458 | #define pnp_info(format, arg...) printk(KERN_INFO "pnp: " format "\n" , ## arg) | 459 | #define pnp_info(format, arg...) printk(KERN_INFO "pnp: " format "\n" , ## arg) |
459 | #define pnp_warn(format, arg...) printk(KERN_WARNING "pnp: " format "\n" , ## arg) | 460 | #define pnp_warn(format, arg...) printk(KERN_WARNING "pnp: " format "\n" , ## arg) |
diff --git a/include/linux/pnpbios.h b/include/linux/pnpbios.h index 0a282ac1f6b2..329192adc9dd 100644 --- a/include/linux/pnpbios.h +++ b/include/linux/pnpbios.h | |||
@@ -99,32 +99,32 @@ | |||
99 | 99 | ||
100 | #pragma pack(1) | 100 | #pragma pack(1) |
101 | struct pnp_dev_node_info { | 101 | struct pnp_dev_node_info { |
102 | __u16 no_nodes; | 102 | __u16 no_nodes; |
103 | __u16 max_node_size; | 103 | __u16 max_node_size; |
104 | }; | 104 | }; |
105 | struct pnp_docking_station_info { | 105 | struct pnp_docking_station_info { |
106 | __u32 location_id; | 106 | __u32 location_id; |
107 | __u32 serial; | 107 | __u32 serial; |
108 | __u16 capabilities; | 108 | __u16 capabilities; |
109 | }; | 109 | }; |
110 | struct pnp_isa_config_struc { | 110 | struct pnp_isa_config_struc { |
111 | __u8 revision; | 111 | __u8 revision; |
112 | __u8 no_csns; | 112 | __u8 no_csns; |
113 | __u16 isa_rd_data_port; | 113 | __u16 isa_rd_data_port; |
114 | __u16 reserved; | 114 | __u16 reserved; |
115 | }; | 115 | }; |
116 | struct escd_info_struc { | 116 | struct escd_info_struc { |
117 | __u16 min_escd_write_size; | 117 | __u16 min_escd_write_size; |
118 | __u16 escd_size; | 118 | __u16 escd_size; |
119 | __u32 nv_storage_base; | 119 | __u32 nv_storage_base; |
120 | }; | 120 | }; |
121 | struct pnp_bios_node { | 121 | struct pnp_bios_node { |
122 | __u16 size; | 122 | __u16 size; |
123 | __u8 handle; | 123 | __u8 handle; |
124 | __u32 eisa_id; | 124 | __u32 eisa_id; |
125 | __u8 type_code[3]; | 125 | __u8 type_code[3]; |
126 | __u16 flags; | 126 | __u16 flags; |
127 | __u8 data[0]; | 127 | __u8 data[0]; |
128 | }; | 128 | }; |
129 | #pragma pack() | 129 | #pragma pack() |
130 | 130 | ||
@@ -133,22 +133,16 @@ struct pnp_bios_node { | |||
133 | /* non-exported */ | 133 | /* non-exported */ |
134 | extern struct pnp_dev_node_info node_info; | 134 | extern struct pnp_dev_node_info node_info; |
135 | 135 | ||
136 | extern int pnp_bios_dev_node_info (struct pnp_dev_node_info *data); | 136 | extern int pnp_bios_dev_node_info(struct pnp_dev_node_info *data); |
137 | extern int pnp_bios_get_dev_node (u8 *nodenum, char config, struct pnp_bios_node *data); | 137 | extern int pnp_bios_get_dev_node(u8 *nodenum, char config, |
138 | extern int pnp_bios_set_dev_node (u8 nodenum, char config, struct pnp_bios_node *data); | 138 | struct pnp_bios_node *data); |
139 | extern int pnp_bios_get_stat_res (char *info); | 139 | extern int pnp_bios_set_dev_node(u8 nodenum, char config, |
140 | extern int pnp_bios_isapnp_config (struct pnp_isa_config_struc *data); | 140 | struct pnp_bios_node *data); |
141 | extern int pnp_bios_escd_info (struct escd_info_struc *data); | 141 | extern int pnp_bios_get_stat_res(char *info); |
142 | extern int pnp_bios_read_escd (char *data, u32 nvram_base); | 142 | extern int pnp_bios_isapnp_config(struct pnp_isa_config_struc *data); |
143 | extern int pnp_bios_escd_info(struct escd_info_struc *data); | ||
144 | extern int pnp_bios_read_escd(char *data, u32 nvram_base); | ||
143 | extern int pnp_bios_dock_station_info(struct pnp_docking_station_info *data); | 145 | extern int pnp_bios_dock_station_info(struct pnp_docking_station_info *data); |
144 | #define needed 0 | ||
145 | #if needed | ||
146 | extern int pnp_bios_get_event (u16 *message); | ||
147 | extern int pnp_bios_send_message (u16 message); | ||
148 | extern int pnp_bios_set_stat_res (char *info); | ||
149 | extern int pnp_bios_apm_id_table (char *table, u16 *size); | ||
150 | extern int pnp_bios_write_escd (char *data, u32 nvram_base); | ||
151 | #endif | ||
152 | 146 | ||
153 | #endif /* CONFIG_PNPBIOS */ | 147 | #endif /* CONFIG_PNPBIOS */ |
154 | 148 | ||
diff --git a/include/linux/preempt.h b/include/linux/preempt.h index d0926d63406c..484988ed301e 100644 --- a/include/linux/preempt.h +++ b/include/linux/preempt.h | |||
@@ -8,6 +8,7 @@ | |||
8 | 8 | ||
9 | #include <linux/thread_info.h> | 9 | #include <linux/thread_info.h> |
10 | #include <linux/linkage.h> | 10 | #include <linux/linkage.h> |
11 | #include <linux/list.h> | ||
11 | 12 | ||
12 | #ifdef CONFIG_DEBUG_PREEMPT | 13 | #ifdef CONFIG_DEBUG_PREEMPT |
13 | extern void fastcall add_preempt_count(int val); | 14 | extern void fastcall add_preempt_count(int val); |
@@ -60,4 +61,47 @@ do { \ | |||
60 | 61 | ||
61 | #endif | 62 | #endif |
62 | 63 | ||
64 | #ifdef CONFIG_PREEMPT_NOTIFIERS | ||
65 | |||
66 | struct preempt_notifier; | ||
67 | |||
68 | /** | ||
69 | * preempt_ops - notifiers called when a task is preempted and rescheduled | ||
70 | * @sched_in: we're about to be rescheduled: | ||
71 | * notifier: struct preempt_notifier for the task being scheduled | ||
72 | * cpu: cpu we're scheduled on | ||
73 | * @sched_out: we've just been preempted | ||
74 | * notifier: struct preempt_notifier for the task being preempted | ||
75 | * next: the task that's kicking us out | ||
76 | */ | ||
77 | struct preempt_ops { | ||
78 | void (*sched_in)(struct preempt_notifier *notifier, int cpu); | ||
79 | void (*sched_out)(struct preempt_notifier *notifier, | ||
80 | struct task_struct *next); | ||
81 | }; | ||
82 | |||
83 | /** | ||
84 | * preempt_notifier - key for installing preemption notifiers | ||
85 | * @link: internal use | ||
86 | * @ops: defines the notifier functions to be called | ||
87 | * | ||
88 | * Usually used in conjunction with container_of(). | ||
89 | */ | ||
90 | struct preempt_notifier { | ||
91 | struct hlist_node link; | ||
92 | struct preempt_ops *ops; | ||
93 | }; | ||
94 | |||
95 | void preempt_notifier_register(struct preempt_notifier *notifier); | ||
96 | void preempt_notifier_unregister(struct preempt_notifier *notifier); | ||
97 | |||
98 | static inline void preempt_notifier_init(struct preempt_notifier *notifier, | ||
99 | struct preempt_ops *ops) | ||
100 | { | ||
101 | INIT_HLIST_NODE(¬ifier->link); | ||
102 | notifier->ops = ops; | ||
103 | } | ||
104 | |||
105 | #endif | ||
106 | |||
63 | #endif /* __LINUX_PREEMPT_H */ | 107 | #endif /* __LINUX_PREEMPT_H */ |
diff --git a/include/linux/raid/md_k.h b/include/linux/raid/md_k.h index 28ac632b42dd..dcb729244f47 100644 --- a/include/linux/raid/md_k.h +++ b/include/linux/raid/md_k.h | |||
@@ -227,7 +227,7 @@ struct mddev_s | |||
227 | unsigned int safemode_delay; | 227 | unsigned int safemode_delay; |
228 | struct timer_list safemode_timer; | 228 | struct timer_list safemode_timer; |
229 | atomic_t writes_pending; | 229 | atomic_t writes_pending; |
230 | request_queue_t *queue; /* for plugging ... */ | 230 | struct request_queue *queue; /* for plugging ... */ |
231 | 231 | ||
232 | atomic_t write_behind; /* outstanding async IO */ | 232 | atomic_t write_behind; /* outstanding async IO */ |
233 | unsigned int max_write_behind; /* 0 = sync */ | 233 | unsigned int max_write_behind; /* 0 = sync */ |
@@ -265,7 +265,7 @@ struct mdk_personality | |||
265 | int level; | 265 | int level; |
266 | struct list_head list; | 266 | struct list_head list; |
267 | struct module *owner; | 267 | struct module *owner; |
268 | int (*make_request)(request_queue_t *q, struct bio *bio); | 268 | int (*make_request)(struct request_queue *q, struct bio *bio); |
269 | int (*run)(mddev_t *mddev); | 269 | int (*run)(mddev_t *mddev); |
270 | int (*stop)(mddev_t *mddev); | 270 | int (*stop)(mddev_t *mddev); |
271 | void (*status)(struct seq_file *seq, mddev_t *mddev); | 271 | void (*status)(struct seq_file *seq, mddev_t *mddev); |
diff --git a/include/linux/sched.h b/include/linux/sched.h index 33b9b4841ee7..2e490271acf6 100644 --- a/include/linux/sched.h +++ b/include/linux/sched.h | |||
@@ -681,7 +681,7 @@ enum cpu_idle_type { | |||
681 | #define SCHED_LOAD_SHIFT 10 | 681 | #define SCHED_LOAD_SHIFT 10 |
682 | #define SCHED_LOAD_SCALE (1L << SCHED_LOAD_SHIFT) | 682 | #define SCHED_LOAD_SCALE (1L << SCHED_LOAD_SHIFT) |
683 | 683 | ||
684 | #define SCHED_LOAD_SCALE_FUZZ (SCHED_LOAD_SCALE >> 5) | 684 | #define SCHED_LOAD_SCALE_FUZZ (SCHED_LOAD_SCALE >> 1) |
685 | 685 | ||
686 | #ifdef CONFIG_SMP | 686 | #ifdef CONFIG_SMP |
687 | #define SD_LOAD_BALANCE 1 /* Do load balancing on this domain. */ | 687 | #define SD_LOAD_BALANCE 1 /* Do load balancing on this domain. */ |
@@ -786,6 +786,22 @@ extern int partition_sched_domains(cpumask_t *partition1, | |||
786 | 786 | ||
787 | #endif /* CONFIG_SMP */ | 787 | #endif /* CONFIG_SMP */ |
788 | 788 | ||
789 | /* | ||
790 | * A runqueue laden with a single nice 0 task scores a weighted_cpuload of | ||
791 | * SCHED_LOAD_SCALE. This function returns 1 if any cpu is laden with a | ||
792 | * task of nice 0 or enough lower priority tasks to bring up the | ||
793 | * weighted_cpuload | ||
794 | */ | ||
795 | static inline int above_background_load(void) | ||
796 | { | ||
797 | unsigned long cpu; | ||
798 | |||
799 | for_each_online_cpu(cpu) { | ||
800 | if (weighted_cpuload(cpu) >= SCHED_LOAD_SCALE) | ||
801 | return 1; | ||
802 | } | ||
803 | return 0; | ||
804 | } | ||
789 | 805 | ||
790 | struct io_context; /* See blkdev.h */ | 806 | struct io_context; /* See blkdev.h */ |
791 | struct cpuset; | 807 | struct cpuset; |
@@ -935,6 +951,11 @@ struct task_struct { | |||
935 | struct sched_class *sched_class; | 951 | struct sched_class *sched_class; |
936 | struct sched_entity se; | 952 | struct sched_entity se; |
937 | 953 | ||
954 | #ifdef CONFIG_PREEMPT_NOTIFIERS | ||
955 | /* list of struct preempt_notifier: */ | ||
956 | struct hlist_head preempt_notifiers; | ||
957 | #endif | ||
958 | |||
938 | unsigned short ioprio; | 959 | unsigned short ioprio; |
939 | #ifdef CONFIG_BLK_DEV_IO_TRACE | 960 | #ifdef CONFIG_BLK_DEV_IO_TRACE |
940 | unsigned int btrace_seq; | 961 | unsigned int btrace_seq; |
diff --git a/include/linux/suspend.h b/include/linux/suspend.h index e8e6da394c92..618f93c32b7f 100644 --- a/include/linux/suspend.h +++ b/include/linux/suspend.h | |||
@@ -125,6 +125,9 @@ static inline int unregister_pm_notifier(struct notifier_block *nb) | |||
125 | static inline void register_nosave_region(unsigned long b, unsigned long e) | 125 | static inline void register_nosave_region(unsigned long b, unsigned long e) |
126 | { | 126 | { |
127 | } | 127 | } |
128 | static inline void register_nosave_region_late(unsigned long b, unsigned long e) | ||
129 | { | ||
130 | } | ||
128 | #endif | 131 | #endif |
129 | 132 | ||
130 | #endif /* _LINUX_SWSUSP_H */ | 133 | #endif /* _LINUX_SWSUSP_H */ |
diff --git a/include/linux/time.h b/include/linux/time.h index e6aea5146e5d..6a5f503b4f1d 100644 --- a/include/linux/time.h +++ b/include/linux/time.h | |||
@@ -99,15 +99,11 @@ extern int update_persistent_clock(struct timespec now); | |||
99 | extern int no_sync_cmos_clock __read_mostly; | 99 | extern int no_sync_cmos_clock __read_mostly; |
100 | void timekeeping_init(void); | 100 | void timekeeping_init(void); |
101 | 101 | ||
102 | static inline unsigned long get_seconds(void) | 102 | unsigned long get_seconds(void); |
103 | { | ||
104 | return xtime.tv_sec; | ||
105 | } | ||
106 | |||
107 | struct timespec current_kernel_time(void); | 103 | struct timespec current_kernel_time(void); |
108 | 104 | ||
109 | #define CURRENT_TIME (current_kernel_time()) | 105 | #define CURRENT_TIME (current_kernel_time()) |
110 | #define CURRENT_TIME_SEC ((struct timespec) { xtime.tv_sec, 0 }) | 106 | #define CURRENT_TIME_SEC ((struct timespec) { get_seconds(), 0 }) |
111 | 107 | ||
112 | extern void do_gettimeofday(struct timeval *tv); | 108 | extern void do_gettimeofday(struct timeval *tv); |
113 | extern int do_settimeofday(struct timespec *tv); | 109 | extern int do_settimeofday(struct timespec *tv); |
diff --git a/include/net/netfilter/nf_conntrack_tuple.h b/include/net/netfilter/nf_conntrack_tuple.h index 040dae5f0c9e..c48e390f4b0f 100644 --- a/include/net/netfilter/nf_conntrack_tuple.h +++ b/include/net/netfilter/nf_conntrack_tuple.h | |||
@@ -35,7 +35,7 @@ union nf_conntrack_address { | |||
35 | union nf_conntrack_man_proto | 35 | union nf_conntrack_man_proto |
36 | { | 36 | { |
37 | /* Add other protocols here. */ | 37 | /* Add other protocols here. */ |
38 | u_int16_t all; | 38 | __be16 all; |
39 | 39 | ||
40 | struct { | 40 | struct { |
41 | __be16 port; | 41 | __be16 port; |
@@ -73,7 +73,7 @@ struct nf_conntrack_tuple | |||
73 | union nf_conntrack_address u3; | 73 | union nf_conntrack_address u3; |
74 | union { | 74 | union { |
75 | /* Add other protocols here. */ | 75 | /* Add other protocols here. */ |
76 | u_int16_t all; | 76 | __be16 all; |
77 | 77 | ||
78 | struct { | 78 | struct { |
79 | __be16 port; | 79 | __be16 port; |
diff --git a/include/scsi/sd.h b/include/scsi/sd.h index 5261488e1108..78583fee0ab2 100644 --- a/include/scsi/sd.h +++ b/include/scsi/sd.h | |||
@@ -57,7 +57,7 @@ static int sd_resume(struct device *dev); | |||
57 | static void sd_rescan(struct device *); | 57 | static void sd_rescan(struct device *); |
58 | static int sd_init_command(struct scsi_cmnd *); | 58 | static int sd_init_command(struct scsi_cmnd *); |
59 | static int sd_issue_flush(struct device *, sector_t *); | 59 | static int sd_issue_flush(struct device *, sector_t *); |
60 | static void sd_prepare_flush(request_queue_t *, struct request *); | 60 | static void sd_prepare_flush(struct request_queue *, struct request *); |
61 | static void sd_read_capacity(struct scsi_disk *sdkp, unsigned char *buffer); | 61 | static void sd_read_capacity(struct scsi_disk *sdkp, unsigned char *buffer); |
62 | static void scsi_disk_release(struct class_device *cdev); | 62 | static void scsi_disk_release(struct class_device *cdev); |
63 | static void sd_print_sense_hdr(struct scsi_disk *, struct scsi_sense_hdr *); | 63 | static void sd_print_sense_hdr(struct scsi_disk *, struct scsi_sense_hdr *); |
diff --git a/include/xen/page.h b/include/xen/page.h index 1df6c1930578..c0c8fcb27899 100644 --- a/include/xen/page.h +++ b/include/xen/page.h | |||
@@ -4,6 +4,7 @@ | |||
4 | #include <linux/pfn.h> | 4 | #include <linux/pfn.h> |
5 | 5 | ||
6 | #include <asm/uaccess.h> | 6 | #include <asm/uaccess.h> |
7 | #include <asm/pgtable.h> | ||
7 | 8 | ||
8 | #include <xen/features.h> | 9 | #include <xen/features.h> |
9 | 10 | ||
diff --git a/init/initramfs.c b/init/initramfs.c index 00eff7a11085..1db02a0025db 100644 --- a/init/initramfs.c +++ b/init/initramfs.c | |||
@@ -133,7 +133,7 @@ static __initdata loff_t this_header, next_header; | |||
133 | 133 | ||
134 | static __initdata int dry_run; | 134 | static __initdata int dry_run; |
135 | 135 | ||
136 | static inline void eat(unsigned n) | 136 | static inline void __init eat(unsigned n) |
137 | { | 137 | { |
138 | victim += n; | 138 | victim += n; |
139 | this_header += n; | 139 | this_header += n; |
diff --git a/kernel/Kconfig.preempt b/kernel/Kconfig.preempt index c64ce9c14207..6b066632e40c 100644 --- a/kernel/Kconfig.preempt +++ b/kernel/Kconfig.preempt | |||
@@ -63,3 +63,6 @@ config PREEMPT_BKL | |||
63 | Say Y here if you are building a kernel for a desktop system. | 63 | Say Y here if you are building a kernel for a desktop system. |
64 | Say N if you are unsure. | 64 | Say N if you are unsure. |
65 | 65 | ||
66 | config PREEMPT_NOTIFIERS | ||
67 | bool | ||
68 | |||
diff --git a/kernel/acct.c b/kernel/acct.c index 70d0d88e5554..24f0f8b2ba72 100644 --- a/kernel/acct.c +++ b/kernel/acct.c | |||
@@ -468,7 +468,7 @@ static void do_acct_process(struct file *file) | |||
468 | } | 468 | } |
469 | #endif | 469 | #endif |
470 | do_div(elapsed, AHZ); | 470 | do_div(elapsed, AHZ); |
471 | ac.ac_btime = xtime.tv_sec - elapsed; | 471 | ac.ac_btime = get_seconds() - elapsed; |
472 | /* we really need to bite the bullet and change layout */ | 472 | /* we really need to bite the bullet and change layout */ |
473 | ac.ac_uid = current->uid; | 473 | ac.ac_uid = current->uid; |
474 | ac.ac_gid = current->gid; | 474 | ac.ac_gid = current->gid; |
diff --git a/kernel/hrtimer.c b/kernel/hrtimer.c index eb1ddebd2c04..c21ca6bfaa66 100644 --- a/kernel/hrtimer.c +++ b/kernel/hrtimer.c | |||
@@ -141,11 +141,7 @@ static void hrtimer_get_softirq_time(struct hrtimer_cpu_base *base) | |||
141 | 141 | ||
142 | do { | 142 | do { |
143 | seq = read_seqbegin(&xtime_lock); | 143 | seq = read_seqbegin(&xtime_lock); |
144 | #ifdef CONFIG_NO_HZ | 144 | xts = current_kernel_time(); |
145 | getnstimeofday(&xts); | ||
146 | #else | ||
147 | xts = xtime; | ||
148 | #endif | ||
149 | tom = wall_to_monotonic; | 145 | tom = wall_to_monotonic; |
150 | } while (read_seqretry(&xtime_lock, seq)); | 146 | } while (read_seqretry(&xtime_lock, seq)); |
151 | 147 | ||
diff --git a/kernel/kmod.c b/kernel/kmod.c index beedbdc64608..9809cc1f33d6 100644 --- a/kernel/kmod.c +++ b/kernel/kmod.c | |||
@@ -351,11 +351,11 @@ static inline void register_pm_notifier_callback(void) {} | |||
351 | 351 | ||
352 | /** | 352 | /** |
353 | * call_usermodehelper_setup - prepare to call a usermode helper | 353 | * call_usermodehelper_setup - prepare to call a usermode helper |
354 | * @path - path to usermode executable | 354 | * @path: path to usermode executable |
355 | * @argv - arg vector for process | 355 | * @argv: arg vector for process |
356 | * @envp - environment for process | 356 | * @envp: environment for process |
357 | * | 357 | * |
358 | * Returns either NULL on allocation failure, or a subprocess_info | 358 | * Returns either %NULL on allocation failure, or a subprocess_info |
359 | * structure. This should be passed to call_usermodehelper_exec to | 359 | * structure. This should be passed to call_usermodehelper_exec to |
360 | * exec the process and free the structure. | 360 | * exec the process and free the structure. |
361 | */ | 361 | */ |
diff --git a/kernel/power/disk.c b/kernel/power/disk.c index 324ac0188ce1..eb72255b5c86 100644 --- a/kernel/power/disk.c +++ b/kernel/power/disk.c | |||
@@ -216,6 +216,7 @@ int hibernation_platform_enter(void) | |||
216 | * sleep state after all | 216 | * sleep state after all |
217 | */ | 217 | */ |
218 | error = hibernation_ops->prepare(); | 218 | error = hibernation_ops->prepare(); |
219 | sysdev_shutdown(); | ||
219 | if (!error) | 220 | if (!error) |
220 | error = hibernation_ops->enter(); | 221 | error = hibernation_ops->enter(); |
221 | } else { | 222 | } else { |
diff --git a/kernel/sched.c b/kernel/sched.c index 93cf241cfbe9..5c51d7e5dcc1 100644 --- a/kernel/sched.c +++ b/kernel/sched.c | |||
@@ -53,6 +53,7 @@ | |||
53 | #include <linux/percpu.h> | 53 | #include <linux/percpu.h> |
54 | #include <linux/kthread.h> | 54 | #include <linux/kthread.h> |
55 | #include <linux/seq_file.h> | 55 | #include <linux/seq_file.h> |
56 | #include <linux/sysctl.h> | ||
56 | #include <linux/syscalls.h> | 57 | #include <linux/syscalls.h> |
57 | #include <linux/times.h> | 58 | #include <linux/times.h> |
58 | #include <linux/tsacct_kern.h> | 59 | #include <linux/tsacct_kern.h> |
@@ -263,8 +264,6 @@ struct rq { | |||
263 | unsigned int clock_warps, clock_overflows; | 264 | unsigned int clock_warps, clock_overflows; |
264 | unsigned int clock_unstable_events; | 265 | unsigned int clock_unstable_events; |
265 | 266 | ||
266 | struct sched_class *load_balance_class; | ||
267 | |||
268 | atomic_t nr_iowait; | 267 | atomic_t nr_iowait; |
269 | 268 | ||
270 | #ifdef CONFIG_SMP | 269 | #ifdef CONFIG_SMP |
@@ -385,13 +384,12 @@ static inline unsigned long long rq_clock(struct rq *rq) | |||
385 | */ | 384 | */ |
386 | unsigned long long cpu_clock(int cpu) | 385 | unsigned long long cpu_clock(int cpu) |
387 | { | 386 | { |
388 | struct rq *rq = cpu_rq(cpu); | ||
389 | unsigned long long now; | 387 | unsigned long long now; |
390 | unsigned long flags; | 388 | unsigned long flags; |
391 | 389 | ||
392 | spin_lock_irqsave(&rq->lock, flags); | 390 | local_irq_save(flags); |
393 | now = rq_clock(rq); | 391 | now = rq_clock(cpu_rq(cpu)); |
394 | spin_unlock_irqrestore(&rq->lock, flags); | 392 | local_irq_restore(flags); |
395 | 393 | ||
396 | return now; | 394 | return now; |
397 | } | 395 | } |
@@ -1592,6 +1590,10 @@ static void __sched_fork(struct task_struct *p) | |||
1592 | INIT_LIST_HEAD(&p->run_list); | 1590 | INIT_LIST_HEAD(&p->run_list); |
1593 | p->se.on_rq = 0; | 1591 | p->se.on_rq = 0; |
1594 | 1592 | ||
1593 | #ifdef CONFIG_PREEMPT_NOTIFIERS | ||
1594 | INIT_HLIST_HEAD(&p->preempt_notifiers); | ||
1595 | #endif | ||
1596 | |||
1595 | /* | 1597 | /* |
1596 | * We mark the process as running here, but have not actually | 1598 | * We mark the process as running here, but have not actually |
1597 | * inserted it onto the runqueue yet. This guarantees that | 1599 | * inserted it onto the runqueue yet. This guarantees that |
@@ -1673,6 +1675,63 @@ void fastcall wake_up_new_task(struct task_struct *p, unsigned long clone_flags) | |||
1673 | task_rq_unlock(rq, &flags); | 1675 | task_rq_unlock(rq, &flags); |
1674 | } | 1676 | } |
1675 | 1677 | ||
1678 | #ifdef CONFIG_PREEMPT_NOTIFIERS | ||
1679 | |||
1680 | /** | ||
1681 | * preempt_notifier_register - tell me when current is being being preempted | ||
1682 | * and rescheduled | ||
1683 | */ | ||
1684 | void preempt_notifier_register(struct preempt_notifier *notifier) | ||
1685 | { | ||
1686 | hlist_add_head(¬ifier->link, ¤t->preempt_notifiers); | ||
1687 | } | ||
1688 | EXPORT_SYMBOL_GPL(preempt_notifier_register); | ||
1689 | |||
1690 | /** | ||
1691 | * preempt_notifier_unregister - no longer interested in preemption notifications | ||
1692 | * | ||
1693 | * This is safe to call from within a preemption notifier. | ||
1694 | */ | ||
1695 | void preempt_notifier_unregister(struct preempt_notifier *notifier) | ||
1696 | { | ||
1697 | hlist_del(¬ifier->link); | ||
1698 | } | ||
1699 | EXPORT_SYMBOL_GPL(preempt_notifier_unregister); | ||
1700 | |||
1701 | static void fire_sched_in_preempt_notifiers(struct task_struct *curr) | ||
1702 | { | ||
1703 | struct preempt_notifier *notifier; | ||
1704 | struct hlist_node *node; | ||
1705 | |||
1706 | hlist_for_each_entry(notifier, node, &curr->preempt_notifiers, link) | ||
1707 | notifier->ops->sched_in(notifier, raw_smp_processor_id()); | ||
1708 | } | ||
1709 | |||
1710 | static void | ||
1711 | fire_sched_out_preempt_notifiers(struct task_struct *curr, | ||
1712 | struct task_struct *next) | ||
1713 | { | ||
1714 | struct preempt_notifier *notifier; | ||
1715 | struct hlist_node *node; | ||
1716 | |||
1717 | hlist_for_each_entry(notifier, node, &curr->preempt_notifiers, link) | ||
1718 | notifier->ops->sched_out(notifier, next); | ||
1719 | } | ||
1720 | |||
1721 | #else | ||
1722 | |||
1723 | static void fire_sched_in_preempt_notifiers(struct task_struct *curr) | ||
1724 | { | ||
1725 | } | ||
1726 | |||
1727 | static void | ||
1728 | fire_sched_out_preempt_notifiers(struct task_struct *curr, | ||
1729 | struct task_struct *next) | ||
1730 | { | ||
1731 | } | ||
1732 | |||
1733 | #endif | ||
1734 | |||
1676 | /** | 1735 | /** |
1677 | * prepare_task_switch - prepare to switch tasks | 1736 | * prepare_task_switch - prepare to switch tasks |
1678 | * @rq: the runqueue preparing to switch | 1737 | * @rq: the runqueue preparing to switch |
@@ -1685,8 +1744,11 @@ void fastcall wake_up_new_task(struct task_struct *p, unsigned long clone_flags) | |||
1685 | * prepare_task_switch sets up locking and calls architecture specific | 1744 | * prepare_task_switch sets up locking and calls architecture specific |
1686 | * hooks. | 1745 | * hooks. |
1687 | */ | 1746 | */ |
1688 | static inline void prepare_task_switch(struct rq *rq, struct task_struct *next) | 1747 | static inline void |
1748 | prepare_task_switch(struct rq *rq, struct task_struct *prev, | ||
1749 | struct task_struct *next) | ||
1689 | { | 1750 | { |
1751 | fire_sched_out_preempt_notifiers(prev, next); | ||
1690 | prepare_lock_switch(rq, next); | 1752 | prepare_lock_switch(rq, next); |
1691 | prepare_arch_switch(next); | 1753 | prepare_arch_switch(next); |
1692 | } | 1754 | } |
@@ -1728,6 +1790,7 @@ static inline void finish_task_switch(struct rq *rq, struct task_struct *prev) | |||
1728 | prev_state = prev->state; | 1790 | prev_state = prev->state; |
1729 | finish_arch_switch(prev); | 1791 | finish_arch_switch(prev); |
1730 | finish_lock_switch(rq, prev); | 1792 | finish_lock_switch(rq, prev); |
1793 | fire_sched_in_preempt_notifiers(current); | ||
1731 | if (mm) | 1794 | if (mm) |
1732 | mmdrop(mm); | 1795 | mmdrop(mm); |
1733 | if (unlikely(prev_state == TASK_DEAD)) { | 1796 | if (unlikely(prev_state == TASK_DEAD)) { |
@@ -1768,7 +1831,7 @@ context_switch(struct rq *rq, struct task_struct *prev, | |||
1768 | { | 1831 | { |
1769 | struct mm_struct *mm, *oldmm; | 1832 | struct mm_struct *mm, *oldmm; |
1770 | 1833 | ||
1771 | prepare_task_switch(rq, next); | 1834 | prepare_task_switch(rq, prev, next); |
1772 | mm = next->mm; | 1835 | mm = next->mm; |
1773 | oldmm = prev->active_mm; | 1836 | oldmm = prev->active_mm; |
1774 | /* | 1837 | /* |
@@ -5140,10 +5203,129 @@ static void migrate_dead_tasks(unsigned int dead_cpu) | |||
5140 | if (!next) | 5203 | if (!next) |
5141 | break; | 5204 | break; |
5142 | migrate_dead(dead_cpu, next); | 5205 | migrate_dead(dead_cpu, next); |
5206 | |||
5143 | } | 5207 | } |
5144 | } | 5208 | } |
5145 | #endif /* CONFIG_HOTPLUG_CPU */ | 5209 | #endif /* CONFIG_HOTPLUG_CPU */ |
5146 | 5210 | ||
5211 | #if defined(CONFIG_SCHED_DEBUG) && defined(CONFIG_SYSCTL) | ||
5212 | |||
5213 | static struct ctl_table sd_ctl_dir[] = { | ||
5214 | {CTL_UNNUMBERED, "sched_domain", NULL, 0, 0755, NULL, }, | ||
5215 | {0,}, | ||
5216 | }; | ||
5217 | |||
5218 | static struct ctl_table sd_ctl_root[] = { | ||
5219 | {CTL_UNNUMBERED, "kernel", NULL, 0, 0755, sd_ctl_dir, }, | ||
5220 | {0,}, | ||
5221 | }; | ||
5222 | |||
5223 | static struct ctl_table *sd_alloc_ctl_entry(int n) | ||
5224 | { | ||
5225 | struct ctl_table *entry = | ||
5226 | kmalloc(n * sizeof(struct ctl_table), GFP_KERNEL); | ||
5227 | |||
5228 | BUG_ON(!entry); | ||
5229 | memset(entry, 0, n * sizeof(struct ctl_table)); | ||
5230 | |||
5231 | return entry; | ||
5232 | } | ||
5233 | |||
5234 | static void | ||
5235 | set_table_entry(struct ctl_table *entry, int ctl_name, | ||
5236 | const char *procname, void *data, int maxlen, | ||
5237 | mode_t mode, proc_handler *proc_handler) | ||
5238 | { | ||
5239 | entry->ctl_name = ctl_name; | ||
5240 | entry->procname = procname; | ||
5241 | entry->data = data; | ||
5242 | entry->maxlen = maxlen; | ||
5243 | entry->mode = mode; | ||
5244 | entry->proc_handler = proc_handler; | ||
5245 | } | ||
5246 | |||
5247 | static struct ctl_table * | ||
5248 | sd_alloc_ctl_domain_table(struct sched_domain *sd) | ||
5249 | { | ||
5250 | struct ctl_table *table = sd_alloc_ctl_entry(14); | ||
5251 | |||
5252 | set_table_entry(&table[0], 1, "min_interval", &sd->min_interval, | ||
5253 | sizeof(long), 0644, proc_doulongvec_minmax); | ||
5254 | set_table_entry(&table[1], 2, "max_interval", &sd->max_interval, | ||
5255 | sizeof(long), 0644, proc_doulongvec_minmax); | ||
5256 | set_table_entry(&table[2], 3, "busy_idx", &sd->busy_idx, | ||
5257 | sizeof(int), 0644, proc_dointvec_minmax); | ||
5258 | set_table_entry(&table[3], 4, "idle_idx", &sd->idle_idx, | ||
5259 | sizeof(int), 0644, proc_dointvec_minmax); | ||
5260 | set_table_entry(&table[4], 5, "newidle_idx", &sd->newidle_idx, | ||
5261 | sizeof(int), 0644, proc_dointvec_minmax); | ||
5262 | set_table_entry(&table[5], 6, "wake_idx", &sd->wake_idx, | ||
5263 | sizeof(int), 0644, proc_dointvec_minmax); | ||
5264 | set_table_entry(&table[6], 7, "forkexec_idx", &sd->forkexec_idx, | ||
5265 | sizeof(int), 0644, proc_dointvec_minmax); | ||
5266 | set_table_entry(&table[7], 8, "busy_factor", &sd->busy_factor, | ||
5267 | sizeof(int), 0644, proc_dointvec_minmax); | ||
5268 | set_table_entry(&table[8], 9, "imbalance_pct", &sd->imbalance_pct, | ||
5269 | sizeof(int), 0644, proc_dointvec_minmax); | ||
5270 | set_table_entry(&table[9], 10, "cache_hot_time", &sd->cache_hot_time, | ||
5271 | sizeof(long long), 0644, proc_doulongvec_minmax); | ||
5272 | set_table_entry(&table[10], 11, "cache_nice_tries", | ||
5273 | &sd->cache_nice_tries, | ||
5274 | sizeof(int), 0644, proc_dointvec_minmax); | ||
5275 | set_table_entry(&table[12], 13, "flags", &sd->flags, | ||
5276 | sizeof(int), 0644, proc_dointvec_minmax); | ||
5277 | |||
5278 | return table; | ||
5279 | } | ||
5280 | |||
5281 | static ctl_table *sd_alloc_ctl_cpu_table(int cpu) | ||
5282 | { | ||
5283 | struct ctl_table *entry, *table; | ||
5284 | struct sched_domain *sd; | ||
5285 | int domain_num = 0, i; | ||
5286 | char buf[32]; | ||
5287 | |||
5288 | for_each_domain(cpu, sd) | ||
5289 | domain_num++; | ||
5290 | entry = table = sd_alloc_ctl_entry(domain_num + 1); | ||
5291 | |||
5292 | i = 0; | ||
5293 | for_each_domain(cpu, sd) { | ||
5294 | snprintf(buf, 32, "domain%d", i); | ||
5295 | entry->ctl_name = i + 1; | ||
5296 | entry->procname = kstrdup(buf, GFP_KERNEL); | ||
5297 | entry->mode = 0755; | ||
5298 | entry->child = sd_alloc_ctl_domain_table(sd); | ||
5299 | entry++; | ||
5300 | i++; | ||
5301 | } | ||
5302 | return table; | ||
5303 | } | ||
5304 | |||
5305 | static struct ctl_table_header *sd_sysctl_header; | ||
5306 | static void init_sched_domain_sysctl(void) | ||
5307 | { | ||
5308 | int i, cpu_num = num_online_cpus(); | ||
5309 | struct ctl_table *entry = sd_alloc_ctl_entry(cpu_num + 1); | ||
5310 | char buf[32]; | ||
5311 | |||
5312 | sd_ctl_dir[0].child = entry; | ||
5313 | |||
5314 | for (i = 0; i < cpu_num; i++, entry++) { | ||
5315 | snprintf(buf, 32, "cpu%d", i); | ||
5316 | entry->ctl_name = i + 1; | ||
5317 | entry->procname = kstrdup(buf, GFP_KERNEL); | ||
5318 | entry->mode = 0755; | ||
5319 | entry->child = sd_alloc_ctl_cpu_table(i); | ||
5320 | } | ||
5321 | sd_sysctl_header = register_sysctl_table(sd_ctl_root); | ||
5322 | } | ||
5323 | #else | ||
5324 | static void init_sched_domain_sysctl(void) | ||
5325 | { | ||
5326 | } | ||
5327 | #endif | ||
5328 | |||
5147 | /* | 5329 | /* |
5148 | * migration_call - callback that gets triggered when a CPU is added. | 5330 | * migration_call - callback that gets triggered when a CPU is added. |
5149 | * Here we can start up the necessary migration thread for the new CPU. | 5331 | * Here we can start up the necessary migration thread for the new CPU. |
@@ -6249,6 +6431,8 @@ void __init sched_init_smp(void) | |||
6249 | /* XXX: Theoretical race here - CPU may be hotplugged now */ | 6431 | /* XXX: Theoretical race here - CPU may be hotplugged now */ |
6250 | hotcpu_notifier(update_sched_domains, 0); | 6432 | hotcpu_notifier(update_sched_domains, 0); |
6251 | 6433 | ||
6434 | init_sched_domain_sysctl(); | ||
6435 | |||
6252 | /* Move init over to a non-isolated CPU */ | 6436 | /* Move init over to a non-isolated CPU */ |
6253 | if (set_cpus_allowed(current, non_isolated_cpus) < 0) | 6437 | if (set_cpus_allowed(current, non_isolated_cpus) < 0) |
6254 | BUG(); | 6438 | BUG(); |
@@ -6335,6 +6519,10 @@ void __init sched_init(void) | |||
6335 | 6519 | ||
6336 | set_load_weight(&init_task); | 6520 | set_load_weight(&init_task); |
6337 | 6521 | ||
6522 | #ifdef CONFIG_PREEMPT_NOTIFIERS | ||
6523 | INIT_HLIST_HEAD(&init_task.preempt_notifiers); | ||
6524 | #endif | ||
6525 | |||
6338 | #ifdef CONFIG_SMP | 6526 | #ifdef CONFIG_SMP |
6339 | nr_cpu_ids = highest_cpu + 1; | 6527 | nr_cpu_ids = highest_cpu + 1; |
6340 | open_softirq(SCHED_SOFTIRQ, run_rebalance_domains, NULL); | 6528 | open_softirq(SCHED_SOFTIRQ, run_rebalance_domains, NULL); |
diff --git a/kernel/sched_debug.c b/kernel/sched_debug.c index 29f2c21e7da2..42970f723a97 100644 --- a/kernel/sched_debug.c +++ b/kernel/sched_debug.c | |||
@@ -186,7 +186,7 @@ static int sched_debug_show(struct seq_file *m, void *v) | |||
186 | return 0; | 186 | return 0; |
187 | } | 187 | } |
188 | 188 | ||
189 | void sysrq_sched_debug_show(void) | 189 | static void sysrq_sched_debug_show(void) |
190 | { | 190 | { |
191 | sched_debug_show(NULL, NULL); | 191 | sched_debug_show(NULL, NULL); |
192 | } | 192 | } |
diff --git a/kernel/sys.c b/kernel/sys.c index 08562f419768..14f8adcfffd9 100644 --- a/kernel/sys.c +++ b/kernel/sys.c | |||
@@ -804,6 +804,7 @@ static void kernel_restart_prepare(char *cmd) | |||
804 | blocking_notifier_call_chain(&reboot_notifier_list, SYS_RESTART, cmd); | 804 | blocking_notifier_call_chain(&reboot_notifier_list, SYS_RESTART, cmd); |
805 | system_state = SYSTEM_RESTART; | 805 | system_state = SYSTEM_RESTART; |
806 | device_shutdown(); | 806 | device_shutdown(); |
807 | sysdev_shutdown(); | ||
807 | } | 808 | } |
808 | 809 | ||
809 | /** | 810 | /** |
@@ -860,6 +861,7 @@ void kernel_shutdown_prepare(enum system_states state) | |||
860 | void kernel_halt(void) | 861 | void kernel_halt(void) |
861 | { | 862 | { |
862 | kernel_shutdown_prepare(SYSTEM_HALT); | 863 | kernel_shutdown_prepare(SYSTEM_HALT); |
864 | sysdev_shutdown(); | ||
863 | printk(KERN_EMERG "System halted.\n"); | 865 | printk(KERN_EMERG "System halted.\n"); |
864 | machine_halt(); | 866 | machine_halt(); |
865 | } | 867 | } |
@@ -876,6 +878,7 @@ void kernel_power_off(void) | |||
876 | kernel_shutdown_prepare(SYSTEM_POWER_OFF); | 878 | kernel_shutdown_prepare(SYSTEM_POWER_OFF); |
877 | if (pm_power_off_prepare) | 879 | if (pm_power_off_prepare) |
878 | pm_power_off_prepare(); | 880 | pm_power_off_prepare(); |
881 | sysdev_shutdown(); | ||
879 | printk(KERN_EMERG "Power down.\n"); | 882 | printk(KERN_EMERG "Power down.\n"); |
880 | machine_power_off(); | 883 | machine_power_off(); |
881 | } | 884 | } |
diff --git a/kernel/sysctl.c b/kernel/sysctl.c index ddebf3f2affe..eb26f2ba51ed 100644 --- a/kernel/sysctl.c +++ b/kernel/sysctl.c | |||
@@ -689,7 +689,7 @@ static ctl_table kern_table[] = { | |||
689 | .proc_handler = &proc_dointvec, | 689 | .proc_handler = &proc_dointvec, |
690 | }, | 690 | }, |
691 | #endif | 691 | #endif |
692 | #ifdef CONFIG_ACPI_SLEEP | 692 | #if defined(CONFIG_ACPI) && defined(CONFIG_X86) |
693 | { | 693 | { |
694 | .ctl_name = KERN_ACPI_VIDEO_FLAGS, | 694 | .ctl_name = KERN_ACPI_VIDEO_FLAGS, |
695 | .procname = "acpi_video_flags", | 695 | .procname = "acpi_video_flags", |
diff --git a/kernel/time.c b/kernel/time.c index 5b81da08bbdb..2289a8d68314 100644 --- a/kernel/time.c +++ b/kernel/time.c | |||
@@ -215,22 +215,6 @@ asmlinkage long sys_adjtimex(struct timex __user *txc_p) | |||
215 | return copy_to_user(txc_p, &txc, sizeof(struct timex)) ? -EFAULT : ret; | 215 | return copy_to_user(txc_p, &txc, sizeof(struct timex)) ? -EFAULT : ret; |
216 | } | 216 | } |
217 | 217 | ||
218 | inline struct timespec current_kernel_time(void) | ||
219 | { | ||
220 | struct timespec now; | ||
221 | unsigned long seq; | ||
222 | |||
223 | do { | ||
224 | seq = read_seqbegin(&xtime_lock); | ||
225 | |||
226 | now = xtime; | ||
227 | } while (read_seqretry(&xtime_lock, seq)); | ||
228 | |||
229 | return now; | ||
230 | } | ||
231 | |||
232 | EXPORT_SYMBOL(current_kernel_time); | ||
233 | |||
234 | /** | 218 | /** |
235 | * current_fs_time - Return FS time | 219 | * current_fs_time - Return FS time |
236 | * @sb: Superblock. | 220 | * @sb: Superblock. |
diff --git a/kernel/time/timekeeping.c b/kernel/time/timekeeping.c index 88c81026e003..acc417b5a9b7 100644 --- a/kernel/time/timekeeping.c +++ b/kernel/time/timekeeping.c | |||
@@ -47,10 +47,22 @@ EXPORT_SYMBOL(xtime_lock); | |||
47 | struct timespec xtime __attribute__ ((aligned (16))); | 47 | struct timespec xtime __attribute__ ((aligned (16))); |
48 | struct timespec wall_to_monotonic __attribute__ ((aligned (16))); | 48 | struct timespec wall_to_monotonic __attribute__ ((aligned (16))); |
49 | static unsigned long total_sleep_time; /* seconds */ | 49 | static unsigned long total_sleep_time; /* seconds */ |
50 | |||
51 | EXPORT_SYMBOL(xtime); | 50 | EXPORT_SYMBOL(xtime); |
52 | 51 | ||
53 | 52 | ||
53 | #ifdef CONFIG_NO_HZ | ||
54 | static struct timespec xtime_cache __attribute__ ((aligned (16))); | ||
55 | static inline void update_xtime_cache(u64 nsec) | ||
56 | { | ||
57 | xtime_cache = xtime; | ||
58 | timespec_add_ns(&xtime_cache, nsec); | ||
59 | } | ||
60 | #else | ||
61 | #define xtime_cache xtime | ||
62 | /* We do *not* want to evaluate the argument for this case */ | ||
63 | #define update_xtime_cache(n) do { } while (0) | ||
64 | #endif | ||
65 | |||
54 | static struct clocksource *clock; /* pointer to current clocksource */ | 66 | static struct clocksource *clock; /* pointer to current clocksource */ |
55 | 67 | ||
56 | 68 | ||
@@ -478,6 +490,8 @@ void update_wall_time(void) | |||
478 | xtime.tv_nsec = (s64)clock->xtime_nsec >> clock->shift; | 490 | xtime.tv_nsec = (s64)clock->xtime_nsec >> clock->shift; |
479 | clock->xtime_nsec -= (s64)xtime.tv_nsec << clock->shift; | 491 | clock->xtime_nsec -= (s64)xtime.tv_nsec << clock->shift; |
480 | 492 | ||
493 | update_xtime_cache(cyc2ns(clock, offset)); | ||
494 | |||
481 | /* check to see if there is a new clocksource to use */ | 495 | /* check to see if there is a new clocksource to use */ |
482 | change_clocksource(); | 496 | change_clocksource(); |
483 | update_vsyscall(&xtime, clock); | 497 | update_vsyscall(&xtime, clock); |
@@ -509,3 +523,25 @@ void monotonic_to_bootbased(struct timespec *ts) | |||
509 | { | 523 | { |
510 | ts->tv_sec += total_sleep_time; | 524 | ts->tv_sec += total_sleep_time; |
511 | } | 525 | } |
526 | |||
527 | unsigned long get_seconds(void) | ||
528 | { | ||
529 | return xtime_cache.tv_sec; | ||
530 | } | ||
531 | EXPORT_SYMBOL(get_seconds); | ||
532 | |||
533 | |||
534 | struct timespec current_kernel_time(void) | ||
535 | { | ||
536 | struct timespec now; | ||
537 | unsigned long seq; | ||
538 | |||
539 | do { | ||
540 | seq = read_seqbegin(&xtime_lock); | ||
541 | |||
542 | now = xtime_cache; | ||
543 | } while (read_seqretry(&xtime_lock, seq)); | ||
544 | |||
545 | return now; | ||
546 | } | ||
547 | EXPORT_SYMBOL(current_kernel_time); | ||
diff --git a/kernel/tsacct.c b/kernel/tsacct.c index 658f638c402c..c122131a122f 100644 --- a/kernel/tsacct.c +++ b/kernel/tsacct.c | |||
@@ -39,7 +39,7 @@ void bacct_add_tsk(struct taskstats *stats, struct task_struct *tsk) | |||
39 | ac_etime = timespec_to_ns(&ts); | 39 | ac_etime = timespec_to_ns(&ts); |
40 | do_div(ac_etime, NSEC_PER_USEC); | 40 | do_div(ac_etime, NSEC_PER_USEC); |
41 | stats->ac_etime = ac_etime; | 41 | stats->ac_etime = ac_etime; |
42 | stats->ac_btime = xtime.tv_sec - ts.tv_sec; | 42 | stats->ac_btime = get_seconds() - ts.tv_sec; |
43 | if (thread_group_leader(tsk)) { | 43 | if (thread_group_leader(tsk)) { |
44 | stats->ac_exitcode = tsk->exit_code; | 44 | stats->ac_exitcode = tsk->exit_code; |
45 | if (tsk->flags & PF_FORKNOEXEC) | 45 | if (tsk->flags & PF_FORKNOEXEC) |
diff --git a/mm/bounce.c b/mm/bounce.c index ad401fc57440..179fe38a2416 100644 --- a/mm/bounce.c +++ b/mm/bounce.c | |||
@@ -190,7 +190,7 @@ static int bounce_end_io_read_isa(struct bio *bio, unsigned int bytes_done, int | |||
190 | return 0; | 190 | return 0; |
191 | } | 191 | } |
192 | 192 | ||
193 | static void __blk_queue_bounce(request_queue_t *q, struct bio **bio_orig, | 193 | static void __blk_queue_bounce(struct request_queue *q, struct bio **bio_orig, |
194 | mempool_t *pool) | 194 | mempool_t *pool) |
195 | { | 195 | { |
196 | struct page *page; | 196 | struct page *page; |
@@ -275,7 +275,7 @@ static void __blk_queue_bounce(request_queue_t *q, struct bio **bio_orig, | |||
275 | *bio_orig = bio; | 275 | *bio_orig = bio; |
276 | } | 276 | } |
277 | 277 | ||
278 | void blk_queue_bounce(request_queue_t *q, struct bio **bio_orig) | 278 | void blk_queue_bounce(struct request_queue *q, struct bio **bio_orig) |
279 | { | 279 | { |
280 | mempool_t *pool; | 280 | mempool_t *pool; |
281 | 281 | ||
diff --git a/mm/migrate.c b/mm/migrate.c index 34d8ada053e4..37c73b902008 100644 --- a/mm/migrate.c +++ b/mm/migrate.c | |||
@@ -49,9 +49,8 @@ int isolate_lru_page(struct page *page, struct list_head *pagelist) | |||
49 | struct zone *zone = page_zone(page); | 49 | struct zone *zone = page_zone(page); |
50 | 50 | ||
51 | spin_lock_irq(&zone->lru_lock); | 51 | spin_lock_irq(&zone->lru_lock); |
52 | if (PageLRU(page)) { | 52 | if (PageLRU(page) && get_page_unless_zero(page)) { |
53 | ret = 0; | 53 | ret = 0; |
54 | get_page(page); | ||
55 | ClearPageLRU(page); | 54 | ClearPageLRU(page); |
56 | if (PageActive(page)) | 55 | if (PageActive(page)) |
57 | del_page_from_active_list(zone, page); | 56 | del_page_from_active_list(zone, page); |
@@ -632,18 +631,35 @@ static int unmap_and_move(new_page_t get_new_page, unsigned long private, | |||
632 | goto unlock; | 631 | goto unlock; |
633 | wait_on_page_writeback(page); | 632 | wait_on_page_writeback(page); |
634 | } | 633 | } |
635 | |||
636 | /* | 634 | /* |
637 | * Establish migration ptes or remove ptes | 635 | * By try_to_unmap(), page->mapcount goes down to 0 here. In this case, |
636 | * we cannot notice that anon_vma is freed while we migrates a page. | ||
637 | * This rcu_read_lock() delays freeing anon_vma pointer until the end | ||
638 | * of migration. File cache pages are no problem because of page_lock() | ||
639 | */ | ||
640 | rcu_read_lock(); | ||
641 | /* | ||
642 | * This is a corner case handling. | ||
643 | * When a new swap-cache is read into, it is linked to LRU | ||
644 | * and treated as swapcache but has no rmap yet. | ||
645 | * Calling try_to_unmap() against a page->mapping==NULL page is | ||
646 | * BUG. So handle it here. | ||
638 | */ | 647 | */ |
648 | if (!page->mapping) | ||
649 | goto rcu_unlock; | ||
650 | /* Establish migration ptes or remove ptes */ | ||
639 | try_to_unmap(page, 1); | 651 | try_to_unmap(page, 1); |
652 | |||
640 | if (!page_mapped(page)) | 653 | if (!page_mapped(page)) |
641 | rc = move_to_new_page(newpage, page); | 654 | rc = move_to_new_page(newpage, page); |
642 | 655 | ||
643 | if (rc) | 656 | if (rc) |
644 | remove_migration_ptes(page, page); | 657 | remove_migration_ptes(page, page); |
658 | rcu_unlock: | ||
659 | rcu_read_unlock(); | ||
645 | 660 | ||
646 | unlock: | 661 | unlock: |
662 | |||
647 | unlock_page(page); | 663 | unlock_page(page); |
648 | 664 | ||
649 | if (rc != -EAGAIN) { | 665 | if (rc != -EAGAIN) { |
diff --git a/mm/page_alloc.c b/mm/page_alloc.c index 40954fb81598..6d3550ca0282 100644 --- a/mm/page_alloc.c +++ b/mm/page_alloc.c | |||
@@ -2775,11 +2775,11 @@ unsigned long __meminit __absent_pages_in_range(int nid, | |||
2775 | if (i == -1) | 2775 | if (i == -1) |
2776 | return 0; | 2776 | return 0; |
2777 | 2777 | ||
2778 | prev_end_pfn = min(early_node_map[i].start_pfn, range_end_pfn); | ||
2779 | |||
2778 | /* Account for ranges before physical memory on this node */ | 2780 | /* Account for ranges before physical memory on this node */ |
2779 | if (early_node_map[i].start_pfn > range_start_pfn) | 2781 | if (early_node_map[i].start_pfn > range_start_pfn) |
2780 | hole_pages = early_node_map[i].start_pfn - range_start_pfn; | 2782 | hole_pages = prev_end_pfn - range_start_pfn; |
2781 | |||
2782 | prev_end_pfn = early_node_map[i].start_pfn; | ||
2783 | 2783 | ||
2784 | /* Find all holes for the zone within the node */ | 2784 | /* Find all holes for the zone within the node */ |
2785 | for (; i != -1; i = next_active_region_index_in_nid(i, nid)) { | 2785 | for (; i != -1; i = next_active_region_index_in_nid(i, nid)) { |
diff --git a/net/8021q/vlan.c b/net/8021q/vlan.c index cda936b77d22..1583c5ef963f 100644 --- a/net/8021q/vlan.c +++ b/net/8021q/vlan.c | |||
@@ -810,6 +810,7 @@ static int vlan_ioctl_handler(void __user *arg) | |||
810 | err = -EINVAL; | 810 | err = -EINVAL; |
811 | break; | 811 | break; |
812 | case GET_VLAN_REALDEV_NAME_CMD: | 812 | case GET_VLAN_REALDEV_NAME_CMD: |
813 | err = 0; | ||
813 | vlan_dev_get_realdev_name(dev, args.u.device2); | 814 | vlan_dev_get_realdev_name(dev, args.u.device2); |
814 | if (copy_to_user(arg, &args, | 815 | if (copy_to_user(arg, &args, |
815 | sizeof(struct vlan_ioctl_args))) { | 816 | sizeof(struct vlan_ioctl_args))) { |
@@ -818,6 +819,7 @@ static int vlan_ioctl_handler(void __user *arg) | |||
818 | break; | 819 | break; |
819 | 820 | ||
820 | case GET_VLAN_VID_CMD: | 821 | case GET_VLAN_VID_CMD: |
822 | err = 0; | ||
821 | vlan_dev_get_vid(dev, &vid); | 823 | vlan_dev_get_vid(dev, &vid); |
822 | args.u.VID = vid; | 824 | args.u.VID = vid; |
823 | if (copy_to_user(arg, &args, | 825 | if (copy_to_user(arg, &args, |
diff --git a/net/bridge/br_input.c b/net/bridge/br_input.c index 420bbb9955e9..5c18595b7616 100644 --- a/net/bridge/br_input.c +++ b/net/bridge/br_input.c | |||
@@ -112,9 +112,9 @@ static int br_handle_local_finish(struct sk_buff *skb) | |||
112 | */ | 112 | */ |
113 | static inline int is_link_local(const unsigned char *dest) | 113 | static inline int is_link_local(const unsigned char *dest) |
114 | { | 114 | { |
115 | const u16 *a = (const u16 *) dest; | 115 | __be16 *a = (__be16 *)dest; |
116 | static const u16 *const b = (const u16 *const ) br_group_address; | 116 | static const __be16 *b = (const __be16 *)br_group_address; |
117 | static const u16 m = __constant_cpu_to_be16(0xfff0); | 117 | static const __be16 m = __constant_cpu_to_be16(0xfff0); |
118 | 118 | ||
119 | return ((a[0] ^ b[0]) | (a[1] ^ b[1]) | ((a[2] ^ b[2]) & m)) == 0; | 119 | return ((a[0] ^ b[0]) | (a[1] ^ b[1]) | ((a[2] ^ b[2]) & m)) == 0; |
120 | } | 120 | } |
diff --git a/net/bridge/netfilter/ebt_log.c b/net/bridge/netfilter/ebt_log.c index 031bfa4a51fc..457815fb5584 100644 --- a/net/bridge/netfilter/ebt_log.c +++ b/net/bridge/netfilter/ebt_log.c | |||
@@ -9,7 +9,6 @@ | |||
9 | * | 9 | * |
10 | */ | 10 | */ |
11 | 11 | ||
12 | #include <linux/in.h> | ||
13 | #include <linux/netfilter_bridge/ebtables.h> | 12 | #include <linux/netfilter_bridge/ebtables.h> |
14 | #include <linux/netfilter_bridge/ebt_log.h> | 13 | #include <linux/netfilter_bridge/ebt_log.h> |
15 | #include <linux/netfilter.h> | 14 | #include <linux/netfilter.h> |
@@ -196,10 +195,8 @@ static int __init ebt_log_init(void) | |||
196 | ret = ebt_register_watcher(&log); | 195 | ret = ebt_register_watcher(&log); |
197 | if (ret < 0) | 196 | if (ret < 0) |
198 | return ret; | 197 | return ret; |
199 | ret = nf_log_register(PF_BRIDGE, &ebt_log_logger); | 198 | nf_log_register(PF_BRIDGE, &ebt_log_logger); |
200 | if (ret < 0 && ret != -EEXIST) | 199 | return 0; |
201 | ebt_unregister_watcher(&log); | ||
202 | return ret; | ||
203 | } | 200 | } |
204 | 201 | ||
205 | static void __exit ebt_log_fini(void) | 202 | static void __exit ebt_log_fini(void) |
diff --git a/net/bridge/netfilter/ebt_ulog.c b/net/bridge/netfilter/ebt_ulog.c index 9411db625917..204c968fa86d 100644 --- a/net/bridge/netfilter/ebt_ulog.c +++ b/net/bridge/netfilter/ebt_ulog.c | |||
@@ -36,7 +36,6 @@ | |||
36 | #include <linux/timer.h> | 36 | #include <linux/timer.h> |
37 | #include <linux/netlink.h> | 37 | #include <linux/netlink.h> |
38 | #include <linux/netdevice.h> | 38 | #include <linux/netdevice.h> |
39 | #include <linux/module.h> | ||
40 | #include <linux/netfilter_bridge/ebtables.h> | 39 | #include <linux/netfilter_bridge/ebtables.h> |
41 | #include <linux/netfilter_bridge/ebt_ulog.h> | 40 | #include <linux/netfilter_bridge/ebt_ulog.h> |
42 | #include <net/sock.h> | 41 | #include <net/sock.h> |
@@ -308,12 +307,8 @@ static int __init ebt_ulog_init(void) | |||
308 | else if ((ret = ebt_register_watcher(&ulog))) | 307 | else if ((ret = ebt_register_watcher(&ulog))) |
309 | sock_release(ebtulognl->sk_socket); | 308 | sock_release(ebtulognl->sk_socket); |
310 | 309 | ||
311 | if (nf_log_register(PF_BRIDGE, &ebt_ulog_logger) < 0) { | 310 | if (ret == 0) |
312 | printk(KERN_WARNING "ebt_ulog: not logging via ulog " | 311 | nf_log_register(PF_BRIDGE, &ebt_ulog_logger); |
313 | "since somebody else already registered for PF_BRIDGE\n"); | ||
314 | /* we cannot make module load fail here, since otherwise | ||
315 | * ebtables userspace would abort */ | ||
316 | } | ||
317 | 312 | ||
318 | return ret; | 313 | return ret; |
319 | } | 314 | } |
diff --git a/net/ipv4/netfilter/ipt_LOG.c b/net/ipv4/netfilter/ipt_LOG.c index 5937ad150b9f..127a5e89bf14 100644 --- a/net/ipv4/netfilter/ipt_LOG.c +++ b/net/ipv4/netfilter/ipt_LOG.c | |||
@@ -479,10 +479,8 @@ static int __init ipt_log_init(void) | |||
479 | ret = xt_register_target(&ipt_log_reg); | 479 | ret = xt_register_target(&ipt_log_reg); |
480 | if (ret < 0) | 480 | if (ret < 0) |
481 | return ret; | 481 | return ret; |
482 | ret = nf_log_register(PF_INET, &ipt_log_logger); | 482 | nf_log_register(PF_INET, &ipt_log_logger); |
483 | if (ret < 0 && ret != -EEXIST) | 483 | return 0; |
484 | xt_unregister_target(&ipt_log_reg); | ||
485 | return ret; | ||
486 | } | 484 | } |
487 | 485 | ||
488 | static void __exit ipt_log_fini(void) | 486 | static void __exit ipt_log_fini(void) |
diff --git a/net/ipv4/netfilter/nf_nat_core.c b/net/ipv4/netfilter/nf_nat_core.c index e848d8d6292f..deab27facbad 100644 --- a/net/ipv4/netfilter/nf_nat_core.c +++ b/net/ipv4/netfilter/nf_nat_core.c | |||
@@ -77,7 +77,8 @@ static inline unsigned int | |||
77 | hash_by_src(const struct nf_conntrack_tuple *tuple) | 77 | hash_by_src(const struct nf_conntrack_tuple *tuple) |
78 | { | 78 | { |
79 | /* Original src, to ensure we map it consistently if poss. */ | 79 | /* Original src, to ensure we map it consistently if poss. */ |
80 | return jhash_3words((__force u32)tuple->src.u3.ip, tuple->src.u.all, | 80 | return jhash_3words((__force u32)tuple->src.u3.ip, |
81 | (__force u32)tuple->src.u.all, | ||
81 | tuple->dst.protonum, 0) % nf_nat_htable_size; | 82 | tuple->dst.protonum, 0) % nf_nat_htable_size; |
82 | } | 83 | } |
83 | 84 | ||
diff --git a/net/ipv4/netfilter/nf_nat_rule.c b/net/ipv4/netfilter/nf_nat_rule.c index 0f45427e5fdc..76ec59ae524d 100644 --- a/net/ipv4/netfilter/nf_nat_rule.c +++ b/net/ipv4/netfilter/nf_nat_rule.c | |||
@@ -192,7 +192,7 @@ alloc_null_binding_confirmed(struct nf_conn *ct, unsigned int hooknum) | |||
192 | = (HOOK2MANIP(hooknum) == IP_NAT_MANIP_SRC | 192 | = (HOOK2MANIP(hooknum) == IP_NAT_MANIP_SRC |
193 | ? ct->tuplehash[IP_CT_DIR_REPLY].tuple.dst.u3.ip | 193 | ? ct->tuplehash[IP_CT_DIR_REPLY].tuple.dst.u3.ip |
194 | : ct->tuplehash[IP_CT_DIR_REPLY].tuple.src.u3.ip); | 194 | : ct->tuplehash[IP_CT_DIR_REPLY].tuple.src.u3.ip); |
195 | u_int16_t all | 195 | __be16 all |
196 | = (HOOK2MANIP(hooknum) == IP_NAT_MANIP_SRC | 196 | = (HOOK2MANIP(hooknum) == IP_NAT_MANIP_SRC |
197 | ? ct->tuplehash[IP_CT_DIR_REPLY].tuple.dst.u.all | 197 | ? ct->tuplehash[IP_CT_DIR_REPLY].tuple.dst.u.all |
198 | : ct->tuplehash[IP_CT_DIR_REPLY].tuple.src.u.all); | 198 | : ct->tuplehash[IP_CT_DIR_REPLY].tuple.src.u.all); |
diff --git a/net/ipv6/ip6_tunnel.c b/net/ipv6/ip6_tunnel.c index df30976f6dfd..ca774d8e3be3 100644 --- a/net/ipv6/ip6_tunnel.c +++ b/net/ipv6/ip6_tunnel.c | |||
@@ -385,7 +385,7 @@ parse_tlv_tnl_enc_lim(struct sk_buff *skb, __u8 * raw) | |||
385 | 385 | ||
386 | static int | 386 | static int |
387 | ip6_tnl_err(struct sk_buff *skb, __u8 ipproto, struct inet6_skb_parm *opt, | 387 | ip6_tnl_err(struct sk_buff *skb, __u8 ipproto, struct inet6_skb_parm *opt, |
388 | int *type, int *code, int *msg, __be32 *info, int offset) | 388 | int *type, int *code, int *msg, __u32 *info, int offset) |
389 | { | 389 | { |
390 | struct ipv6hdr *ipv6h = (struct ipv6hdr *) skb->data; | 390 | struct ipv6hdr *ipv6h = (struct ipv6hdr *) skb->data; |
391 | struct ip6_tnl *t; | 391 | struct ip6_tnl *t; |
@@ -435,7 +435,7 @@ ip6_tnl_err(struct sk_buff *skb, __u8 ipproto, struct inet6_skb_parm *opt, | |||
435 | if ((*code) == ICMPV6_HDR_FIELD) | 435 | if ((*code) == ICMPV6_HDR_FIELD) |
436 | teli = parse_tlv_tnl_enc_lim(skb, skb->data); | 436 | teli = parse_tlv_tnl_enc_lim(skb, skb->data); |
437 | 437 | ||
438 | if (teli && teli == ntohl(*info) - 2) { | 438 | if (teli && teli == *info - 2) { |
439 | tel = (struct ipv6_tlv_tnl_enc_lim *) &skb->data[teli]; | 439 | tel = (struct ipv6_tlv_tnl_enc_lim *) &skb->data[teli]; |
440 | if (tel->encap_limit == 0) { | 440 | if (tel->encap_limit == 0) { |
441 | if (net_ratelimit()) | 441 | if (net_ratelimit()) |
@@ -452,7 +452,7 @@ ip6_tnl_err(struct sk_buff *skb, __u8 ipproto, struct inet6_skb_parm *opt, | |||
452 | } | 452 | } |
453 | break; | 453 | break; |
454 | case ICMPV6_PKT_TOOBIG: | 454 | case ICMPV6_PKT_TOOBIG: |
455 | mtu = ntohl(*info) - offset; | 455 | mtu = *info - offset; |
456 | if (mtu < IPV6_MIN_MTU) | 456 | if (mtu < IPV6_MIN_MTU) |
457 | mtu = IPV6_MIN_MTU; | 457 | mtu = IPV6_MIN_MTU; |
458 | t->dev->mtu = mtu; | 458 | t->dev->mtu = mtu; |
@@ -478,12 +478,12 @@ out: | |||
478 | 478 | ||
479 | static int | 479 | static int |
480 | ip4ip6_err(struct sk_buff *skb, struct inet6_skb_parm *opt, | 480 | ip4ip6_err(struct sk_buff *skb, struct inet6_skb_parm *opt, |
481 | int type, int code, int offset, __u32 info) | 481 | int type, int code, int offset, __be32 info) |
482 | { | 482 | { |
483 | int rel_msg = 0; | 483 | int rel_msg = 0; |
484 | int rel_type = type; | 484 | int rel_type = type; |
485 | int rel_code = code; | 485 | int rel_code = code; |
486 | __u32 rel_info = info; | 486 | __u32 rel_info = ntohl(info); |
487 | int err; | 487 | int err; |
488 | struct sk_buff *skb2; | 488 | struct sk_buff *skb2; |
489 | struct iphdr *eiph; | 489 | struct iphdr *eiph; |
@@ -564,10 +564,9 @@ ip4ip6_err(struct sk_buff *skb, struct inet6_skb_parm *opt, | |||
564 | goto out; | 564 | goto out; |
565 | 565 | ||
566 | skb2->dst->ops->update_pmtu(skb2->dst, rel_info); | 566 | skb2->dst->ops->update_pmtu(skb2->dst, rel_info); |
567 | rel_info = htonl(rel_info); | ||
568 | } | 567 | } |
569 | 568 | ||
570 | icmp_send(skb2, rel_type, rel_code, rel_info); | 569 | icmp_send(skb2, rel_type, rel_code, htonl(rel_info)); |
571 | 570 | ||
572 | out: | 571 | out: |
573 | kfree_skb(skb2); | 572 | kfree_skb(skb2); |
@@ -576,12 +575,12 @@ out: | |||
576 | 575 | ||
577 | static int | 576 | static int |
578 | ip6ip6_err(struct sk_buff *skb, struct inet6_skb_parm *opt, | 577 | ip6ip6_err(struct sk_buff *skb, struct inet6_skb_parm *opt, |
579 | int type, int code, int offset, __u32 info) | 578 | int type, int code, int offset, __be32 info) |
580 | { | 579 | { |
581 | int rel_msg = 0; | 580 | int rel_msg = 0; |
582 | int rel_type = type; | 581 | int rel_type = type; |
583 | int rel_code = code; | 582 | int rel_code = code; |
584 | __u32 rel_info = info; | 583 | __u32 rel_info = ntohl(info); |
585 | int err; | 584 | int err; |
586 | 585 | ||
587 | err = ip6_tnl_err(skb, IPPROTO_IPV6, opt, &rel_type, &rel_code, | 586 | err = ip6_tnl_err(skb, IPPROTO_IPV6, opt, &rel_type, &rel_code, |
diff --git a/net/ipv6/netfilter/ip6t_LOG.c b/net/ipv6/netfilter/ip6t_LOG.c index b05327ebd332..6ab99001dccc 100644 --- a/net/ipv6/netfilter/ip6t_LOG.c +++ b/net/ipv6/netfilter/ip6t_LOG.c | |||
@@ -493,10 +493,8 @@ static int __init ip6t_log_init(void) | |||
493 | ret = xt_register_target(&ip6t_log_reg); | 493 | ret = xt_register_target(&ip6t_log_reg); |
494 | if (ret < 0) | 494 | if (ret < 0) |
495 | return ret; | 495 | return ret; |
496 | ret = nf_log_register(PF_INET6, &ip6t_logger); | 496 | nf_log_register(PF_INET6, &ip6t_logger); |
497 | if (ret < 0 && ret != -EEXIST) | 497 | return 0; |
498 | xt_unregister_target(&ip6t_log_reg); | ||
499 | return ret; | ||
500 | } | 498 | } |
501 | 499 | ||
502 | static void __exit ip6t_log_fini(void) | 500 | static void __exit ip6t_log_fini(void) |
diff --git a/net/ipv6/tcp_ipv6.c b/net/ipv6/tcp_ipv6.c index d67fb1ef751e..f10f3689d671 100644 --- a/net/ipv6/tcp_ipv6.c +++ b/net/ipv6/tcp_ipv6.c | |||
@@ -633,6 +633,7 @@ static int tcp_v6_md5_do_del(struct sock *sk, struct in6_addr *peer) | |||
633 | if (tp->md5sig_info->entries6 == 0) { | 633 | if (tp->md5sig_info->entries6 == 0) { |
634 | kfree(tp->md5sig_info->keys6); | 634 | kfree(tp->md5sig_info->keys6); |
635 | tp->md5sig_info->keys6 = NULL; | 635 | tp->md5sig_info->keys6 = NULL; |
636 | tp->md5sig_info->alloced6 = 0; | ||
636 | 637 | ||
637 | tcp_free_md5sig_pool(); | 638 | tcp_free_md5sig_pool(); |
638 | 639 | ||
diff --git a/net/key/af_key.c b/net/key/af_key.c index 0f8304b0246b..7b0a95abe934 100644 --- a/net/key/af_key.c +++ b/net/key/af_key.c | |||
@@ -2540,7 +2540,7 @@ static int pfkey_migrate(struct sock *sk, struct sk_buff *skb, | |||
2540 | sel.proto = pfkey_proto_to_xfrm(sa->sadb_address_proto); | 2540 | sel.proto = pfkey_proto_to_xfrm(sa->sadb_address_proto); |
2541 | sel.sport = ((struct sockaddr_in *)(sa + 1))->sin_port; | 2541 | sel.sport = ((struct sockaddr_in *)(sa + 1))->sin_port; |
2542 | if (sel.sport) | 2542 | if (sel.sport) |
2543 | sel.sport_mask = ~0; | 2543 | sel.sport_mask = htons(0xffff); |
2544 | 2544 | ||
2545 | /* set destination address info of selector */ | 2545 | /* set destination address info of selector */ |
2546 | sa = ext_hdrs[SADB_EXT_ADDRESS_DST - 1], | 2546 | sa = ext_hdrs[SADB_EXT_ADDRESS_DST - 1], |
@@ -2549,7 +2549,7 @@ static int pfkey_migrate(struct sock *sk, struct sk_buff *skb, | |||
2549 | sel.proto = pfkey_proto_to_xfrm(sa->sadb_address_proto); | 2549 | sel.proto = pfkey_proto_to_xfrm(sa->sadb_address_proto); |
2550 | sel.dport = ((struct sockaddr_in *)(sa + 1))->sin_port; | 2550 | sel.dport = ((struct sockaddr_in *)(sa + 1))->sin_port; |
2551 | if (sel.dport) | 2551 | if (sel.dport) |
2552 | sel.dport_mask = ~0; | 2552 | sel.dport_mask = htons(0xffff); |
2553 | 2553 | ||
2554 | rq = (struct sadb_x_ipsecrequest *)(pol + 1); | 2554 | rq = (struct sadb_x_ipsecrequest *)(pol + 1); |
2555 | 2555 | ||
diff --git a/net/netfilter/nf_conntrack_core.c b/net/netfilter/nf_conntrack_core.c index aa086c83af80..0fe11889ce14 100644 --- a/net/netfilter/nf_conntrack_core.c +++ b/net/netfilter/nf_conntrack_core.c | |||
@@ -79,7 +79,8 @@ static u_int32_t __hash_conntrack(const struct nf_conntrack_tuple *tuple, | |||
79 | a = jhash2(tuple->src.u3.all, ARRAY_SIZE(tuple->src.u3.all), | 79 | a = jhash2(tuple->src.u3.all, ARRAY_SIZE(tuple->src.u3.all), |
80 | (tuple->src.l3num << 16) | tuple->dst.protonum); | 80 | (tuple->src.l3num << 16) | tuple->dst.protonum); |
81 | b = jhash2(tuple->dst.u3.all, ARRAY_SIZE(tuple->dst.u3.all), | 81 | b = jhash2(tuple->dst.u3.all, ARRAY_SIZE(tuple->dst.u3.all), |
82 | (tuple->src.u.all << 16) | tuple->dst.u.all); | 82 | ((__force __u16)tuple->src.u.all << 16) | |
83 | (__force __u16)tuple->dst.u.all); | ||
83 | 84 | ||
84 | return jhash_2words(a, b, rnd) % size; | 85 | return jhash_2words(a, b, rnd) % size; |
85 | } | 86 | } |
diff --git a/net/netfilter/nf_conntrack_expect.c b/net/netfilter/nf_conntrack_expect.c index 1aa6229ca99f..eb6695dcd73b 100644 --- a/net/netfilter/nf_conntrack_expect.c +++ b/net/netfilter/nf_conntrack_expect.c | |||
@@ -80,7 +80,7 @@ static unsigned int nf_ct_expect_dst_hash(const struct nf_conntrack_tuple *tuple | |||
80 | 80 | ||
81 | return jhash2(tuple->dst.u3.all, ARRAY_SIZE(tuple->dst.u3.all), | 81 | return jhash2(tuple->dst.u3.all, ARRAY_SIZE(tuple->dst.u3.all), |
82 | (((tuple->dst.protonum ^ tuple->src.l3num) << 16) | | 82 | (((tuple->dst.protonum ^ tuple->src.l3num) << 16) | |
83 | tuple->dst.u.all) ^ nf_ct_expect_hash_rnd) % | 83 | (__force __u16)tuple->dst.u.all) ^ nf_ct_expect_hash_rnd) % |
84 | nf_ct_expect_hsize; | 84 | nf_ct_expect_hsize; |
85 | } | 85 | } |
86 | 86 | ||
@@ -259,8 +259,8 @@ void nf_ct_expect_init(struct nf_conntrack_expect *exp, int family, | |||
259 | } | 259 | } |
260 | 260 | ||
261 | if (src) { | 261 | if (src) { |
262 | exp->tuple.src.u.all = (__force u16)*src; | 262 | exp->tuple.src.u.all = *src; |
263 | exp->mask.src.u.all = 0xFFFF; | 263 | exp->mask.src.u.all = htons(0xFFFF); |
264 | } else { | 264 | } else { |
265 | exp->tuple.src.u.all = 0; | 265 | exp->tuple.src.u.all = 0; |
266 | exp->mask.src.u.all = 0; | 266 | exp->mask.src.u.all = 0; |
@@ -272,7 +272,7 @@ void nf_ct_expect_init(struct nf_conntrack_expect *exp, int family, | |||
272 | memset((void *)&exp->tuple.dst.u3 + len, 0x00, | 272 | memset((void *)&exp->tuple.dst.u3 + len, 0x00, |
273 | sizeof(exp->tuple.dst.u3) - len); | 273 | sizeof(exp->tuple.dst.u3) - len); |
274 | 274 | ||
275 | exp->tuple.dst.u.all = (__force u16)*dst; | 275 | exp->tuple.dst.u.all = *dst; |
276 | } | 276 | } |
277 | EXPORT_SYMBOL_GPL(nf_ct_expect_init); | 277 | EXPORT_SYMBOL_GPL(nf_ct_expect_init); |
278 | 278 | ||
diff --git a/net/netfilter/nf_conntrack_helper.c b/net/netfilter/nf_conntrack_helper.c index ca10df40784f..96aa637c0932 100644 --- a/net/netfilter/nf_conntrack_helper.c +++ b/net/netfilter/nf_conntrack_helper.c | |||
@@ -39,7 +39,7 @@ static int nf_ct_helper_vmalloc; | |||
39 | static unsigned int helper_hash(const struct nf_conntrack_tuple *tuple) | 39 | static unsigned int helper_hash(const struct nf_conntrack_tuple *tuple) |
40 | { | 40 | { |
41 | return (((tuple->src.l3num << 8) | tuple->dst.protonum) ^ | 41 | return (((tuple->src.l3num << 8) | tuple->dst.protonum) ^ |
42 | tuple->src.u.all) % nf_ct_helper_hsize; | 42 | (__force __u16)tuple->src.u.all) % nf_ct_helper_hsize; |
43 | } | 43 | } |
44 | 44 | ||
45 | struct nf_conntrack_helper * | 45 | struct nf_conntrack_helper * |
diff --git a/net/netfilter/nf_conntrack_proto_tcp.c b/net/netfilter/nf_conntrack_proto_tcp.c index 87ad3ccf8aff..eb3fe7401466 100644 --- a/net/netfilter/nf_conntrack_proto_tcp.c +++ b/net/netfilter/nf_conntrack_proto_tcp.c | |||
@@ -8,7 +8,6 @@ | |||
8 | 8 | ||
9 | #include <linux/types.h> | 9 | #include <linux/types.h> |
10 | #include <linux/timer.h> | 10 | #include <linux/timer.h> |
11 | #include <linux/netfilter.h> | ||
12 | #include <linux/module.h> | 11 | #include <linux/module.h> |
13 | #include <linux/in.h> | 12 | #include <linux/in.h> |
14 | #include <linux/tcp.h> | 13 | #include <linux/tcp.h> |
diff --git a/net/netfilter/nf_conntrack_proto_udp.c b/net/netfilter/nf_conntrack_proto_udp.c index 13d94a025723..2a2fd1a764ea 100644 --- a/net/netfilter/nf_conntrack_proto_udp.c +++ b/net/netfilter/nf_conntrack_proto_udp.c | |||
@@ -9,7 +9,6 @@ | |||
9 | #include <linux/types.h> | 9 | #include <linux/types.h> |
10 | #include <linux/timer.h> | 10 | #include <linux/timer.h> |
11 | #include <linux/module.h> | 11 | #include <linux/module.h> |
12 | #include <linux/netfilter.h> | ||
13 | #include <linux/udp.h> | 12 | #include <linux/udp.h> |
14 | #include <linux/seq_file.h> | 13 | #include <linux/seq_file.h> |
15 | #include <linux/skbuff.h> | 14 | #include <linux/skbuff.h> |
diff --git a/net/netfilter/nf_conntrack_proto_udplite.c b/net/netfilter/nf_conntrack_proto_udplite.c index 93e747b5396e..b906b413997c 100644 --- a/net/netfilter/nf_conntrack_proto_udplite.c +++ b/net/netfilter/nf_conntrack_proto_udplite.c | |||
@@ -10,7 +10,6 @@ | |||
10 | #include <linux/types.h> | 10 | #include <linux/types.h> |
11 | #include <linux/timer.h> | 11 | #include <linux/timer.h> |
12 | #include <linux/module.h> | 12 | #include <linux/module.h> |
13 | #include <linux/netfilter.h> | ||
14 | #include <linux/udp.h> | 13 | #include <linux/udp.h> |
15 | #include <linux/seq_file.h> | 14 | #include <linux/seq_file.h> |
16 | #include <linux/skbuff.h> | 15 | #include <linux/skbuff.h> |
diff --git a/net/netfilter/xt_connlimit.c b/net/netfilter/xt_connlimit.c index 3335dd5be962..06cff1d13690 100644 --- a/net/netfilter/xt_connlimit.c +++ b/net/netfilter/xt_connlimit.c | |||
@@ -42,13 +42,13 @@ struct xt_connlimit_data { | |||
42 | static u_int32_t connlimit_rnd; | 42 | static u_int32_t connlimit_rnd; |
43 | static bool connlimit_rnd_inited; | 43 | static bool connlimit_rnd_inited; |
44 | 44 | ||
45 | static inline unsigned int connlimit_iphash(u_int32_t addr) | 45 | static inline unsigned int connlimit_iphash(__be32 addr) |
46 | { | 46 | { |
47 | if (unlikely(!connlimit_rnd_inited)) { | 47 | if (unlikely(!connlimit_rnd_inited)) { |
48 | get_random_bytes(&connlimit_rnd, sizeof(connlimit_rnd)); | 48 | get_random_bytes(&connlimit_rnd, sizeof(connlimit_rnd)); |
49 | connlimit_rnd_inited = true; | 49 | connlimit_rnd_inited = true; |
50 | } | 50 | } |
51 | return jhash_1word(addr, connlimit_rnd) & 0xFF; | 51 | return jhash_1word((__force __u32)addr, connlimit_rnd) & 0xFF; |
52 | } | 52 | } |
53 | 53 | ||
54 | static inline unsigned int | 54 | static inline unsigned int |
@@ -66,7 +66,7 @@ connlimit_iphash6(const union nf_conntrack_address *addr, | |||
66 | for (i = 0; i < ARRAY_SIZE(addr->ip6); ++i) | 66 | for (i = 0; i < ARRAY_SIZE(addr->ip6); ++i) |
67 | res.ip6[i] = addr->ip6[i] & mask->ip6[i]; | 67 | res.ip6[i] = addr->ip6[i] & mask->ip6[i]; |
68 | 68 | ||
69 | return jhash2(res.ip6, ARRAY_SIZE(res.ip6), connlimit_rnd) & 0xFF; | 69 | return jhash2((u32 *)res.ip6, ARRAY_SIZE(res.ip6), connlimit_rnd) & 0xFF; |
70 | } | 70 | } |
71 | 71 | ||
72 | static inline bool already_closed(const struct nf_conn *conn) | 72 | static inline bool already_closed(const struct nf_conn *conn) |
diff --git a/net/netfilter/xt_physdev.c b/net/netfilter/xt_physdev.c index f47cab7a696d..a4bab043a6d1 100644 --- a/net/netfilter/xt_physdev.c +++ b/net/netfilter/xt_physdev.c | |||
@@ -13,7 +13,6 @@ | |||
13 | #include <linux/netfilter_bridge.h> | 13 | #include <linux/netfilter_bridge.h> |
14 | #include <linux/netfilter/xt_physdev.h> | 14 | #include <linux/netfilter/xt_physdev.h> |
15 | #include <linux/netfilter/x_tables.h> | 15 | #include <linux/netfilter/x_tables.h> |
16 | #include <linux/netfilter_bridge.h> | ||
17 | 16 | ||
18 | MODULE_LICENSE("GPL"); | 17 | MODULE_LICENSE("GPL"); |
19 | MODULE_AUTHOR("Bart De Schuymer <bdschuym@pandora.be>"); | 18 | MODULE_AUTHOR("Bart De Schuymer <bdschuym@pandora.be>"); |
diff --git a/net/netfilter/xt_u32.c b/net/netfilter/xt_u32.c index 04b677ae8dae..74f9b14c012f 100644 --- a/net/netfilter/xt_u32.c +++ b/net/netfilter/xt_u32.c | |||
@@ -21,6 +21,7 @@ static bool u32_match_it(const struct xt_u32 *data, | |||
21 | unsigned int nnums; | 21 | unsigned int nnums; |
22 | unsigned int nvals; | 22 | unsigned int nvals; |
23 | unsigned int i; | 23 | unsigned int i; |
24 | __be32 n; | ||
24 | u_int32_t pos; | 25 | u_int32_t pos; |
25 | u_int32_t val; | 26 | u_int32_t val; |
26 | u_int32_t at; | 27 | u_int32_t at; |
@@ -38,9 +39,9 @@ static bool u32_match_it(const struct xt_u32 *data, | |||
38 | if (skb->len < 4 || pos > skb->len - 4); | 39 | if (skb->len < 4 || pos > skb->len - 4); |
39 | return false; | 40 | return false; |
40 | 41 | ||
41 | ret = skb_copy_bits(skb, pos, &val, sizeof(val)); | 42 | ret = skb_copy_bits(skb, pos, &n, sizeof(n)); |
42 | BUG_ON(ret < 0); | 43 | BUG_ON(ret < 0); |
43 | val = ntohl(val); | 44 | val = ntohl(n); |
44 | nnums = ct->nnums; | 45 | nnums = ct->nnums; |
45 | 46 | ||
46 | /* Inner loop runs over "&", "<<", ">>" and "@" operands */ | 47 | /* Inner loop runs over "&", "<<", ">>" and "@" operands */ |
@@ -65,10 +66,10 @@ static bool u32_match_it(const struct xt_u32 *data, | |||
65 | pos > skb->len - at - 4) | 66 | pos > skb->len - at - 4) |
66 | return false; | 67 | return false; |
67 | 68 | ||
68 | ret = skb_copy_bits(skb, at + pos, &val, | 69 | ret = skb_copy_bits(skb, at + pos, &n, |
69 | sizeof(val)); | 70 | sizeof(n)); |
70 | BUG_ON(ret < 0); | 71 | BUG_ON(ret < 0); |
71 | val = ntohl(val); | 72 | val = ntohl(n); |
72 | break; | 73 | break; |
73 | } | 74 | } |
74 | } | 75 | } |
diff --git a/net/netlink/genetlink.c b/net/netlink/genetlink.c index e146531faf1d..8c11ca4a2121 100644 --- a/net/netlink/genetlink.c +++ b/net/netlink/genetlink.c | |||
@@ -184,7 +184,7 @@ int genl_register_mc_group(struct genl_family *family, | |||
184 | } | 184 | } |
185 | 185 | ||
186 | err = netlink_change_ngroups(genl_sock, | 186 | err = netlink_change_ngroups(genl_sock, |
187 | sizeof(unsigned long) * NETLINK_GENERIC); | 187 | mc_groups_longs * BITS_PER_LONG); |
188 | if (err) | 188 | if (err) |
189 | goto out; | 189 | goto out; |
190 | 190 | ||
@@ -196,10 +196,22 @@ int genl_register_mc_group(struct genl_family *family, | |||
196 | genl_ctrl_event(CTRL_CMD_NEWMCAST_GRP, grp); | 196 | genl_ctrl_event(CTRL_CMD_NEWMCAST_GRP, grp); |
197 | out: | 197 | out: |
198 | genl_unlock(); | 198 | genl_unlock(); |
199 | return 0; | 199 | return err; |
200 | } | 200 | } |
201 | EXPORT_SYMBOL(genl_register_mc_group); | 201 | EXPORT_SYMBOL(genl_register_mc_group); |
202 | 202 | ||
203 | static void __genl_unregister_mc_group(struct genl_family *family, | ||
204 | struct genl_multicast_group *grp) | ||
205 | { | ||
206 | BUG_ON(grp->family != family); | ||
207 | netlink_clear_multicast_users(genl_sock, grp->id); | ||
208 | clear_bit(grp->id, mc_groups); | ||
209 | list_del(&grp->list); | ||
210 | genl_ctrl_event(CTRL_CMD_DELMCAST_GRP, grp); | ||
211 | grp->id = 0; | ||
212 | grp->family = NULL; | ||
213 | } | ||
214 | |||
203 | /** | 215 | /** |
204 | * genl_unregister_mc_group - unregister a multicast group | 216 | * genl_unregister_mc_group - unregister a multicast group |
205 | * | 217 | * |
@@ -217,14 +229,8 @@ EXPORT_SYMBOL(genl_register_mc_group); | |||
217 | void genl_unregister_mc_group(struct genl_family *family, | 229 | void genl_unregister_mc_group(struct genl_family *family, |
218 | struct genl_multicast_group *grp) | 230 | struct genl_multicast_group *grp) |
219 | { | 231 | { |
220 | BUG_ON(grp->family != family); | ||
221 | genl_lock(); | 232 | genl_lock(); |
222 | netlink_clear_multicast_users(genl_sock, grp->id); | 233 | __genl_unregister_mc_group(family, grp); |
223 | clear_bit(grp->id, mc_groups); | ||
224 | list_del(&grp->list); | ||
225 | genl_ctrl_event(CTRL_CMD_DELMCAST_GRP, grp); | ||
226 | grp->id = 0; | ||
227 | grp->family = NULL; | ||
228 | genl_unlock(); | 234 | genl_unlock(); |
229 | } | 235 | } |
230 | 236 | ||
@@ -232,8 +238,10 @@ static void genl_unregister_mc_groups(struct genl_family *family) | |||
232 | { | 238 | { |
233 | struct genl_multicast_group *grp, *tmp; | 239 | struct genl_multicast_group *grp, *tmp; |
234 | 240 | ||
241 | genl_lock(); | ||
235 | list_for_each_entry_safe(grp, tmp, &family->mcast_groups, list) | 242 | list_for_each_entry_safe(grp, tmp, &family->mcast_groups, list) |
236 | genl_unregister_mc_group(family, grp); | 243 | __genl_unregister_mc_group(family, grp); |
244 | genl_unlock(); | ||
237 | } | 245 | } |
238 | 246 | ||
239 | /** | 247 | /** |
diff --git a/net/rxrpc/af_rxrpc.c b/net/rxrpc/af_rxrpc.c index 16a68df4e36b..c58fa0d1be26 100644 --- a/net/rxrpc/af_rxrpc.c +++ b/net/rxrpc/af_rxrpc.c | |||
@@ -787,7 +787,7 @@ static int __init af_rxrpc_init(void) | |||
787 | 787 | ||
788 | BUILD_BUG_ON(sizeof(struct rxrpc_skb_priv) > sizeof(dummy_skb->cb)); | 788 | BUILD_BUG_ON(sizeof(struct rxrpc_skb_priv) > sizeof(dummy_skb->cb)); |
789 | 789 | ||
790 | rxrpc_epoch = htonl(xtime.tv_sec); | 790 | rxrpc_epoch = htonl(get_seconds()); |
791 | 791 | ||
792 | ret = -ENOMEM; | 792 | ret = -ENOMEM; |
793 | rxrpc_call_jar = kmem_cache_create( | 793 | rxrpc_call_jar = kmem_cache_create( |
diff --git a/net/rxrpc/ar-connection.c b/net/rxrpc/ar-connection.c index 482750efc235..d6667f7bc85e 100644 --- a/net/rxrpc/ar-connection.c +++ b/net/rxrpc/ar-connection.c | |||
@@ -71,7 +71,7 @@ struct rxrpc_conn_bundle *rxrpc_get_bundle(struct rxrpc_sock *rx, | |||
71 | struct rb_node *p, *parent, **pp; | 71 | struct rb_node *p, *parent, **pp; |
72 | 72 | ||
73 | _enter("%p{%x},%x,%hx,", | 73 | _enter("%p{%x},%x,%hx,", |
74 | rx, key_serial(key), trans->debug_id, ntohl(service_id)); | 74 | rx, key_serial(key), trans->debug_id, ntohs(service_id)); |
75 | 75 | ||
76 | if (rx->trans == trans && rx->bundle) { | 76 | if (rx->trans == trans && rx->bundle) { |
77 | atomic_inc(&rx->bundle->usage); | 77 | atomic_inc(&rx->bundle->usage); |
@@ -791,7 +791,7 @@ void rxrpc_put_connection(struct rxrpc_connection *conn) | |||
791 | 791 | ||
792 | ASSERTCMP(atomic_read(&conn->usage), >, 0); | 792 | ASSERTCMP(atomic_read(&conn->usage), >, 0); |
793 | 793 | ||
794 | conn->put_time = xtime.tv_sec; | 794 | conn->put_time = get_seconds(); |
795 | if (atomic_dec_and_test(&conn->usage)) { | 795 | if (atomic_dec_and_test(&conn->usage)) { |
796 | _debug("zombie"); | 796 | _debug("zombie"); |
797 | rxrpc_queue_delayed_work(&rxrpc_connection_reap, 0); | 797 | rxrpc_queue_delayed_work(&rxrpc_connection_reap, 0); |
@@ -835,7 +835,7 @@ void rxrpc_connection_reaper(struct work_struct *work) | |||
835 | 835 | ||
836 | _enter(""); | 836 | _enter(""); |
837 | 837 | ||
838 | now = xtime.tv_sec; | 838 | now = get_seconds(); |
839 | earliest = ULONG_MAX; | 839 | earliest = ULONG_MAX; |
840 | 840 | ||
841 | write_lock_bh(&rxrpc_connection_lock); | 841 | write_lock_bh(&rxrpc_connection_lock); |
diff --git a/net/rxrpc/ar-transport.c b/net/rxrpc/ar-transport.c index d43d78f19302..bb282a6a19f0 100644 --- a/net/rxrpc/ar-transport.c +++ b/net/rxrpc/ar-transport.c | |||
@@ -183,7 +183,7 @@ void rxrpc_put_transport(struct rxrpc_transport *trans) | |||
183 | 183 | ||
184 | ASSERTCMP(atomic_read(&trans->usage), >, 0); | 184 | ASSERTCMP(atomic_read(&trans->usage), >, 0); |
185 | 185 | ||
186 | trans->put_time = xtime.tv_sec; | 186 | trans->put_time = get_seconds(); |
187 | if (unlikely(atomic_dec_and_test(&trans->usage))) | 187 | if (unlikely(atomic_dec_and_test(&trans->usage))) |
188 | _debug("zombie"); | 188 | _debug("zombie"); |
189 | /* let the reaper determine the timeout to avoid a race with | 189 | /* let the reaper determine the timeout to avoid a race with |
@@ -219,7 +219,7 @@ static void rxrpc_transport_reaper(struct work_struct *work) | |||
219 | 219 | ||
220 | _enter(""); | 220 | _enter(""); |
221 | 221 | ||
222 | now = xtime.tv_sec; | 222 | now = get_seconds(); |
223 | earliest = ULONG_MAX; | 223 | earliest = ULONG_MAX; |
224 | 224 | ||
225 | /* extract all the transports that have been dead too long */ | 225 | /* extract all the transports that have been dead too long */ |
diff --git a/net/rxrpc/rxkad.c b/net/rxrpc/rxkad.c index 5ec705144e10..ac3cabdca78c 100644 --- a/net/rxrpc/rxkad.c +++ b/net/rxrpc/rxkad.c | |||
@@ -916,7 +916,7 @@ static int rxkad_decrypt_ticket(struct rxrpc_connection *conn, | |||
916 | issue = be32_to_cpu(stamp); | 916 | issue = be32_to_cpu(stamp); |
917 | } | 917 | } |
918 | p += 4; | 918 | p += 4; |
919 | now = xtime.tv_sec; | 919 | now = get_seconds(); |
920 | _debug("KIV ISSUE: %lx [%lx]", issue, now); | 920 | _debug("KIV ISSUE: %lx [%lx]", issue, now); |
921 | 921 | ||
922 | /* check the ticket is in date */ | 922 | /* check the ticket is in date */ |
diff --git a/net/sunrpc/svcsock.c b/net/sunrpc/svcsock.c index 64b9b8c743c4..12ff5da8160e 100644 --- a/net/sunrpc/svcsock.c +++ b/net/sunrpc/svcsock.c | |||
@@ -131,13 +131,13 @@ static char *__svc_print_addr(struct sockaddr *addr, char *buf, size_t len) | |||
131 | case AF_INET: | 131 | case AF_INET: |
132 | snprintf(buf, len, "%u.%u.%u.%u, port=%u", | 132 | snprintf(buf, len, "%u.%u.%u.%u, port=%u", |
133 | NIPQUAD(((struct sockaddr_in *) addr)->sin_addr), | 133 | NIPQUAD(((struct sockaddr_in *) addr)->sin_addr), |
134 | htons(((struct sockaddr_in *) addr)->sin_port)); | 134 | ntohs(((struct sockaddr_in *) addr)->sin_port)); |
135 | break; | 135 | break; |
136 | 136 | ||
137 | case AF_INET6: | 137 | case AF_INET6: |
138 | snprintf(buf, len, "%x:%x:%x:%x:%x:%x:%x:%x, port=%u", | 138 | snprintf(buf, len, "%x:%x:%x:%x:%x:%x:%x:%x, port=%u", |
139 | NIP6(((struct sockaddr_in6 *) addr)->sin6_addr), | 139 | NIP6(((struct sockaddr_in6 *) addr)->sin6_addr), |
140 | htons(((struct sockaddr_in6 *) addr)->sin6_port)); | 140 | ntohs(((struct sockaddr_in6 *) addr)->sin6_port)); |
141 | break; | 141 | break; |
142 | 142 | ||
143 | default: | 143 | default: |
diff --git a/net/tipc/msg.h b/net/tipc/msg.h index 35d5ba1d4f42..ce2659836374 100644 --- a/net/tipc/msg.h +++ b/net/tipc/msg.h | |||
@@ -72,10 +72,8 @@ static inline void msg_set_bits(struct tipc_msg *m, u32 w, | |||
72 | u32 pos, u32 mask, u32 val) | 72 | u32 pos, u32 mask, u32 val) |
73 | { | 73 | { |
74 | val = (val & mask) << pos; | 74 | val = (val & mask) << pos; |
75 | val = htonl(val); | 75 | m->hdr[w] &= ~htonl(mask << pos); |
76 | mask = htonl(mask << pos); | 76 | m->hdr[w] |= htonl(val); |
77 | m->hdr[w] &= ~mask; | ||
78 | m->hdr[w] |= val; | ||
79 | } | 77 | } |
80 | 78 | ||
81 | /* | 79 | /* |
diff --git a/scripts/Makefile.build b/scripts/Makefile.build index 3f7b451f3955..7fd6055bedfd 100644 --- a/scripts/Makefile.build +++ b/scripts/Makefile.build | |||
@@ -315,7 +315,7 @@ quiet_cmd_link_multi-y = LD $@ | |||
315 | cmd_link_multi-y = $(LD) $(ld_flags) -r -o $@ $(link_multi_deps) | 315 | cmd_link_multi-y = $(LD) $(ld_flags) -r -o $@ $(link_multi_deps) |
316 | 316 | ||
317 | quiet_cmd_link_multi-m = LD [M] $@ | 317 | quiet_cmd_link_multi-m = LD [M] $@ |
318 | cmd_link_multi-m = $(LD) $(ld_flags) $(LDFLAGS_MODULE) -o $@ $(link_multi_deps) | 318 | cmd_link_multi-m = $(cmd_link_multi-y) |
319 | 319 | ||
320 | # We would rather have a list of rules like | 320 | # We would rather have a list of rules like |
321 | # foo.o: $(foo-objs) | 321 | # foo.o: $(foo-objs) |
diff --git a/scripts/Makefile.modpost b/scripts/Makefile.modpost index c6fcc597b3be..d988f5d21e3d 100644 --- a/scripts/Makefile.modpost +++ b/scripts/Makefile.modpost | |||
@@ -56,23 +56,24 @@ _modpost: $(if $(KBUILD_MODPOST_NOFINAL), $(modules:.ko:.o),$(modules)) | |||
56 | 56 | ||
57 | # Step 2), invoke modpost | 57 | # Step 2), invoke modpost |
58 | # Includes step 3,4 | 58 | # Includes step 3,4 |
59 | modpost = scripts/mod/modpost \ | ||
60 | $(if $(CONFIG_MODVERSIONS),-m) \ | ||
61 | $(if $(CONFIG_MODULE_SRCVERSION_ALL),-a,) \ | ||
62 | $(if $(KBUILD_EXTMOD),-i,-o) $(kernelsymfile) \ | ||
63 | $(if $(KBUILD_EXTMOD),-I $(modulesymfile)) \ | ||
64 | $(if $(KBUILD_EXTMOD),-o $(modulesymfile)) \ | ||
65 | $(if $(KBUILD_EXTMOD)$(KBUILD_MODPOST_WARN),-w) | ||
66 | |||
59 | quiet_cmd_modpost = MODPOST $(words $(filter-out vmlinux FORCE, $^)) modules | 67 | quiet_cmd_modpost = MODPOST $(words $(filter-out vmlinux FORCE, $^)) modules |
60 | cmd_modpost = scripts/mod/modpost \ | 68 | cmd_modpost = $(modpost) -s |
61 | $(if $(CONFIG_MODVERSIONS),-m) \ | ||
62 | $(if $(CONFIG_MODULE_SRCVERSION_ALL),-a,) \ | ||
63 | $(if $(KBUILD_EXTMOD),-i,-o) $(kernelsymfile) \ | ||
64 | $(if $(KBUILD_EXTMOD),-I $(modulesymfile)) \ | ||
65 | $(if $(KBUILD_EXTMOD),-o $(modulesymfile)) \ | ||
66 | $(if $(KBUILD_EXTMOD)$(KBUILD_MODPOST_WARN),-w) | ||
67 | 69 | ||
68 | PHONY += __modpost | 70 | PHONY += __modpost |
69 | __modpost: $(modules:.ko=.o) FORCE | 71 | __modpost: $(modules:.ko=.o) FORCE |
70 | $(call cmd,modpost) $(wildcard vmlinux) $(filter-out FORCE,$^) | 72 | $(call cmd,modpost) $(wildcard vmlinux) $(filter-out FORCE,$^) |
71 | 73 | ||
72 | quiet_cmd_kernel-mod = MODPOST $@ | 74 | quiet_cmd_kernel-mod = MODPOST $@ |
73 | cmd_kernel-mod = $(cmd_modpost) $@ | 75 | cmd_kernel-mod = $(modpost) $@ |
74 | 76 | ||
75 | PHONY += vmlinux | ||
76 | vmlinux.o: FORCE | 77 | vmlinux.o: FORCE |
77 | $(call cmd,kernel-mod) | 78 | $(call cmd,kernel-mod) |
78 | 79 | ||
@@ -97,7 +98,7 @@ targets += $(modules:.ko=.mod.o) | |||
97 | 98 | ||
98 | # Step 6), final link of the modules | 99 | # Step 6), final link of the modules |
99 | quiet_cmd_ld_ko_o = LD [M] $@ | 100 | quiet_cmd_ld_ko_o = LD [M] $@ |
100 | cmd_ld_ko_o = $(LD) $(LDFLAGS) $(LDFLAGS_MODULE) -o $@ \ | 101 | cmd_ld_ko_o = $(LD) -r $(LDFLAGS) $(LDFLAGS_MODULE) -o $@ \ |
101 | $(filter-out FORCE,$^) | 102 | $(filter-out FORCE,$^) |
102 | 103 | ||
103 | $(modules): %.ko :%.o %.mod.o FORCE | 104 | $(modules): %.ko :%.o %.mod.o FORCE |
diff --git a/scripts/kconfig/conf.c b/scripts/kconfig/conf.c index 1199baf866ca..8be6a4269e63 100644 --- a/scripts/kconfig/conf.c +++ b/scripts/kconfig/conf.c | |||
@@ -37,6 +37,14 @@ static struct menu *rootEntry; | |||
37 | 37 | ||
38 | static char nohelp_text[] = N_("Sorry, no help available for this option yet.\n"); | 38 | static char nohelp_text[] = N_("Sorry, no help available for this option yet.\n"); |
39 | 39 | ||
40 | static const char *get_help(struct menu *menu) | ||
41 | { | ||
42 | if (menu_has_help(menu)) | ||
43 | return menu_get_help(menu); | ||
44 | else | ||
45 | return nohelp_text; | ||
46 | } | ||
47 | |||
40 | static void strip(char *str) | 48 | static void strip(char *str) |
41 | { | 49 | { |
42 | char *p = str; | 50 | char *p = str; |
@@ -171,7 +179,7 @@ static void conf_askvalue(struct symbol *sym, const char *def) | |||
171 | int conf_string(struct menu *menu) | 179 | int conf_string(struct menu *menu) |
172 | { | 180 | { |
173 | struct symbol *sym = menu->sym; | 181 | struct symbol *sym = menu->sym; |
174 | const char *def, *help; | 182 | const char *def; |
175 | 183 | ||
176 | while (1) { | 184 | while (1) { |
177 | printf("%*s%s ", indent - 1, "", menu->prompt->text); | 185 | printf("%*s%s ", indent - 1, "", menu->prompt->text); |
@@ -186,10 +194,7 @@ int conf_string(struct menu *menu) | |||
186 | case '?': | 194 | case '?': |
187 | /* print help */ | 195 | /* print help */ |
188 | if (line[1] == '\n') { | 196 | if (line[1] == '\n') { |
189 | help = nohelp_text; | 197 | printf("\n%s\n", get_help(menu)); |
190 | if (menu->sym->help) | ||
191 | help = menu->sym->help; | ||
192 | printf("\n%s\n", menu->sym->help); | ||
193 | def = NULL; | 198 | def = NULL; |
194 | break; | 199 | break; |
195 | } | 200 | } |
@@ -207,7 +212,6 @@ static int conf_sym(struct menu *menu) | |||
207 | struct symbol *sym = menu->sym; | 212 | struct symbol *sym = menu->sym; |
208 | int type; | 213 | int type; |
209 | tristate oldval, newval; | 214 | tristate oldval, newval; |
210 | const char *help; | ||
211 | 215 | ||
212 | while (1) { | 216 | while (1) { |
213 | printf("%*s%s ", indent - 1, "", menu->prompt->text); | 217 | printf("%*s%s ", indent - 1, "", menu->prompt->text); |
@@ -233,7 +237,7 @@ static int conf_sym(struct menu *menu) | |||
233 | printf("/m"); | 237 | printf("/m"); |
234 | if (oldval != yes && sym_tristate_within_range(sym, yes)) | 238 | if (oldval != yes && sym_tristate_within_range(sym, yes)) |
235 | printf("/y"); | 239 | printf("/y"); |
236 | if (sym->help) | 240 | if (menu_has_help(menu)) |
237 | printf("/?"); | 241 | printf("/?"); |
238 | printf("] "); | 242 | printf("] "); |
239 | conf_askvalue(sym, sym_get_string_value(sym)); | 243 | conf_askvalue(sym, sym_get_string_value(sym)); |
@@ -269,10 +273,7 @@ static int conf_sym(struct menu *menu) | |||
269 | if (sym_set_tristate_value(sym, newval)) | 273 | if (sym_set_tristate_value(sym, newval)) |
270 | return 0; | 274 | return 0; |
271 | help: | 275 | help: |
272 | help = nohelp_text; | 276 | printf("\n%s\n", get_help(menu)); |
273 | if (sym->help) | ||
274 | help = sym->help; | ||
275 | printf("\n%s\n", help); | ||
276 | } | 277 | } |
277 | } | 278 | } |
278 | 279 | ||
@@ -342,7 +343,7 @@ static int conf_choice(struct menu *menu) | |||
342 | goto conf_childs; | 343 | goto conf_childs; |
343 | } | 344 | } |
344 | printf("[1-%d", cnt); | 345 | printf("[1-%d", cnt); |
345 | if (sym->help) | 346 | if (menu_has_help(menu)) |
346 | printf("?"); | 347 | printf("?"); |
347 | printf("]: "); | 348 | printf("]: "); |
348 | switch (input_mode) { | 349 | switch (input_mode) { |
@@ -359,8 +360,7 @@ static int conf_choice(struct menu *menu) | |||
359 | fgets(line, 128, stdin); | 360 | fgets(line, 128, stdin); |
360 | strip(line); | 361 | strip(line); |
361 | if (line[0] == '?') { | 362 | if (line[0] == '?') { |
362 | printf("\n%s\n", menu->sym->help ? | 363 | printf("\n%s\n", get_help(menu)); |
363 | menu->sym->help : nohelp_text); | ||
364 | continue; | 364 | continue; |
365 | } | 365 | } |
366 | if (!line[0]) | 366 | if (!line[0]) |
@@ -391,8 +391,7 @@ static int conf_choice(struct menu *menu) | |||
391 | if (!child) | 391 | if (!child) |
392 | continue; | 392 | continue; |
393 | if (line[strlen(line) - 1] == '?') { | 393 | if (line[strlen(line) - 1] == '?') { |
394 | printf("\n%s\n", child->sym->help ? | 394 | printf("\n%s\n", get_help(child)); |
395 | child->sym->help : nohelp_text); | ||
396 | continue; | 395 | continue; |
397 | } | 396 | } |
398 | sym_set_choice_value(sym, child->sym); | 397 | sym_set_choice_value(sym, child->sym); |
diff --git a/scripts/kconfig/expr.h b/scripts/kconfig/expr.h index 6084525f604b..a195986eec6f 100644 --- a/scripts/kconfig/expr.h +++ b/scripts/kconfig/expr.h | |||
@@ -71,14 +71,12 @@ enum { | |||
71 | struct symbol { | 71 | struct symbol { |
72 | struct symbol *next; | 72 | struct symbol *next; |
73 | char *name; | 73 | char *name; |
74 | char *help; | ||
75 | enum symbol_type type; | 74 | enum symbol_type type; |
76 | struct symbol_value curr; | 75 | struct symbol_value curr; |
77 | struct symbol_value def[4]; | 76 | struct symbol_value def[4]; |
78 | tristate visible; | 77 | tristate visible; |
79 | int flags; | 78 | int flags; |
80 | struct property *prop; | 79 | struct property *prop; |
81 | struct expr *dep, *dep2; | ||
82 | struct expr_value rev_dep; | 80 | struct expr_value rev_dep; |
83 | }; | 81 | }; |
84 | 82 | ||
@@ -139,7 +137,7 @@ struct menu { | |||
139 | struct property *prompt; | 137 | struct property *prompt; |
140 | struct expr *dep; | 138 | struct expr *dep; |
141 | unsigned int flags; | 139 | unsigned int flags; |
142 | //char *help; | 140 | char *help; |
143 | struct file *file; | 141 | struct file *file; |
144 | int lineno; | 142 | int lineno; |
145 | void *data; | 143 | void *data; |
diff --git a/scripts/kconfig/gconf.c b/scripts/kconfig/gconf.c index 61d8166166ef..262908cfc2ac 100644 --- a/scripts/kconfig/gconf.c +++ b/scripts/kconfig/gconf.c | |||
@@ -38,9 +38,6 @@ static gboolean show_all = FALSE; | |||
38 | static gboolean show_debug = FALSE; | 38 | static gboolean show_debug = FALSE; |
39 | static gboolean resizeable = FALSE; | 39 | static gboolean resizeable = FALSE; |
40 | 40 | ||
41 | static char nohelp_text[] = | ||
42 | N_("Sorry, no help available for this option yet.\n"); | ||
43 | |||
44 | GtkWidget *main_wnd = NULL; | 41 | GtkWidget *main_wnd = NULL; |
45 | GtkWidget *tree1_w = NULL; // left frame | 42 | GtkWidget *tree1_w = NULL; // left frame |
46 | GtkWidget *tree2_w = NULL; // right frame | 43 | GtkWidget *tree2_w = NULL; // right frame |
@@ -462,12 +459,9 @@ static void text_insert_help(struct menu *menu) | |||
462 | GtkTextIter start, end; | 459 | GtkTextIter start, end; |
463 | const char *prompt = menu_get_prompt(menu); | 460 | const char *prompt = menu_get_prompt(menu); |
464 | gchar *name; | 461 | gchar *name; |
465 | const char *help = _(nohelp_text); | 462 | const char *help; |
466 | 463 | ||
467 | if (!menu->sym) | 464 | help = _(menu_get_help(menu)); |
468 | help = ""; | ||
469 | else if (menu->sym->help) | ||
470 | help = _(menu->sym->help); | ||
471 | 465 | ||
472 | if (menu->sym && menu->sym->name) | 466 | if (menu->sym && menu->sym->name) |
473 | name = g_strdup_printf(_(menu->sym->name)); | 467 | name = g_strdup_printf(_(menu->sym->name)); |
diff --git a/scripts/kconfig/kxgettext.c b/scripts/kconfig/kxgettext.c index 11f7dab94715..6eb72a7f2562 100644 --- a/scripts/kconfig/kxgettext.c +++ b/scripts/kconfig/kxgettext.c | |||
@@ -170,8 +170,8 @@ void menu_build_message_list(struct menu *menu) | |||
170 | menu->file == NULL ? "Root Menu" : menu->file->name, | 170 | menu->file == NULL ? "Root Menu" : menu->file->name, |
171 | menu->lineno); | 171 | menu->lineno); |
172 | 172 | ||
173 | if (menu->sym != NULL && menu->sym->help != NULL) | 173 | if (menu->sym != NULL && menu_has_help(menu)) |
174 | message__add(menu->sym->help, menu->sym->name, | 174 | message__add(menu_get_help(menu), menu->sym->name, |
175 | menu->file == NULL ? "Root Menu" : menu->file->name, | 175 | menu->file == NULL ? "Root Menu" : menu->file->name, |
176 | menu->lineno); | 176 | menu->lineno); |
177 | 177 | ||
diff --git a/scripts/kconfig/lkc_proto.h b/scripts/kconfig/lkc_proto.h index 15030770d1ad..4d09f6ddefe3 100644 --- a/scripts/kconfig/lkc_proto.h +++ b/scripts/kconfig/lkc_proto.h | |||
@@ -15,6 +15,8 @@ P(menu_is_visible,bool,(struct menu *menu)); | |||
15 | P(menu_get_prompt,const char *,(struct menu *menu)); | 15 | P(menu_get_prompt,const char *,(struct menu *menu)); |
16 | P(menu_get_root_menu,struct menu *,(struct menu *menu)); | 16 | P(menu_get_root_menu,struct menu *,(struct menu *menu)); |
17 | P(menu_get_parent_menu,struct menu *,(struct menu *menu)); | 17 | P(menu_get_parent_menu,struct menu *,(struct menu *menu)); |
18 | P(menu_has_help,bool,(struct menu *menu)); | ||
19 | P(menu_get_help,const char *,(struct menu *menu)); | ||
18 | 20 | ||
19 | /* symbol.c */ | 21 | /* symbol.c */ |
20 | P(symbol_hash,struct symbol *,[SYMBOL_HASHSIZE]); | 22 | P(symbol_hash,struct symbol *,[SYMBOL_HASHSIZE]); |
diff --git a/scripts/kconfig/mconf.c b/scripts/kconfig/mconf.c index d2c2a429887b..bc5854ed6055 100644 --- a/scripts/kconfig/mconf.c +++ b/scripts/kconfig/mconf.c | |||
@@ -725,11 +725,11 @@ static void show_help(struct menu *menu) | |||
725 | struct gstr help = str_new(); | 725 | struct gstr help = str_new(); |
726 | struct symbol *sym = menu->sym; | 726 | struct symbol *sym = menu->sym; |
727 | 727 | ||
728 | if (sym->help) | 728 | if (menu_has_help(menu)) |
729 | { | 729 | { |
730 | if (sym->name) { | 730 | if (sym->name) { |
731 | str_printf(&help, "CONFIG_%s:\n\n", sym->name); | 731 | str_printf(&help, "CONFIG_%s:\n\n", sym->name); |
732 | str_append(&help, _(sym->help)); | 732 | str_append(&help, _(menu_get_help(menu))); |
733 | str_append(&help, "\n"); | 733 | str_append(&help, "\n"); |
734 | } | 734 | } |
735 | } else { | 735 | } else { |
diff --git a/scripts/kconfig/menu.c b/scripts/kconfig/menu.c index f14aeac67d4f..f9d0d91a3fe4 100644 --- a/scripts/kconfig/menu.c +++ b/scripts/kconfig/menu.c | |||
@@ -417,3 +417,15 @@ struct menu *menu_get_parent_menu(struct menu *menu) | |||
417 | return menu; | 417 | return menu; |
418 | } | 418 | } |
419 | 419 | ||
420 | bool menu_has_help(struct menu *menu) | ||
421 | { | ||
422 | return menu->help != NULL; | ||
423 | } | ||
424 | |||
425 | const char *menu_get_help(struct menu *menu) | ||
426 | { | ||
427 | if (menu->help) | ||
428 | return menu->help; | ||
429 | else | ||
430 | return ""; | ||
431 | } | ||
diff --git a/scripts/kconfig/qconf.cc b/scripts/kconfig/qconf.cc index f2a23a9c3938..e4eeb59a8c24 100644 --- a/scripts/kconfig/qconf.cc +++ b/scripts/kconfig/qconf.cc | |||
@@ -1041,7 +1041,7 @@ void ConfigInfoView::menuInfo(void) | |||
1041 | if (showDebug()) | 1041 | if (showDebug()) |
1042 | debug = debug_info(sym); | 1042 | debug = debug_info(sym); |
1043 | 1043 | ||
1044 | help = print_filter(_(sym->help)); | 1044 | help = print_filter(_(menu_get_help(menu))); |
1045 | } else if (menu->prompt) { | 1045 | } else if (menu->prompt) { |
1046 | head += "<big><b>"; | 1046 | head += "<big><b>"; |
1047 | head += print_filter(_(menu->prompt->text)); | 1047 | head += print_filter(_(menu->prompt->text)); |
diff --git a/scripts/kconfig/zconf.tab.c_shipped b/scripts/kconfig/zconf.tab.c_shipped index 9a06b6771eee..ec21db77f78b 100644 --- a/scripts/kconfig/zconf.tab.c_shipped +++ b/scripts/kconfig/zconf.tab.c_shipped | |||
@@ -1722,7 +1722,7 @@ yyreduce: | |||
1722 | case 83: | 1722 | case 83: |
1723 | 1723 | ||
1724 | { | 1724 | { |
1725 | current_entry->sym->help = (yyvsp[0].string); | 1725 | current_entry->help = (yyvsp[0].string); |
1726 | ;} | 1726 | ;} |
1727 | break; | 1727 | break; |
1728 | 1728 | ||
@@ -2280,11 +2280,11 @@ void print_symbol(FILE *out, struct menu *menu) | |||
2280 | break; | 2280 | break; |
2281 | } | 2281 | } |
2282 | } | 2282 | } |
2283 | if (sym->help) { | 2283 | if (menu->help) { |
2284 | int len = strlen(sym->help); | 2284 | int len = strlen(menu->help); |
2285 | while (sym->help[--len] == '\n') | 2285 | while (menu->help[--len] == '\n') |
2286 | sym->help[len] = 0; | 2286 | menu->help[len] = 0; |
2287 | fprintf(out, " help\n%s\n", sym->help); | 2287 | fprintf(out, " help\n%s\n", menu->help); |
2288 | } | 2288 | } |
2289 | fputc('\n', out); | 2289 | fputc('\n', out); |
2290 | } | 2290 | } |
diff --git a/scripts/kconfig/zconf.y b/scripts/kconfig/zconf.y index 92eb02bdf9c5..79db4cf22a51 100644 --- a/scripts/kconfig/zconf.y +++ b/scripts/kconfig/zconf.y | |||
@@ -402,7 +402,7 @@ help_start: T_HELP T_EOL | |||
402 | 402 | ||
403 | help: help_start T_HELPTEXT | 403 | help: help_start T_HELPTEXT |
404 | { | 404 | { |
405 | current_entry->sym->help = $2; | 405 | current_entry->help = $2; |
406 | }; | 406 | }; |
407 | 407 | ||
408 | /* depends option */ | 408 | /* depends option */ |
@@ -649,11 +649,11 @@ void print_symbol(FILE *out, struct menu *menu) | |||
649 | break; | 649 | break; |
650 | } | 650 | } |
651 | } | 651 | } |
652 | if (sym->help) { | 652 | if (menu->help) { |
653 | int len = strlen(sym->help); | 653 | int len = strlen(menu->help); |
654 | while (sym->help[--len] == '\n') | 654 | while (menu->help[--len] == '\n') |
655 | sym->help[len] = 0; | 655 | menu->help[len] = 0; |
656 | fprintf(out, " help\n%s\n", sym->help); | 656 | fprintf(out, " help\n%s\n", menu->help); |
657 | } | 657 | } |
658 | fputc('\n', out); | 658 | fputc('\n', out); |
659 | } | 659 | } |
diff --git a/scripts/mod/file2alias.c b/scripts/mod/file2alias.c index f646381dc015..8a09021d8c59 100644 --- a/scripts/mod/file2alias.c +++ b/scripts/mod/file2alias.c | |||
@@ -290,6 +290,14 @@ static int do_serio_entry(const char *filename, | |||
290 | return 1; | 290 | return 1; |
291 | } | 291 | } |
292 | 292 | ||
293 | /* looks like: "acpi:ACPI0003 or acpi:PNP0C0B" or "acpi:LNXVIDEO" */ | ||
294 | static int do_acpi_entry(const char *filename, | ||
295 | struct acpi_device_id *id, char *alias) | ||
296 | { | ||
297 | sprintf(alias, "acpi*:%s:", id->id); | ||
298 | return 1; | ||
299 | } | ||
300 | |||
293 | /* looks like: "pnp:dD" */ | 301 | /* looks like: "pnp:dD" */ |
294 | static int do_pnp_entry(const char *filename, | 302 | static int do_pnp_entry(const char *filename, |
295 | struct pnp_device_id *id, char *alias) | 303 | struct pnp_device_id *id, char *alias) |
@@ -551,6 +559,10 @@ void handle_moddevtable(struct module *mod, struct elf_info *info, | |||
551 | do_table(symval, sym->st_size, | 559 | do_table(symval, sym->st_size, |
552 | sizeof(struct serio_device_id), "serio", | 560 | sizeof(struct serio_device_id), "serio", |
553 | do_serio_entry, mod); | 561 | do_serio_entry, mod); |
562 | else if (sym_is(symname, "__mod_acpi_device_table")) | ||
563 | do_table(symval, sym->st_size, | ||
564 | sizeof(struct acpi_device_id), "acpi", | ||
565 | do_acpi_entry, mod); | ||
554 | else if (sym_is(symname, "__mod_pnp_device_table")) | 566 | else if (sym_is(symname, "__mod_pnp_device_table")) |
555 | do_table(symval, sym->st_size, | 567 | do_table(symval, sym->st_size, |
556 | sizeof(struct pnp_device_id), "pnp", | 568 | sizeof(struct pnp_device_id), "pnp", |
diff --git a/scripts/mod/modpost.c b/scripts/mod/modpost.c index 5ab7914d30ef..ee58ded021d7 100644 --- a/scripts/mod/modpost.c +++ b/scripts/mod/modpost.c | |||
@@ -23,6 +23,8 @@ int have_vmlinux = 0; | |||
23 | static int all_versions = 0; | 23 | static int all_versions = 0; |
24 | /* If we are modposting external module set to 1 */ | 24 | /* If we are modposting external module set to 1 */ |
25 | static int external_module = 0; | 25 | static int external_module = 0; |
26 | /* Warn about section mismatch in vmlinux if set to 1 */ | ||
27 | static int vmlinux_section_warnings = 1; | ||
26 | /* Only warn about unresolved symbols */ | 28 | /* Only warn about unresolved symbols */ |
27 | static int warn_unresolved = 0; | 29 | static int warn_unresolved = 0; |
28 | /* How a symbol is exported */ | 30 | /* How a symbol is exported */ |
@@ -584,13 +586,61 @@ static int strrcmp(const char *s, const char *sub) | |||
584 | return memcmp(s + slen - sublen, sub, sublen); | 586 | return memcmp(s + slen - sublen, sub, sublen); |
585 | } | 587 | } |
586 | 588 | ||
589 | /* | ||
590 | * Functions used only during module init is marked __init and is stored in | ||
591 | * a .init.text section. Likewise data is marked __initdata and stored in | ||
592 | * a .init.data section. | ||
593 | * If this section is one of these sections return 1 | ||
594 | * See include/linux/init.h for the details | ||
595 | */ | ||
596 | static int init_section(const char *name) | ||
597 | { | ||
598 | if (strcmp(name, ".init") == 0) | ||
599 | return 1; | ||
600 | if (strncmp(name, ".init.", strlen(".init.")) == 0) | ||
601 | return 1; | ||
602 | return 0; | ||
603 | } | ||
604 | |||
605 | /* | ||
606 | * Functions used only during module exit is marked __exit and is stored in | ||
607 | * a .exit.text section. Likewise data is marked __exitdata and stored in | ||
608 | * a .exit.data section. | ||
609 | * If this section is one of these sections return 1 | ||
610 | * See include/linux/init.h for the details | ||
611 | **/ | ||
612 | static int exit_section(const char *name) | ||
613 | { | ||
614 | if (strcmp(name, ".exit.text") == 0) | ||
615 | return 1; | ||
616 | if (strcmp(name, ".exit.data") == 0) | ||
617 | return 1; | ||
618 | return 0; | ||
619 | |||
620 | } | ||
621 | |||
622 | /* | ||
623 | * Data sections are named like this: | ||
624 | * .data | .data.rel | .data.rel.* | ||
625 | * Return 1 if the specified section is a data section | ||
626 | */ | ||
627 | static int data_section(const char *name) | ||
628 | { | ||
629 | if ((strcmp(name, ".data") == 0) || | ||
630 | (strcmp(name, ".data.rel") == 0) || | ||
631 | (strncmp(name, ".data.rel.", strlen(".data.rel.")) == 0)) | ||
632 | return 1; | ||
633 | else | ||
634 | return 0; | ||
635 | } | ||
636 | |||
587 | /** | 637 | /** |
588 | * Whitelist to allow certain references to pass with no warning. | 638 | * Whitelist to allow certain references to pass with no warning. |
589 | * | 639 | * |
590 | * Pattern 0: | 640 | * Pattern 0: |
591 | * Do not warn if funtion/data are marked with __init_refok/__initdata_refok. | 641 | * Do not warn if funtion/data are marked with __init_refok/__initdata_refok. |
592 | * The pattern is identified by: | 642 | * The pattern is identified by: |
593 | * fromsec = .text.init.refok | .data.init.refok | 643 | * fromsec = .text.init.refok* | .data.init.refok* |
594 | * | 644 | * |
595 | * Pattern 1: | 645 | * Pattern 1: |
596 | * If a module parameter is declared __initdata and permissions=0 | 646 | * If a module parameter is declared __initdata and permissions=0 |
@@ -608,8 +658,8 @@ static int strrcmp(const char *s, const char *sub) | |||
608 | * These functions may often be marked __init and we do not want to | 658 | * These functions may often be marked __init and we do not want to |
609 | * warn here. | 659 | * warn here. |
610 | * the pattern is identified by: | 660 | * the pattern is identified by: |
611 | * tosec = .init.text | .exit.text | .init.data | 661 | * tosec = init or exit section |
612 | * fromsec = .data | .data.rel | .data.rel.* | 662 | * fromsec = data section |
613 | * atsym = *driver, *_template, *_sht, *_ops, *_probe, *probe_one, *_console, *_timer | 663 | * atsym = *driver, *_template, *_sht, *_ops, *_probe, *probe_one, *_console, *_timer |
614 | * | 664 | * |
615 | * Pattern 3: | 665 | * Pattern 3: |
@@ -625,12 +675,18 @@ static int strrcmp(const char *s, const char *sub) | |||
625 | * This pattern is identified by | 675 | * This pattern is identified by |
626 | * refsymname = __init_begin, _sinittext, _einittext | 676 | * refsymname = __init_begin, _sinittext, _einittext |
627 | * | 677 | * |
678 | * Pattern 5: | ||
679 | * Xtensa uses literal sections for constants that are accessed PC-relative. | ||
680 | * Literal sections may safely reference their text sections. | ||
681 | * (Note that the name for the literal section omits any trailing '.text') | ||
682 | * tosec = <section>[.text] | ||
683 | * fromsec = <section>.literal | ||
628 | **/ | 684 | **/ |
629 | static int secref_whitelist(const char *modname, const char *tosec, | 685 | static int secref_whitelist(const char *modname, const char *tosec, |
630 | const char *fromsec, const char *atsym, | 686 | const char *fromsec, const char *atsym, |
631 | const char *refsymname) | 687 | const char *refsymname) |
632 | { | 688 | { |
633 | int f1 = 1, f2 = 1; | 689 | int len; |
634 | const char **s; | 690 | const char **s; |
635 | const char *pat2sym[] = { | 691 | const char *pat2sym[] = { |
636 | "driver", | 692 | "driver", |
@@ -652,36 +708,21 @@ static int secref_whitelist(const char *modname, const char *tosec, | |||
652 | }; | 708 | }; |
653 | 709 | ||
654 | /* Check for pattern 0 */ | 710 | /* Check for pattern 0 */ |
655 | if ((strcmp(fromsec, ".text.init.refok") == 0) || | 711 | if ((strncmp(fromsec, ".text.init.refok", strlen(".text.init.refok")) == 0) || |
656 | (strcmp(fromsec, ".data.init.refok") == 0)) | 712 | (strncmp(fromsec, ".data.init.refok", strlen(".data.init.refok")) == 0)) |
657 | return 1; | 713 | return 1; |
658 | 714 | ||
659 | /* Check for pattern 1 */ | 715 | /* Check for pattern 1 */ |
660 | if (strcmp(tosec, ".init.data") != 0) | 716 | if ((strcmp(tosec, ".init.data") == 0) && |
661 | f1 = 0; | 717 | (strncmp(fromsec, ".data", strlen(".data")) == 0) && |
662 | if (strncmp(fromsec, ".data", strlen(".data")) != 0) | 718 | (strncmp(atsym, "__param", strlen("__param")) == 0)) |
663 | f1 = 0; | 719 | return 1; |
664 | if (strncmp(atsym, "__param", strlen("__param")) != 0) | ||
665 | f1 = 0; | ||
666 | |||
667 | if (f1) | ||
668 | return f1; | ||
669 | 720 | ||
670 | /* Check for pattern 2 */ | 721 | /* Check for pattern 2 */ |
671 | if ((strcmp(tosec, ".init.text") != 0) && | 722 | if ((init_section(tosec) || exit_section(tosec)) && data_section(fromsec)) |
672 | (strcmp(tosec, ".exit.text") != 0) && | 723 | for (s = pat2sym; *s; s++) |
673 | (strcmp(tosec, ".init.data") != 0)) | 724 | if (strrcmp(atsym, *s) == 0) |
674 | f2 = 0; | 725 | return 1; |
675 | if ((strcmp(fromsec, ".data") != 0) && | ||
676 | (strcmp(fromsec, ".data.rel") != 0) && | ||
677 | (strncmp(fromsec, ".data.rel.", strlen(".data.rel.")) != 0)) | ||
678 | f2 = 0; | ||
679 | |||
680 | for (s = pat2sym; *s; s++) | ||
681 | if (strrcmp(atsym, *s) == 0) | ||
682 | f1 = 1; | ||
683 | if (f1 && f2) | ||
684 | return 1; | ||
685 | 726 | ||
686 | /* Check for pattern 3 */ | 727 | /* Check for pattern 3 */ |
687 | if ((strcmp(fromsec, ".text.head") == 0) && | 728 | if ((strcmp(fromsec, ".text.head") == 0) && |
@@ -694,6 +735,15 @@ static int secref_whitelist(const char *modname, const char *tosec, | |||
694 | if (strcmp(refsymname, *s) == 0) | 735 | if (strcmp(refsymname, *s) == 0) |
695 | return 1; | 736 | return 1; |
696 | 737 | ||
738 | /* Check for pattern 5 */ | ||
739 | if (strrcmp(tosec, ".text") == 0) | ||
740 | len = strlen(tosec) - strlen(".text"); | ||
741 | else | ||
742 | len = strlen(tosec); | ||
743 | if ((strncmp(tosec, fromsec, len) == 0) && (strlen(fromsec) > len) && | ||
744 | (strcmp(fromsec + len, ".literal") == 0)) | ||
745 | return 1; | ||
746 | |||
697 | return 0; | 747 | return 0; |
698 | } | 748 | } |
699 | 749 | ||
@@ -822,9 +872,9 @@ static void warn_sec_mismatch(const char *modname, const char *fromsec, | |||
822 | refsymname = elf->strtab + refsym->st_name; | 872 | refsymname = elf->strtab + refsym->st_name; |
823 | 873 | ||
824 | /* check whitelist - we may ignore it */ | 874 | /* check whitelist - we may ignore it */ |
825 | if (before && | 875 | if (secref_whitelist(modname, secname, fromsec, |
826 | secref_whitelist(modname, secname, fromsec, | 876 | before ? elf->strtab + before->st_name : "", |
827 | elf->strtab + before->st_name, refsymname)) | 877 | refsymname)) |
828 | return; | 878 | return; |
829 | 879 | ||
830 | if (before && after) { | 880 | if (before && after) { |
@@ -1077,6 +1127,8 @@ static int initexit_section_ref_ok(const char *name) | |||
1077 | ".smp_locks", | 1127 | ".smp_locks", |
1078 | ".stab", | 1128 | ".stab", |
1079 | ".m68k_fixup", | 1129 | ".m68k_fixup", |
1130 | ".xt.prop", /* xtensa informational section */ | ||
1131 | ".xt.lit", /* xtensa informational section */ | ||
1080 | NULL | 1132 | NULL |
1081 | }; | 1133 | }; |
1082 | /* Start of section names */ | 1134 | /* Start of section names */ |
@@ -1106,21 +1158,6 @@ static int initexit_section_ref_ok(const char *name) | |||
1106 | return 0; | 1158 | return 0; |
1107 | } | 1159 | } |
1108 | 1160 | ||
1109 | /** | ||
1110 | * Functions used only during module init is marked __init and is stored in | ||
1111 | * a .init.text section. Likewise data is marked __initdata and stored in | ||
1112 | * a .init.data section. | ||
1113 | * If this section is one of these sections return 1 | ||
1114 | * See include/linux/init.h for the details | ||
1115 | **/ | ||
1116 | static int init_section(const char *name) | ||
1117 | { | ||
1118 | if (strcmp(name, ".init") == 0) | ||
1119 | return 1; | ||
1120 | if (strncmp(name, ".init.", strlen(".init.")) == 0) | ||
1121 | return 1; | ||
1122 | return 0; | ||
1123 | } | ||
1124 | 1161 | ||
1125 | /* | 1162 | /* |
1126 | * Identify sections from which references to a .init section is OK. | 1163 | * Identify sections from which references to a .init section is OK. |
@@ -1178,23 +1215,6 @@ static int init_section_ref_ok(const char *name) | |||
1178 | } | 1215 | } |
1179 | 1216 | ||
1180 | /* | 1217 | /* |
1181 | * Functions used only during module exit is marked __exit and is stored in | ||
1182 | * a .exit.text section. Likewise data is marked __exitdata and stored in | ||
1183 | * a .exit.data section. | ||
1184 | * If this section is one of these sections return 1 | ||
1185 | * See include/linux/init.h for the details | ||
1186 | **/ | ||
1187 | static int exit_section(const char *name) | ||
1188 | { | ||
1189 | if (strcmp(name, ".exit.text") == 0) | ||
1190 | return 1; | ||
1191 | if (strcmp(name, ".exit.data") == 0) | ||
1192 | return 1; | ||
1193 | return 0; | ||
1194 | |||
1195 | } | ||
1196 | |||
1197 | /* | ||
1198 | * Identify sections from which references to a .exit section is OK. | 1218 | * Identify sections from which references to a .exit section is OK. |
1199 | */ | 1219 | */ |
1200 | static int exit_section_ref_ok(const char *name) | 1220 | static int exit_section_ref_ok(const char *name) |
@@ -1257,8 +1277,10 @@ static void read_symbols(char *modname) | |||
1257 | handle_modversions(mod, &info, sym, symname); | 1277 | handle_modversions(mod, &info, sym, symname); |
1258 | handle_moddevtable(mod, &info, sym, symname); | 1278 | handle_moddevtable(mod, &info, sym, symname); |
1259 | } | 1279 | } |
1260 | check_sec_ref(mod, modname, &info, init_section, init_section_ref_ok); | 1280 | if (is_vmlinux(modname) && vmlinux_section_warnings) { |
1261 | check_sec_ref(mod, modname, &info, exit_section, exit_section_ref_ok); | 1281 | check_sec_ref(mod, modname, &info, init_section, init_section_ref_ok); |
1282 | check_sec_ref(mod, modname, &info, exit_section, exit_section_ref_ok); | ||
1283 | } | ||
1262 | 1284 | ||
1263 | version = get_modinfo(info.modinfo, info.modinfo_len, "version"); | 1285 | version = get_modinfo(info.modinfo, info.modinfo_len, "version"); |
1264 | if (version) | 1286 | if (version) |
@@ -1626,7 +1648,7 @@ int main(int argc, char **argv) | |||
1626 | int opt; | 1648 | int opt; |
1627 | int err; | 1649 | int err; |
1628 | 1650 | ||
1629 | while ((opt = getopt(argc, argv, "i:I:mo:aw")) != -1) { | 1651 | while ((opt = getopt(argc, argv, "i:I:mso:aw")) != -1) { |
1630 | switch(opt) { | 1652 | switch(opt) { |
1631 | case 'i': | 1653 | case 'i': |
1632 | kernel_read = optarg; | 1654 | kernel_read = optarg; |
@@ -1644,6 +1666,9 @@ int main(int argc, char **argv) | |||
1644 | case 'a': | 1666 | case 'a': |
1645 | all_versions = 1; | 1667 | all_versions = 1; |
1646 | break; | 1668 | break; |
1669 | case 's': | ||
1670 | vmlinux_section_warnings = 0; | ||
1671 | break; | ||
1647 | case 'w': | 1672 | case 'w': |
1648 | warn_unresolved = 1; | 1673 | warn_unresolved = 1; |
1649 | break; | 1674 | break; |
diff --git a/scripts/ver_linux b/scripts/ver_linux index 72876dfadc8a..8f8df93141a9 100755 --- a/scripts/ver_linux +++ b/scripts/ver_linux | |||
@@ -66,8 +66,8 @@ showmount --version 2>&1 | grep nfs-utils | awk \ | |||
66 | 'NR==1{print "nfs-utils ", $NF}' | 66 | 'NR==1{print "nfs-utils ", $NF}' |
67 | 67 | ||
68 | ls -l `ldd /bin/sh | awk '/libc/{print $3}'` | sed \ | 68 | ls -l `ldd /bin/sh | awk '/libc/{print $3}'` | sed \ |
69 | -e 's/\.so$//' | awk -F'[.-]' '{print "Linux C Library " \ | 69 | -e 's/\.so$//' | sed -e 's/>//' | \ |
70 | $(NF-2)"."$(NF-1)"."$NF}' | 70 | awk -F'[.-]' '{print "Linux C Library "$(NF-1)"."$NF}' |
71 | 71 | ||
72 | ldd -v > /dev/null 2>&1 && ldd -v || ldd --version |head -n 1 | awk \ | 72 | ldd -v > /dev/null 2>&1 && ldd -v || ldd --version |head -n 1 | awk \ |
73 | 'NR==1{print "Dynamic linker (ldd) ", $NF}' | 73 | 'NR==1{print "Dynamic linker (ldd) ", $NF}' |
diff --git a/security/selinux/ss/services.c b/security/selinux/ss/services.c index b5f017f07a75..0ae032f3876b 100644 --- a/security/selinux/ss/services.c +++ b/security/selinux/ss/services.c | |||
@@ -2417,8 +2417,10 @@ static void security_netlbl_cache_add(struct netlbl_lsm_secattr *secattr, | |||
2417 | 2417 | ||
2418 | cache->type = NETLBL_CACHE_T_MLS; | 2418 | cache->type = NETLBL_CACHE_T_MLS; |
2419 | if (ebitmap_cpy(&cache->data.mls_label.level[0].cat, | 2419 | if (ebitmap_cpy(&cache->data.mls_label.level[0].cat, |
2420 | &ctx->range.level[0].cat) != 0) | 2420 | &ctx->range.level[0].cat) != 0) { |
2421 | kfree(cache); | ||
2421 | return; | 2422 | return; |
2423 | } | ||
2422 | cache->data.mls_label.level[1].cat.highbit = | 2424 | cache->data.mls_label.level[1].cat.highbit = |
2423 | cache->data.mls_label.level[0].cat.highbit; | 2425 | cache->data.mls_label.level[0].cat.highbit; |
2424 | cache->data.mls_label.level[1].cat.node = | 2426 | cache->data.mls_label.level[1].cat.node = |
diff --git a/security/selinux/xfrm.c b/security/selinux/xfrm.c index bd8d1ef40a90..ba715f40b658 100644 --- a/security/selinux/xfrm.c +++ b/security/selinux/xfrm.c | |||
@@ -216,7 +216,7 @@ static int selinux_xfrm_sec_ctx_alloc(struct xfrm_sec_ctx **ctxp, | |||
216 | return -ENOMEM; | 216 | return -ENOMEM; |
217 | 217 | ||
218 | *ctxp = ctx = kmalloc(sizeof(*ctx) + | 218 | *ctxp = ctx = kmalloc(sizeof(*ctx) + |
219 | uctx->ctx_len, | 219 | uctx->ctx_len + 1, |
220 | GFP_KERNEL); | 220 | GFP_KERNEL); |
221 | 221 | ||
222 | if (!ctx) | 222 | if (!ctx) |
@@ -229,6 +229,7 @@ static int selinux_xfrm_sec_ctx_alloc(struct xfrm_sec_ctx **ctxp, | |||
229 | memcpy(ctx->ctx_str, | 229 | memcpy(ctx->ctx_str, |
230 | uctx+1, | 230 | uctx+1, |
231 | ctx->ctx_len); | 231 | ctx->ctx_len); |
232 | ctx->ctx_str[ctx->ctx_len] = 0; | ||
232 | rc = security_context_to_sid(ctx->ctx_str, | 233 | rc = security_context_to_sid(ctx->ctx_str, |
233 | ctx->ctx_len, | 234 | ctx->ctx_len, |
234 | &ctx->ctx_sid); | 235 | &ctx->ctx_sid); |