Details | Last modification | View Log | RSS feed
Rev | Author | Line No. | Line |
---|---|---|---|
422 | giacomo | 1 | #ifndef _LINUX_SWAP_H |
2 | #define _LINUX_SWAP_H |
||
3 | |||
4 | #include <linux/config.h> |
||
5 | #include <linux/spinlock.h> |
||
6 | #include <linux/linkage.h> |
||
7 | #include <linux/mmzone.h> |
||
8 | #include <linux/list.h> |
||
9 | #include <linux/sched.h> |
||
10 | #include <asm/atomic.h> |
||
11 | #include <asm/page.h> |
||
12 | |||
13 | #define SWAP_FLAG_PREFER 0x8000 /* set if swap priority specified */ |
||
14 | #define SWAP_FLAG_PRIO_MASK 0x7fff |
||
15 | #define SWAP_FLAG_PRIO_SHIFT 0 |
||
16 | |||
17 | static inline int current_is_kswapd(void) |
||
18 | { |
||
19 | return current->flags & PF_KSWAPD; |
||
20 | } |
||
21 | |||
22 | /* |
||
23 | * MAX_SWAPFILES defines the maximum number of swaptypes: things which can |
||
24 | * be swapped to. The swap type and the offset into that swap type are |
||
25 | * encoded into pte's and into pgoff_t's in the swapcache. Using five bits |
||
26 | * for the type means that the maximum number of swapcache pages is 27 bits |
||
27 | * on 32-bit-pgoff_t architectures. And that assumes that the architecture packs |
||
28 | * the type/offset into the pte as 5/27 as well. |
||
29 | */ |
||
30 | #define MAX_SWAPFILES_SHIFT 5 |
||
31 | #define MAX_SWAPFILES (1 << MAX_SWAPFILES_SHIFT) |
||
32 | |||
33 | /* |
||
34 | * Magic header for a swap area. The first part of the union is |
||
35 | * what the swap magic looks like for the old (limited to 128MB) |
||
36 | * swap area format, the second part of the union adds - in the |
||
37 | * old reserved area - some extra information. Note that the first |
||
38 | * kilobyte is reserved for boot loader or disk label stuff... |
||
39 | * |
||
40 | * Having the magic at the end of the PAGE_SIZE makes detecting swap |
||
41 | * areas somewhat tricky on machines that support multiple page sizes. |
||
42 | * For 2.5 we'll probably want to move the magic to just beyond the |
||
43 | * bootbits... |
||
44 | */ |
||
45 | union swap_header { |
||
46 | struct { |
||
47 | char reserved[PAGE_SIZE - 10]; |
||
48 | char magic[10]; /* SWAP-SPACE or SWAPSPACE2 */ |
||
49 | } magic; |
||
50 | struct { |
||
51 | char bootbits[1024]; /* Space for disklabel etc. */ |
||
52 | unsigned int version; |
||
53 | unsigned int last_page; |
||
54 | unsigned int nr_badpages; |
||
55 | unsigned int padding[125]; |
||
56 | unsigned int badpages[1]; |
||
57 | } info; |
||
58 | }; |
||
59 | |||
60 | /* A swap entry has to fit into a "unsigned long", as |
||
61 | * the entry is hidden in the "index" field of the |
||
62 | * swapper address space. |
||
63 | */ |
||
64 | typedef struct { |
||
65 | unsigned long val; |
||
66 | } swp_entry_t; |
||
67 | |||
68 | /* |
||
69 | * current->reclaim_state points to one of these when a task is running |
||
70 | * memory reclaim |
||
71 | */ |
||
72 | struct reclaim_state { |
||
73 | unsigned long reclaimed_slab; |
||
74 | }; |
||
75 | |||
76 | #ifdef __KERNEL__ |
||
77 | |||
78 | struct address_space; |
||
79 | struct pte_chain; |
||
80 | struct sysinfo; |
||
81 | struct writeback_control; |
||
82 | struct zone; |
||
83 | |||
84 | /* |
||
85 | * A swap extent maps a range of a swapfile's PAGE_SIZE pages onto a range of |
||
86 | * disk blocks. A list of swap extents maps the entire swapfile. (Where the |
||
87 | * term `swapfile' refers to either a blockdevice or an IS_REG file. Apart |
||
88 | * from setup, they're handled identically. |
||
89 | * |
||
90 | * We always assume that blocks are of size PAGE_SIZE. |
||
91 | */ |
||
92 | struct swap_extent { |
||
93 | struct list_head list; |
||
94 | pgoff_t start_page; |
||
95 | pgoff_t nr_pages; |
||
96 | sector_t start_block; |
||
97 | }; |
||
98 | |||
99 | /* |
||
100 | * Max bad pages in the new format.. |
||
101 | */ |
||
102 | #define __swapoffset(x) ((unsigned long)&((union swap_header *)0)->x) |
||
103 | #define MAX_SWAP_BADPAGES \ |
||
104 | ((__swapoffset(magic.magic) - __swapoffset(info.badpages)) / sizeof(int)) |
||
105 | |||
106 | enum { |
||
107 | SWP_USED = (1 << 0), /* is slot in swap_info[] used? */ |
||
108 | SWP_WRITEOK = (1 << 1), /* ok to write to this swap? */ |
||
109 | SWP_ACTIVE = (SWP_USED | SWP_WRITEOK), |
||
110 | }; |
||
111 | |||
112 | #define SWAP_CLUSTER_MAX 32 |
||
113 | |||
114 | #define SWAP_MAP_MAX 0x7fff |
||
115 | #define SWAP_MAP_BAD 0x8000 |
||
116 | |||
117 | /* |
||
118 | * The in-memory structure used to track swap areas. |
||
119 | * extent_list.prev points at the lowest-index extent. That list is |
||
120 | * sorted. |
||
121 | */ |
||
122 | struct swap_info_struct { |
||
123 | unsigned int flags; |
||
124 | spinlock_t sdev_lock; |
||
125 | struct file *swap_file; |
||
126 | struct block_device *bdev; |
||
127 | struct list_head extent_list; |
||
128 | int nr_extents; |
||
129 | struct swap_extent *curr_swap_extent; |
||
130 | unsigned old_block_size; |
||
131 | unsigned short * swap_map; |
||
132 | unsigned int lowest_bit; |
||
133 | unsigned int highest_bit; |
||
134 | unsigned int cluster_next; |
||
135 | unsigned int cluster_nr; |
||
136 | int prio; /* swap priority */ |
||
137 | int pages; |
||
138 | unsigned long max; |
||
139 | unsigned long inuse_pages; |
||
140 | int next; /* next entry on swap list */ |
||
141 | }; |
||
142 | |||
143 | struct swap_list_t { |
||
144 | int head; /* head of priority-ordered swapfile list */ |
||
145 | int next; /* swapfile to be used next */ |
||
146 | }; |
||
147 | |||
148 | /* Swap 50% full? Release swapcache more aggressively.. */ |
||
149 | #define vm_swap_full() (nr_swap_pages*2 < total_swap_pages) |
||
150 | |||
151 | /* linux/mm/oom_kill.c */ |
||
152 | extern void out_of_memory(void); |
||
153 | |||
154 | /* linux/mm/memory.c */ |
||
155 | extern void swapin_readahead(swp_entry_t); |
||
156 | |||
157 | /* linux/mm/page_alloc.c */ |
||
158 | extern unsigned long totalram_pages; |
||
159 | extern unsigned long totalhigh_pages; |
||
160 | extern int nr_swap_pages; /* XXX: shouldn't this be ulong? --hch */ |
||
161 | extern unsigned int nr_free_pages(void); |
||
162 | extern unsigned int nr_free_pages_pgdat(pg_data_t *pgdat); |
||
163 | extern unsigned int nr_free_buffer_pages(void); |
||
164 | extern unsigned int nr_free_pagecache_pages(void); |
||
165 | |||
166 | /* linux/mm/swap.c */ |
||
167 | extern void FASTCALL(lru_cache_add(struct page *)); |
||
168 | extern void FASTCALL(lru_cache_add_active(struct page *)); |
||
169 | extern void FASTCALL(activate_page(struct page *)); |
||
170 | extern void FASTCALL(mark_page_accessed(struct page *)); |
||
171 | extern void lru_add_drain(void); |
||
172 | extern int rotate_reclaimable_page(struct page *page); |
||
173 | extern void swap_setup(void); |
||
174 | |||
175 | /* linux/mm/vmscan.c */ |
||
176 | extern int try_to_free_pages(struct zone *, unsigned int, unsigned int); |
||
177 | extern int shrink_all_memory(int); |
||
178 | extern int vm_swappiness; |
||
179 | |||
180 | /* linux/mm/rmap.c */ |
||
181 | #ifdef CONFIG_MMU |
||
182 | int FASTCALL(page_referenced(struct page *)); |
||
183 | struct pte_chain *FASTCALL(page_add_rmap(struct page *, pte_t *, |
||
184 | struct pte_chain *)); |
||
185 | void FASTCALL(page_remove_rmap(struct page *, pte_t *)); |
||
186 | int FASTCALL(try_to_unmap(struct page *)); |
||
187 | |||
188 | /* linux/mm/shmem.c */ |
||
189 | extern int shmem_unuse(swp_entry_t entry, struct page *page); |
||
190 | #else |
||
191 | #define page_referenced(page) TestClearPageReferenced(page) |
||
192 | #define try_to_unmap(page) SWAP_FAIL |
||
193 | #endif /* CONFIG_MMU */ |
||
194 | |||
195 | /* return values of try_to_unmap */ |
||
196 | #define SWAP_SUCCESS 0 |
||
197 | #define SWAP_AGAIN 1 |
||
198 | #define SWAP_FAIL 2 |
||
199 | |||
200 | #ifdef CONFIG_SWAP |
||
201 | /* linux/mm/page_io.c */ |
||
202 | extern int swap_readpage(struct file *, struct page *); |
||
203 | extern int swap_writepage(struct page *page, struct writeback_control *wbc); |
||
204 | extern int rw_swap_page_sync(int, swp_entry_t, struct page *); |
||
205 | |||
206 | /* linux/mm/swap_state.c */ |
||
207 | extern struct address_space swapper_space; |
||
208 | #define total_swapcache_pages swapper_space.nrpages |
||
209 | extern void show_swap_cache_info(void); |
||
210 | extern int add_to_swap(struct page *); |
||
211 | extern void __delete_from_swap_cache(struct page *); |
||
212 | extern void delete_from_swap_cache(struct page *); |
||
213 | extern int move_to_swap_cache(struct page *, swp_entry_t); |
||
214 | extern int move_from_swap_cache(struct page *, unsigned long, |
||
215 | struct address_space *); |
||
216 | extern void free_page_and_swap_cache(struct page *); |
||
217 | extern void free_pages_and_swap_cache(struct page **, int); |
||
218 | extern struct page * lookup_swap_cache(swp_entry_t); |
||
219 | extern struct page * read_swap_cache_async(swp_entry_t); |
||
220 | |||
221 | /* linux/mm/swapfile.c */ |
||
222 | extern int total_swap_pages; |
||
223 | extern unsigned int nr_swapfiles; |
||
224 | extern struct swap_info_struct swap_info[]; |
||
225 | extern void si_swapinfo(struct sysinfo *); |
||
226 | extern swp_entry_t get_swap_page(void); |
||
227 | extern int swap_duplicate(swp_entry_t); |
||
228 | extern int valid_swaphandles(swp_entry_t, unsigned long *); |
||
229 | extern void swap_free(swp_entry_t); |
||
230 | extern void free_swap_and_cache(swp_entry_t); |
||
231 | extern sector_t map_swap_page(struct swap_info_struct *, pgoff_t); |
||
232 | extern struct swap_info_struct *get_swap_info_struct(unsigned); |
||
233 | extern int can_share_swap_page(struct page *); |
||
234 | extern int remove_exclusive_swap_page(struct page *); |
||
235 | |||
236 | extern struct swap_list_t swap_list; |
||
237 | extern spinlock_t swaplock; |
||
238 | |||
239 | #define swap_list_lock() spin_lock(&swaplock) |
||
240 | #define swap_list_unlock() spin_unlock(&swaplock) |
||
241 | #define swap_device_lock(p) spin_lock(&p->sdev_lock) |
||
242 | #define swap_device_unlock(p) spin_unlock(&p->sdev_lock) |
||
243 | |||
244 | #else /* CONFIG_SWAP */ |
||
245 | |||
246 | #define total_swap_pages 0 |
||
247 | #define total_swapcache_pages 0UL |
||
248 | |||
249 | #define si_swapinfo(val) \ |
||
250 | do { (val)->freeswap = (val)->totalswap = 0; } while (0) |
||
251 | #define free_page_and_swap_cache(page) \ |
||
252 | page_cache_release(page) |
||
253 | #define free_pages_and_swap_cache(pages, nr) \ |
||
254 | release_pages((pages), (nr), 0); |
||
255 | |||
256 | #define show_swap_cache_info() /*NOTHING*/ |
||
257 | #define free_swap_and_cache(swp) /*NOTHING*/ |
||
258 | #define swap_duplicate(swp) /*NOTHING*/ |
||
259 | #define swap_free(swp) /*NOTHING*/ |
||
260 | #define read_swap_cache_async(swp) NULL |
||
261 | #define lookup_swap_cache(swp) NULL |
||
262 | #define valid_swaphandles(swp, off) 0 |
||
263 | #define can_share_swap_page(p) 0 |
||
264 | #define move_to_swap_cache(p, swp) 1 |
||
265 | #define move_from_swap_cache(p, i, m) 1 |
||
266 | #define __delete_from_swap_cache(p) /*NOTHING*/ |
||
267 | #define delete_from_swap_cache(p) /*NOTHING*/ |
||
268 | |||
269 | static inline int remove_exclusive_swap_page(struct page *p) |
||
270 | { |
||
271 | return 0; |
||
272 | } |
||
273 | |||
274 | static inline swp_entry_t get_swap_page(void) |
||
275 | { |
||
276 | swp_entry_t entry; |
||
277 | entry.val = 0; |
||
278 | return entry; |
||
279 | } |
||
280 | |||
281 | #endif /* CONFIG_SWAP */ |
||
282 | #endif /* __KERNEL__*/ |
||
283 | #endif /* _LINUX_SWAP_H */ |