Blame view

lib/sg_pool.c 4.22 KB
457c89965   Thomas Gleixner   treewide: Add SPD...
1
  // SPDX-License-Identifier: GPL-2.0-only
9b1d6c895   Ming Lin   lib: scatterlist:...
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
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
  #include <linux/module.h>
  #include <linux/scatterlist.h>
  #include <linux/mempool.h>
  #include <linux/slab.h>
  
  #define SG_MEMPOOL_NR		ARRAY_SIZE(sg_pools)
  #define SG_MEMPOOL_SIZE		2
  
  struct sg_pool {
  	size_t		size;
  	char		*name;
  	struct kmem_cache	*slab;
  	mempool_t	*pool;
  };
  
  #define SP(x) { .size = x, "sgpool-" __stringify(x) }
  #if (SG_CHUNK_SIZE < 32)
  #error SG_CHUNK_SIZE is too small (must be 32 or greater)
  #endif
  static struct sg_pool sg_pools[] = {
  	SP(8),
  	SP(16),
  #if (SG_CHUNK_SIZE > 32)
  	SP(32),
  #if (SG_CHUNK_SIZE > 64)
  	SP(64),
  #if (SG_CHUNK_SIZE > 128)
  	SP(128),
  #if (SG_CHUNK_SIZE > 256)
  #error SG_CHUNK_SIZE is too large (256 MAX)
  #endif
  #endif
  #endif
  #endif
  	SP(SG_CHUNK_SIZE)
  };
  #undef SP
  
  static inline unsigned int sg_pool_index(unsigned short nents)
  {
  	unsigned int index;
  
  	BUG_ON(nents > SG_CHUNK_SIZE);
  
  	if (nents <= 8)
  		index = 0;
  	else
  		index = get_count_order(nents) - 3;
  
  	return index;
  }
  
  static void sg_pool_free(struct scatterlist *sgl, unsigned int nents)
  {
  	struct sg_pool *sgp;
  
  	sgp = sg_pools + sg_pool_index(nents);
  	mempool_free(sgl, sgp->pool);
  }
  
  static struct scatterlist *sg_pool_alloc(unsigned int nents, gfp_t gfp_mask)
  {
  	struct sg_pool *sgp;
  
  	sgp = sg_pools + sg_pool_index(nents);
  	return mempool_alloc(sgp->pool, gfp_mask);
  }
  
  /**
   * sg_free_table_chained - Free a previously mapped sg table
   * @table:	The sg table header to use
4635873c5   Ming Lei   scsi: lib/sg_pool...
73
74
   * @nents_first_chunk: size of the first_chunk SGL passed to
   *		sg_alloc_table_chained
9b1d6c895   Ming Lin   lib: scatterlist:...
75
76
77
78
79
   *
   *  Description:
   *    Free an sg table previously allocated and setup with
   *    sg_alloc_table_chained().
   *
4635873c5   Ming Lei   scsi: lib/sg_pool...
80
81
82
   *    @nents_first_chunk has to be same with that same parameter passed
   *    to sg_alloc_table_chained().
   *
9b1d6c895   Ming Lin   lib: scatterlist:...
83
   **/
4635873c5   Ming Lei   scsi: lib/sg_pool...
84
85
  void sg_free_table_chained(struct sg_table *table,
  		unsigned nents_first_chunk)
9b1d6c895   Ming Lin   lib: scatterlist:...
86
  {
4635873c5   Ming Lei   scsi: lib/sg_pool...
87
  	if (table->orig_nents <= nents_first_chunk)
9b1d6c895   Ming Lin   lib: scatterlist:...
88
  		return;
4635873c5   Ming Lei   scsi: lib/sg_pool...
89
90
91
92
93
  
  	if (nents_first_chunk == 1)
  		nents_first_chunk = 0;
  
  	__sg_free_table(table, SG_CHUNK_SIZE, nents_first_chunk, sg_pool_free);
9b1d6c895   Ming Lin   lib: scatterlist:...
94
95
96
97
98
99
100
101
  }
  EXPORT_SYMBOL_GPL(sg_free_table_chained);
  
  /**
   * sg_alloc_table_chained - Allocate and chain SGLs in an sg table
   * @table:	The sg table header to use
   * @nents:	Number of entries in sg list
   * @first_chunk: first SGL
4635873c5   Ming Lei   scsi: lib/sg_pool...
102
   * @nents_first_chunk: number of the SGL of @first_chunk
9b1d6c895   Ming Lin   lib: scatterlist:...
103
104
105
   *
   *  Description:
   *    Allocate and chain SGLs in an sg table. If @nents@ is larger than
b79d9a09a   Ming Lei   scsi: lib/sg_pool...
106
107
108
   *    @nents_first_chunk a chained sg table will be setup. @first_chunk is
   *    ignored if nents_first_chunk <= 1 because user expects the SGL points
   *    non-chain SGL.
9b1d6c895   Ming Lin   lib: scatterlist:...
109
110
111
   *
   **/
  int sg_alloc_table_chained(struct sg_table *table, int nents,
4635873c5   Ming Lei   scsi: lib/sg_pool...
112
  		struct scatterlist *first_chunk, unsigned nents_first_chunk)
