blob: 6b5110a474582fb296e7abbb0076c99a466d7fc8 [file] [log] [blame]
Ed L. Cashin52e112b2008-02-08 04:20:09 -08001/* Copyright (c) 2007 Coraid, Inc. See COPYING for GPL terms. */
Linus Torvalds1da177e2005-04-16 15:20:36 -07002/*
3 * aoedev.c
4 * AoE device utility functions; maintains device list.
5 */
6
7#include <linux/hdreg.h>
8#include <linux/blkdev.h>
9#include <linux/netdevice.h>
Ed L. Cashin9bb237b2008-02-08 04:20:05 -080010#include <linux/delay.h>
Tejun Heo5a0e3ad2010-03-24 17:04:11 +090011#include <linux/slab.h>
Linus Torvalds1da177e2005-04-16 15:20:36 -070012#include "aoe.h"
13
Ed L. Cashin262bf542008-02-08 04:20:03 -080014static void dummy_timer(ulong);
15static void aoedev_freedev(struct aoedev *);
Ed L. Cashin9bb237b2008-02-08 04:20:05 -080016static void freetgt(struct aoedev *d, struct aoetgt *t);
17static void skbpoolfree(struct aoedev *d);
Ed L. Cashin262bf542008-02-08 04:20:03 -080018
Linus Torvalds1da177e2005-04-16 15:20:36 -070019static struct aoedev *devlist;
Andrew Morton476aed32008-02-08 04:20:10 -080020static DEFINE_SPINLOCK(devlist_lock);
Linus Torvalds1da177e2005-04-16 15:20:36 -070021
22struct aoedev *
ecashin@coraid.com32465c62005-04-18 22:00:18 -070023aoedev_by_aoeaddr(int maj, int min)
Linus Torvalds1da177e2005-04-16 15:20:36 -070024{
25 struct aoedev *d;
26 ulong flags;
27
28 spin_lock_irqsave(&devlist_lock, flags);
29
30 for (d=devlist; d; d=d->next)
ecashin@coraid.com32465c62005-04-18 22:00:18 -070031 if (d->aoemajor == maj && d->aoeminor == min)
Linus Torvalds1da177e2005-04-16 15:20:36 -070032 break;
33
34 spin_unlock_irqrestore(&devlist_lock, flags);
35 return d;
36}
37
Ed L. Cashin3ae1c242006-01-19 13:46:19 -050038static void
39dummy_timer(ulong vp)
40{
41 struct aoedev *d;
42
43 d = (struct aoedev *)vp;
44 if (d->flags & DEVFL_TKILL)
45 return;
46 d->timer.expires = jiffies + HZ;
47 add_timer(&d->timer);
48}
49
Linus Torvalds1da177e2005-04-16 15:20:36 -070050void
51aoedev_downdev(struct aoedev *d)
52{
Ed L. Cashin68e0d422008-02-08 04:20:00 -080053 struct aoetgt **t, **te;
Linus Torvalds1da177e2005-04-16 15:20:36 -070054 struct frame *f, *e;
55 struct buf *buf;
56 struct bio *bio;
57
Ed L. Cashin68e0d422008-02-08 04:20:00 -080058 t = d->targets;
59 te = t + NTARGETS;
60 for (; t < te && *t; t++) {
61 f = (*t)->frames;
62 e = f + (*t)->nframes;
63 for (; f < e; f->tag = FREETAG, f->buf = NULL, f++) {
64 if (f->tag == FREETAG || f->buf == NULL)
65 continue;
66 buf = f->buf;
67 bio = buf->bio;
68 if (--buf->nframesout == 0
69 && buf != d->inprocess) {
70 mempool_free(buf, d->bufpool);
71 bio_endio(bio, -EIO);
72 }
Linus Torvalds1da177e2005-04-16 15:20:36 -070073 }
Ed L. Cashin68e0d422008-02-08 04:20:00 -080074 (*t)->maxout = (*t)->nframes;
75 (*t)->nout = 0;
76 }
77 buf = d->inprocess;
78 if (buf) {
79 bio = buf->bio;
80 mempool_free(buf, d->bufpool);
81 bio_endio(bio, -EIO);
Linus Torvalds1da177e2005-04-16 15:20:36 -070082 }
83 d->inprocess = NULL;
Ed L. Cashin68e0d422008-02-08 04:20:00 -080084 d->htgt = NULL;
Linus Torvalds1da177e2005-04-16 15:20:36 -070085
86 while (!list_empty(&d->bufq)) {
87 buf = container_of(d->bufq.next, struct buf, bufs);
88 list_del(d->bufq.next);
89 bio = buf->bio;
90 mempool_free(buf, d->bufpool);
NeilBrown6712ecf2007-09-27 12:47:43 +020091 bio_endio(bio, -EIO);
Linus Torvalds1da177e2005-04-16 15:20:36 -070092 }
93
Linus Torvalds1da177e2005-04-16 15:20:36 -070094 if (d->gd)
Tejun Heo80795ae2008-08-25 19:56:07 +090095 set_capacity(d->gd, 0);
Linus Torvalds1da177e2005-04-16 15:20:36 -070096
Ed L. Cashin68e0d422008-02-08 04:20:00 -080097 d->flags &= ~DEVFL_UP;
Linus Torvalds1da177e2005-04-16 15:20:36 -070098}
99
Ed L. Cashin262bf542008-02-08 04:20:03 -0800100static void
101aoedev_freedev(struct aoedev *d)
102{
103 struct aoetgt **t, **e;
104
Tejun Heo5ad21a332010-10-28 06:15:26 -0600105 cancel_work_sync(&d->work);
Ed L. Cashin262bf542008-02-08 04:20:03 -0800106 if (d->gd) {
107 aoedisk_rm_sysfs(d);
108 del_gendisk(d->gd);
109 put_disk(d->gd);
110 }
111 t = d->targets;
112 e = t + NTARGETS;
113 for (; t < e && *t; t++)
Ed L. Cashin9bb237b2008-02-08 04:20:05 -0800114 freetgt(d, *t);
Ed L. Cashin262bf542008-02-08 04:20:03 -0800115 if (d->bufpool)
116 mempool_destroy(d->bufpool);
Ed L. Cashin9bb237b2008-02-08 04:20:05 -0800117 skbpoolfree(d);
Ed Cashin7135a71b2009-09-09 14:10:18 +0200118 blk_cleanup_queue(d->blkq);
Ed L. Cashin262bf542008-02-08 04:20:03 -0800119 kfree(d);
120}
121
122int
123aoedev_flush(const char __user *str, size_t cnt)
124{
125 ulong flags;
126 struct aoedev *d, **dd;
127 struct aoedev *rmd = NULL;
128 char buf[16];
129 int all = 0;
130
131 if (cnt >= 3) {
132 if (cnt > sizeof buf)
133 cnt = sizeof buf;
134 if (copy_from_user(buf, str, cnt))
135 return -EFAULT;
136 all = !strncmp(buf, "all", 3);
137 }
138
Ed L. Cashin262bf542008-02-08 04:20:03 -0800139 spin_lock_irqsave(&devlist_lock, flags);
140 dd = &devlist;
141 while ((d = *dd)) {
142 spin_lock(&d->lock);
143 if ((!all && (d->flags & DEVFL_UP))
144 || (d->flags & (DEVFL_GDALLOC|DEVFL_NEWSIZE))
145 || d->nopen) {
146 spin_unlock(&d->lock);
147 dd = &d->next;
148 continue;
149 }
150 *dd = d->next;
151 aoedev_downdev(d);
152 d->flags |= DEVFL_TKILL;
153 spin_unlock(&d->lock);
154 d->next = rmd;
155 rmd = d;
156 }
157 spin_unlock_irqrestore(&devlist_lock, flags);
158 while ((d = rmd)) {
159 rmd = d->next;
160 del_timer_sync(&d->timer);
161 aoedev_freedev(d); /* must be able to sleep */
162 }
163 return 0;
164}
165
Ed L. Cashin9bb237b2008-02-08 04:20:05 -0800166/* I'm not really sure that this is a realistic problem, but if the
167network driver goes gonzo let's just leak memory after complaining. */
168static void
169skbfree(struct sk_buff *skb)
170{
171 enum { Sms = 100, Tms = 3*1000};
172 int i = Tms / Sms;
173
174 if (skb == NULL)
175 return;
176 while (atomic_read(&skb_shinfo(skb)->dataref) != 1 && i-- > 0)
177 msleep(Sms);
Roel Kluin94873112009-03-04 00:07:57 -0800178 if (i < 0) {
Ed L. Cashin9bb237b2008-02-08 04:20:05 -0800179 printk(KERN_ERR
180 "aoe: %s holds ref: %s\n",
181 skb->dev ? skb->dev->name : "netif",
182 "cannot free skb -- memory leaked.");
183 return;
184 }
185 skb_shinfo(skb)->nr_frags = skb->data_len = 0;
186 skb_trim(skb, 0);
187 dev_kfree_skb(skb);
188}
189
190static void
191skbpoolfree(struct aoedev *d)
192{
David S. Millere9bb8fb02008-09-21 22:36:49 -0700193 struct sk_buff *skb, *tmp;
Ed L. Cashin9bb237b2008-02-08 04:20:05 -0800194
David S. Millere9bb8fb02008-09-21 22:36:49 -0700195 skb_queue_walk_safe(&d->skbpool, skb, tmp)
Ed L. Cashin9bb237b2008-02-08 04:20:05 -0800196 skbfree(skb);
David S. Millere9bb8fb02008-09-21 22:36:49 -0700197
198 __skb_queue_head_init(&d->skbpool);
Ed L. Cashin9bb237b2008-02-08 04:20:05 -0800199}
200
Ed L. Cashin3ae1c242006-01-19 13:46:19 -0500201/* find it or malloc it */
Linus Torvalds1da177e2005-04-16 15:20:36 -0700202struct aoedev *
Ed L. Cashin68e0d422008-02-08 04:20:00 -0800203aoedev_by_sysminor_m(ulong sysminor)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700204{
205 struct aoedev *d;
206 ulong flags;
207
208 spin_lock_irqsave(&devlist_lock, flags);
209
210 for (d=devlist; d; d=d->next)
Ed L Cashin93d489f2005-04-29 10:24:22 -0400211 if (d->sysminor == sysminor)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700212 break;
Ed L. Cashin68e0d422008-02-08 04:20:00 -0800213 if (d)
214 goto out;
215 d = kcalloc(1, sizeof *d, GFP_ATOMIC);
216 if (!d)
217 goto out;
218 INIT_WORK(&d->work, aoecmd_sleepwork);
219 spin_lock_init(&d->lock);
David S. Millere9bb8fb02008-09-21 22:36:49 -0700220 skb_queue_head_init(&d->sendq);
221 skb_queue_head_init(&d->skbpool);
Ed L. Cashin68e0d422008-02-08 04:20:00 -0800222 init_timer(&d->timer);
223 d->timer.data = (ulong) d;
224 d->timer.function = dummy_timer;
225 d->timer.expires = jiffies + HZ;
226 add_timer(&d->timer);
227 d->bufpool = NULL; /* defer to aoeblk_gdalloc */
228 d->tgt = d->targets;
229 INIT_LIST_HEAD(&d->bufq);
230 d->sysminor = sysminor;
231 d->aoemajor = AOEMAJOR(sysminor);
232 d->aoeminor = AOEMINOR(sysminor);
233 d->mintimer = MINTIMER;
234 d->next = devlist;
235 devlist = d;
236 out:
Ed L. Cashin3ae1c242006-01-19 13:46:19 -0500237 spin_unlock_irqrestore(&devlist_lock, flags);
Linus Torvalds1da177e2005-04-16 15:20:36 -0700238 return d;
239}
240
241static void
Ed L. Cashin9bb237b2008-02-08 04:20:05 -0800242freetgt(struct aoedev *d, struct aoetgt *t)
Linus Torvalds1da177e2005-04-16 15:20:36 -0700243{
Ed L. Cashine407a7f2006-09-20 14:36:49 -0400244 struct frame *f, *e;
245
Ed L. Cashin68e0d422008-02-08 04:20:00 -0800246 f = t->frames;
247 e = f + t->nframes;
Ed L. Cashin9bb237b2008-02-08 04:20:05 -0800248 for (; f < e; f++)
249 skbfree(f->skb);
Ed L. Cashin68e0d422008-02-08 04:20:00 -0800250 kfree(t->frames);
251 kfree(t);
252}
253
Linus Torvalds1da177e2005-04-16 15:20:36 -0700254void
255aoedev_exit(void)
256{
257 struct aoedev *d;
258 ulong flags;
259
Linus Torvalds1da177e2005-04-16 15:20:36 -0700260 while ((d = devlist)) {
261 devlist = d->next;
262
263 spin_lock_irqsave(&d->lock, flags);
264 aoedev_downdev(d);
Ed L. Cashin3ae1c242006-01-19 13:46:19 -0500265 d->flags |= DEVFL_TKILL;
Linus Torvalds1da177e2005-04-16 15:20:36 -0700266 spin_unlock_irqrestore(&d->lock, flags);
267
268 del_timer_sync(&d->timer);
269 aoedev_freedev(d);
270 }
271}
272
273int __init
274aoedev_init(void)
275{
Linus Torvalds1da177e2005-04-16 15:20:36 -0700276 return 0;
277}