Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux
1
fork

Configure Feed

Select the types of activity you want to include in your feed.

at 3bdf590eac36ac5930deb9552febee3ff18cd2d1 260 lines 6.5 kB view raw
1#ifndef _LINUX_SCATTERLIST_H 2#define _LINUX_SCATTERLIST_H 3 4#include <asm/scatterlist.h> 5#include <linux/mm.h> 6#include <linux/string.h> 7#include <asm/io.h> 8 9/* 10 * Notes on SG table design. 11 * 12 * Architectures must provide an unsigned long page_link field in the 13 * scatterlist struct. We use that to place the page pointer AND encode 14 * information about the sg table as well. The two lower bits are reserved 15 * for this information. 16 * 17 * If bit 0 is set, then the page_link contains a pointer to the next sg 18 * table list. Otherwise the next entry is at sg + 1. 19 * 20 * If bit 1 is set, then this sg entry is the last element in a list. 21 * 22 * See sg_next(). 23 * 24 */ 25 26#define SG_MAGIC 0x87654321 27 28/** 29 * sg_set_page - Set sg entry to point at given page 30 * @sg: SG entry 31 * @page: The page 32 * 33 * Description: 34 * Use this function to set an sg entry pointing at a page, never assign 35 * the page directly. We encode sg table information in the lower bits 36 * of the page pointer. See sg_page() for looking up the page belonging 37 * to an sg entry. 38 * 39 **/ 40static inline void sg_set_page(struct scatterlist *sg, struct page *page) 41{ 42 unsigned long page_link = sg->page_link & 0x3; 43 44 /* 45 * In order for the low bit stealing approach to work, pages 46 * must be aligned at a 32-bit boundary as a minimum. 47 */ 48 BUG_ON((unsigned long) page & 0x03); 49#ifdef CONFIG_DEBUG_SG 50 BUG_ON(sg->sg_magic != SG_MAGIC); 51#endif 52 sg->page_link = page_link | (unsigned long) page; 53} 54 55#define sg_page(sg) ((struct page *) ((sg)->page_link & ~0x3)) 56 57/** 58 * sg_set_buf - Set sg entry to point at given data 59 * @sg: SG entry 60 * @buf: Data 61 * @buflen: Data length 62 * 63 **/ 64static inline void sg_set_buf(struct scatterlist *sg, const void *buf, 65 unsigned int buflen) 66{ 67 sg_set_page(sg, virt_to_page(buf)); 68 sg->offset = offset_in_page(buf); 69 sg->length = buflen; 70} 71 72/* 73 * We overload the LSB of the page pointer to indicate whether it's 74 * a valid sg entry, or whether it points to the start of a new scatterlist. 75 * Those low bits are there for everyone! (thanks mason :-) 76 */ 77#define sg_is_chain(sg) ((sg)->page_link & 0x01) 78#define sg_is_last(sg) ((sg)->page_link & 0x02) 79#define sg_chain_ptr(sg) \ 80 ((struct scatterlist *) ((sg)->page_link & ~0x03)) 81 82/** 83 * sg_next - return the next scatterlist entry in a list 84 * @sg: The current sg entry 85 * 86 * Description: 87 * Usually the next entry will be @sg@ + 1, but if this sg element is part 88 * of a chained scatterlist, it could jump to the start of a new 89 * scatterlist array. 90 * 91 **/ 92static inline struct scatterlist *sg_next(struct scatterlist *sg) 93{ 94#ifdef CONFIG_DEBUG_SG 95 BUG_ON(sg->sg_magic != SG_MAGIC); 96#endif 97 if (sg_is_last(sg)) 98 return NULL; 99 100 sg++; 101 if (unlikely(sg_is_chain(sg))) 102 sg = sg_chain_ptr(sg); 103 104 return sg; 105} 106 107/* 108 * Loop over each sg element, following the pointer to a new list if necessary 109 */ 110#define for_each_sg(sglist, sg, nr, __i) \ 111 for (__i = 0, sg = (sglist); __i < (nr); __i++, sg = sg_next(sg)) 112 113/** 114 * sg_last - return the last scatterlist entry in a list 115 * @sgl: First entry in the scatterlist 116 * @nents: Number of entries in the scatterlist 117 * 118 * Description: 119 * Should only be used casually, it (currently) scan the entire list 120 * to get the last entry. 121 * 122 * Note that the @sgl@ pointer passed in need not be the first one, 123 * the important bit is that @nents@ denotes the number of entries that 124 * exist from @sgl@. 125 * 126 **/ 127static inline struct scatterlist *sg_last(struct scatterlist *sgl, 128 unsigned int nents) 129{ 130#ifndef ARCH_HAS_SG_CHAIN 131 struct scatterlist *ret = &sgl[nents - 1]; 132#else 133 struct scatterlist *sg, *ret = NULL; 134 int i; 135 136 for_each_sg(sgl, sg, nents, i) 137 ret = sg; 138 139#endif 140#ifdef CONFIG_DEBUG_SG 141 BUG_ON(sgl[0].sg_magic != SG_MAGIC); 142 BUG_ON(!sg_is_last(ret)); 143#endif 144 return ret; 145} 146 147/** 148 * sg_chain - Chain two sglists together 149 * @prv: First scatterlist 150 * @prv_nents: Number of entries in prv 151 * @sgl: Second scatterlist 152 * 153 * Description: 154 * Links @prv@ and @sgl@ together, to form a longer scatterlist. 155 * 156 **/ 157static inline void sg_chain(struct scatterlist *prv, unsigned int prv_nents, 158 struct scatterlist *sgl) 159{ 160#ifndef ARCH_HAS_SG_CHAIN 161 BUG(); 162#endif 163 prv[prv_nents - 1].page_link = (unsigned long) sgl | 0x01; 164} 165 166/** 167 * sg_mark_end - Mark the end of the scatterlist 168 * @sgl: Scatterlist 169 * @nents: Number of entries in sgl 170 * 171 * Description: 172 * Marks the last entry as the termination point for sg_next() 173 * 174 **/ 175static inline void sg_mark_end(struct scatterlist *sgl, unsigned int nents) 176{ 177 sgl[nents - 1].page_link = 0x02; 178} 179 180static inline void __sg_mark_end(struct scatterlist *sg) 181{ 182 sg->page_link |= 0x02; 183} 184 185/** 186 * sg_init_one - Initialize a single entry sg list 187 * @sg: SG entry 188 * @buf: Virtual address for IO 189 * @buflen: IO length 190 * 191 * Notes: 192 * This should not be used on a single entry that is part of a larger 193 * table. Use sg_init_table() for that. 194 * 195 **/ 196static inline void sg_init_one(struct scatterlist *sg, const void *buf, 197 unsigned int buflen) 198{ 199 memset(sg, 0, sizeof(*sg)); 200#ifdef CONFIG_DEBUG_SG 201 sg->sg_magic = SG_MAGIC; 202#endif 203 sg_mark_end(sg, 1); 204 sg_set_buf(sg, buf, buflen); 205} 206 207/** 208 * sg_init_table - Initialize SG table 209 * @sgl: The SG table 210 * @nents: Number of entries in table 211 * 212 * Notes: 213 * If this is part of a chained sg table, sg_mark_end() should be 214 * used only on the last table part. 215 * 216 **/ 217static inline void sg_init_table(struct scatterlist *sgl, unsigned int nents) 218{ 219 memset(sgl, 0, sizeof(*sgl) * nents); 220 sg_mark_end(sgl, nents); 221#ifdef CONFIG_DEBUG_SG 222 { 223 int i; 224 for (i = 0; i < nents; i++) 225 sgl[i].sg_magic = SG_MAGIC; 226 } 227#endif 228} 229 230/** 231 * sg_phys - Return physical address of an sg entry 232 * @sg: SG entry 233 * 234 * Description: 235 * This calls page_to_phys() on the page in this sg entry, and adds the 236 * sg offset. The caller must know that it is legal to call page_to_phys() 237 * on the sg page. 238 * 239 **/ 240static inline unsigned long sg_phys(struct scatterlist *sg) 241{ 242 return page_to_phys(sg_page(sg)) + sg->offset; 243} 244 245/** 246 * sg_virt - Return virtual address of an sg entry 247 * @sg: SG entry 248 * 249 * Description: 250 * This calls page_address() on the page in this sg entry, and adds the 251 * sg offset. The caller must know that the sg page has a valid virtual 252 * mapping. 253 * 254 **/ 255static inline void *sg_virt(struct scatterlist *sg) 256{ 257 return page_address(sg_page(sg)) + sg->offset; 258} 259 260#endif /* _LINUX_SCATTERLIST_H */