9b1d6c895   Ming Lin   lib: scatterlist:...
113
114
115
116
  {
  	int ret;
  
  	BUG_ON(!nents);
4635873c5   Ming Lei   scsi: lib/sg_pool...
117
118
  	if (first_chunk && nents_first_chunk) {
  		if (nents <= nents_first_chunk) {
9b1d6c895   Ming Lin   lib: scatterlist:...
119
120
121
122
123
  			table->nents = table->orig_nents = nents;
  			sg_init_table(table->sgl, nents);
  			return 0;
  		}
  	}
4635873c5   Ming Lei   scsi: lib/sg_pool...
124
  	/* User supposes that the 1st SGL includes real entry */
b79d9a09a   Ming Lei   scsi: lib/sg_pool...
125
  	if (nents_first_chunk <= 1) {
4635873c5   Ming Lei   scsi: lib/sg_pool...
126
127
128
  		first_chunk = NULL;
  		nents_first_chunk = 0;
  	}
9b1d6c895   Ming Lin   lib: scatterlist:...
129
  	ret = __sg_alloc_table(table, nents, SG_CHUNK_SIZE,
4635873c5   Ming Lei   scsi: lib/sg_pool...
130
131
  			       first_chunk, nents_first_chunk,
  			       GFP_ATOMIC, sg_pool_alloc);
9b1d6c895   Ming Lin   lib: scatterlist:...
132
  	if (unlikely(ret))
4635873c5   Ming Lei   scsi: lib/sg_pool...
133
  		sg_free_table_chained(table, nents_first_chunk);
9b1d6c895   Ming Lin   lib: scatterlist:...
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
  	return ret;
  }
  EXPORT_SYMBOL_GPL(sg_alloc_table_chained);
  
  static __init int sg_pool_init(void)
  {
  	int i;
  
  	for (i = 0; i < SG_MEMPOOL_NR; i++) {
  		struct sg_pool *sgp = sg_pools + i;
  		int size = sgp->size * sizeof(struct scatterlist);
  
  		sgp->slab = kmem_cache_create(sgp->name, size, 0,
  				SLAB_HWCACHE_ALIGN, NULL);
  		if (!sgp->slab) {
  			printk(KERN_ERR "SG_POOL: can't init sg slab %s
  ",
  					sgp->name);
  			goto cleanup_sdb;
  		}
  
  		sgp->pool = mempool_create_slab_pool(SG_MEMPOOL_SIZE,
  						     sgp->slab);
  		if (!sgp->pool) {
  			printk(KERN_ERR "SG_POOL: can't init sg mempool %s
  ",
  					sgp->name);
  			goto cleanup_sdb;
  		}
  	}
  
  	return 0;
  
  cleanup_sdb:
  	for (i = 0; i < SG_MEMPOOL_NR; i++) {
  		struct sg_pool *sgp = sg_pools + i;
7f476715d   zhong jiang   lib/sg_pool.c: re...
170
171
172
  
  		mempool_destroy(sgp->pool);
  		kmem_cache_destroy(sgp->slab);
9b1d6c895   Ming Lin   lib: scatterlist:...
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
  	}
  
  	return -ENOMEM;
  }
  
  static __exit void sg_pool_exit(void)
  {
  	int i;
  
  	for (i = 0; i < SG_MEMPOOL_NR; i++) {
  		struct sg_pool *sgp = sg_pools + i;
  		mempool_destroy(sgp->pool);
  		kmem_cache_destroy(sgp->slab);
  	}
  }
  
  module_init(sg_pool_init);
  module_exit(sg_pool_exit);