1 /*-
2 * Copyright (c) 2003-2004 Poul-Henning Kamp
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 * 1. Redistributions of source code must retain the above copyright
9 * notice, this list of conditions and the following disclaimer.
10 * 2. Redistributions in binary form must reproduce the above copyright
11 * notice, this list of conditions and the following disclaimer in the
12 * documentation and/or other materials provided with the distribution.
13 * 3. The names of the authors may not be used to endorse or promote
14 * products derived from this software without specific prior written
15 * permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
18 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20 * ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
21 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
22 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
23 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
24 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
25 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
26 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
27 * SUCH DAMAGE.
28 */
29
30 #include <sys/cdefs.h>
31 __FBSDID("$FreeBSD$");
32
33 #ifdef _KERNEL
34 #include <sys/param.h>
35 #include <sys/systm.h>
36 #include <sys/malloc.h>
37 #include <sys/kernel.h>
38 #include <sys/module.h>
39 #include <sys/kthread.h>
40 #include <sys/conf.h>
41 #include <sys/bus.h>
42 #include <machine/bus.h>
43 #include <machine/resource.h>
44 #include <sys/rman.h>
45 #include <dev/pci/pcireg.h>
46 #include <dev/pci/pcivar.h>
47 #include <pci_if.h>
48 #include <vm/vm.h>
49 #include <vm/pmap.h>
50
51 #endif /* _KERNEL */
52
53 #include <sys/ioccom.h>
54
55 struct wave {
56 int index;
57 int period;
58 int offset;
59 int length;
60 int avg;
61 off_t mapvir;
62 int flags;
63
64 int npages;
65 void **virtual;
66 };
67
68 #define ADLINK_SETWAVE _IOWR('A', 232, struct wave)
69 #define ADLINK_GETWAVE _IOWR('A', 233, struct wave)
70
71 #ifdef _KERNEL
72
73 #define INTPERPAGE (PAGE_SIZE / sizeof(int))
74 #define I16PERPAGE (PAGE_SIZE / sizeof(int16_t))
75
76 /*
77 * Sample rate
78 */
79 #define SPS 1250000
80
81 /*
82 * We sample one channel (= 16 bits) at 1.25 msps giving 2.5Mbyte/sec,
83 * 100 pages will give us about 1/6 second buffering.
84 */
85 #define NRING 100
86
87 /*
88 * How many waves are we willing to entertain
89 */
90 #define NWAVE 25
91
92 struct info {
93 int nring;
94 off_t o_ring;
95
96 int ngri;
97 int ppgri;
98 off_t o_gri;
99 };
100
101 struct softc {
102 device_t device;
103 void *intrhand;
104 struct resource *r0, *r1, *ri;
105 bus_space_tag_t t0, t1;
106 bus_space_handle_t h0, h1;
107 struct cdev *dev;
108 off_t mapvir;
109
110 struct proc *procp;
111
112 struct info *info;
113
114 struct wave *wave[NWAVE];
115
116 int idx;
117 void *ring[NRING];
118 vm_paddr_t pring[NRING];
119 int stat[NRING];
120
121 uint64_t cnt;
122
123 u_char flags[I16PERPAGE];
124 };
125
126 static void
127 adlink_wave(struct softc *sc, struct wave *wp, int16_t *sp)
128 {
129 int f, i, k, m, *ip;
130
131 f = 0;
132 for (i = 0; i < I16PERPAGE; ) {
133 k = (sc->cnt - wp->offset + i) % wp->period;
134 if (k >= wp->length) {
135 i += wp->period - k;
136 sp += wp->period - k;
137 continue;
138 }
139 m = k % INTPERPAGE;
140 ip = (int *)(wp->virtual[k / INTPERPAGE]) + m;
141 while (m < INTPERPAGE && i < I16PERPAGE && k < wp->length) {
142 if (sc->flags[i] >= wp->index)
143 *ip += (*sp * 8 - *ip) >> wp->avg;
144 if (wp->flags & 1)
145 sc->flags[i] = wp->index;
146 sp++;
147 ip++;
148 m++;
149 i++;
150 k++;
151 }
152 }
153 }
154
155 static void
156 adlink_tickle(struct softc *sc)
157 {
158
159 wakeup(sc);
160 tsleep(&sc->ring, PUSER | PCATCH, "tickle", 1);
161 }
162
163 static int
164 adlink_new_wave(struct softc *sc, int index, int period, int offset, int length, int avg, int flags)
165 {
166 struct wave *wp;
167 int l, i;
168 void **oldvir, **newvir;
169
170 if (index < 0 || index >= NWAVE)
171 return (EINVAL);
172 wp = sc->wave[index];
173 if (wp == NULL) {
174 adlink_tickle(sc);
175 wp = malloc(sizeof *wp, M_DEVBUF, M_WAITOK | M_ZERO);
176 }
177 l = howmany(length, INTPERPAGE);
178 /* Setting a high average here to neuter the realtime bits */
179 wp->avg = 31;
180 if (wp->npages < l) {
181 oldvir = wp->virtual;
182 adlink_tickle(sc);
183 newvir = malloc(sizeof(void *) * l, M_DEVBUF, M_WAITOK | M_ZERO);
184 if (wp->npages > 0) {
185 adlink_tickle(sc);
186 bcopy(oldvir, newvir, wp->npages * sizeof(void *));
187 }
188 for (i = wp->npages; i < l; i++) {
189 adlink_tickle(sc);
190 newvir[i] = malloc(PAGE_SIZE, M_DEVBUF, M_WAITOK);
191 }
192 wp->virtual = newvir;
193 wp->npages = l;
194 wp->mapvir = sc->mapvir;
195 sc->mapvir += l * PAGE_SIZE;
196 } else {
197 oldvir = NULL;
198 }
199 wp->index = index;
200 wp->period = period;
201 wp->offset = offset;
202 wp->length = length;
203 wp->flags = flags;
204
205 for (i = 0; i < l; i++) {
206 adlink_tickle(sc);
207 bzero(wp->virtual[i], PAGE_SIZE);
208 }
209 wp->avg = avg;
210 sc->wave[index] = wp;
211 printf("Wave[%d] {period %d, offset %d, length %d, avg %d, flags %x}\n",
212 wp->index, wp->period, wp->offset, wp->length, wp->avg, wp->flags);
213 free(oldvir, M_DEVBUF);
214 return (0);
215 }
216
217 static void
218 adlink_loran(void *arg)
219 {
220 struct softc *sc;
221 int idx, i;
222
223 sc = arg;
224 idx = 0;
225 mtx_lock(&Giant);
226 for (;;) {
227 while (sc->stat[idx] == 0)
228 msleep(sc, NULL, PRIBIO, "loran", 1);
229 memset(sc->flags, NWAVE, sizeof sc->flags);
230 for (i = 0; i < NWAVE; i++) {
231 if (sc->wave[i] != NULL)
232 adlink_wave(sc, sc->wave[i], sc->ring[idx]);
233 }
234 sc->cnt += I16PERPAGE;
235 sc->stat[idx] = 0;
236 idx++;
237 idx %= NRING;
238 }
239 mtx_unlock(&Giant);
240 kthread_exit(0);
241 }
242
243 static int
244 adlink_open(struct cdev *dev, int oflags, int devtype, struct thread *td)
245 {
246 static int once;
247 struct softc *sc;
248 int i, error;
249 uint32_t u;
250
251 if (once)
252 return (0);
253 once = 1;
254
255 sc = dev->si_drv1;
256 sc->info = malloc(PAGE_SIZE, M_DEVBUF, M_ZERO | M_WAITOK);
257 sc->info->nring = NRING;
258
259 sc->info->o_ring = PAGE_SIZE;
260 for (i = 0; i < NRING; i++) {
261 sc->ring[i] = malloc(PAGE_SIZE, M_DEVBUF, M_ZERO | M_WAITOK);
262 sc->pring[i] = vtophys(sc->ring[i]);
263 }
264
265 error = adlink_new_wave(sc, NWAVE - 1, SPS, 0, SPS, 7, 0);
266 if (error)
267 return (error);
268
269 error = kthread_create(adlink_loran, sc, &sc->procp,
270 0, 0, "adlink%d", device_get_unit(sc->device));
271 if (error)
272 return (error);
273
274 /* Enable interrupts on write complete */
275 bus_space_write_4(sc->t0, sc->h0, 0x38, 0x00004000);
276
277 /* Sample CH0 only */
278 bus_space_write_4(sc->t1, sc->h1, 0x00, 1);
279
280 /* Divide clock by four */
281 bus_space_write_4(sc->t1, sc->h1, 0x04, 4);
282
283 /* Software trigger mode: software */
284 bus_space_write_4(sc->t1, sc->h1, 0x08, 0);
285
286 /* Trigger level zero */
287 bus_space_write_4(sc->t1, sc->h1, 0x0c, 0);
288
289 /* Trigger source CH0 (not used) */
290 bus_space_write_4(sc->t1, sc->h1, 0x10, 0);
291
292 /* Fifo control/status: flush */
293 bus_space_write_4(sc->t1, sc->h1, 0x18, 3);
294
295 /* Clock source: external sine */
296 bus_space_write_4(sc->t1, sc->h1, 0x20, 2);
297
298 /* Set up Write DMA */
299 bus_space_write_4(sc->t0, sc->h0, 0x24, sc->pring[i]);
300 bus_space_write_4(sc->t0, sc->h0, 0x28, PAGE_SIZE);
301 u = bus_space_read_4(sc->t0, sc->h0, 0x3c);
302 bus_space_write_4(sc->t0, sc->h0, 0x3c, u | 0x00000600);
303
304 /* Acquisition Enable Register: go! */
305 bus_space_write_4(sc->t1, sc->h1, 0x1c, 1);
306 return (0);
307 }
308
309 static int
310 adlink_ioctl(struct cdev *dev, u_long cmd, caddr_t data, int fflag, struct thread *td)
311 {
312 struct softc *sc;
313 struct wave *wp;
314 int i, error;
315
316 sc = dev->si_drv1;
317 wp = (struct wave *)data;
318 i = wp->index;
319 if (i < 0 || i >= NWAVE)
320 return (EINVAL);
321 if (cmd == ADLINK_GETWAVE) {
322 if (sc->wave[i] == NULL)
323 return (ENOENT);
324 bcopy(sc->wave[i], wp, sizeof(*wp));
325 return (0);
326 }
327 if (cmd == ADLINK_SETWAVE) {
328 error = adlink_new_wave(sc,
329 i,
330 wp->period,
331 wp->offset,
332 wp->length,
333 wp->avg,
334 wp->flags);
335 if (error)
336 return (error);
337 bcopy(sc->wave[i], wp, sizeof(*wp));
338 return (0);
339 }
340 return (ENOIOCTL);
341 }
342
343 static int
344 adlink_mmap(struct cdev *dev, vm_offset_t offset, vm_paddr_t *paddr, int nprot)
345 {
346 struct softc *sc;
347 struct wave *wp;
348 int i, j;
349
350 sc = dev->si_drv1;
351 if (nprot != VM_PROT_READ)
352 return (-1);
353 for (i = 0; i < NWAVE; i++) {
354 if (sc->wave[i] == NULL)
355 continue;
356 wp = sc->wave[i];
357 if (offset < wp->mapvir)
358 continue;
359 j = (offset - wp->mapvir) / PAGE_SIZE;
360 if (j >= wp->npages)
361 continue;
362 *paddr = vtophys(wp->virtual[j]);
363 return (0);
364 }
365 return (-1);
366 }
367
368 static void
369 adlink_intr(void *arg)
370 {
371 struct softc *sc;
372 uint32_t u;
373 int i, j;
374
375 sc = arg;
376 u = bus_space_read_4(sc->t0, sc->h0, 0x38);
377 if (!(u & 0x00800000))
378 return;
379 bus_space_write_4(sc->t0, sc->h0, 0x38, u | 0x003f4000);
380
381 j = sc->idx;
382 sc->stat[j] = 1;
383 i = (j + 1) % NRING;
384 sc->idx = i;
385 u = bus_space_read_4(sc->t1, sc->h1, 0x18);
386 if (u & 1) {
387 printf("adlink FIFO overrun\n");
388 return;
389 }
390 bus_space_write_4(sc->t0, sc->h0, 0x24, sc->pring[i]);
391 bus_space_write_4(sc->t0, sc->h0, 0x28, PAGE_SIZE);
392 wakeup(sc);
393 if (sc->stat[i]) {
394 printf("adlink page busy\n");
395 }
396 }
397
398 static struct cdevsw adlink_cdevsw = {
399 .d_version = D_VERSION,
400 .d_flags = D_NEEDGIANT,
401 .d_open = adlink_open,
402 .d_ioctl = adlink_ioctl,
403 .d_mmap = adlink_mmap,
404 .d_name = "adlink",
405 };
406
407 static devclass_t adlink_devclass;
408
409 static int
410 adlink_probe(device_t self)
411 {
412
413 if (pci_get_devid(self) != 0x80da10e8)
414 return (ENXIO);
415 device_set_desc(self, "Adlink PCI-9812 4 ch 12 bit 20 msps");
416 return (0);
417 }
418
419 static int
420 adlink_attach(device_t self)
421 {
422 struct softc *sc;
423 int rid, i;
424
425 sc = device_get_softc(self);
426 bzero(sc, sizeof *sc);
427 sc->device = self;
428
429 /*
430 * This is the PCI mapped registers of the AMCC 9535 "matchmaker"
431 * chip.
432 */
433 rid = 0x10;
434 sc->r0 = bus_alloc_resource_any(self, SYS_RES_IOPORT, &rid, RF_ACTIVE);
435 if (sc->r0 == NULL)
436 return(ENODEV);
437 sc->t0 = rman_get_bustag(sc->r0);
438 sc->h0 = rman_get_bushandle(sc->r0);
439 printf("Res0 %x %x\n", sc->t0, sc->h0);
440
441 /*
442 * This is the PCI mapped registers of the ADC hardware, they
443 * are described in the manual which comes with the card.
444 */
445 rid = 0x14;
446 sc->r1 = bus_alloc_resource_any(self, SYS_RES_IOPORT, &rid, RF_ACTIVE);
447 if (sc->r1 == NULL)
448 return(ENODEV);
449 sc->t1 = rman_get_bustag(sc->r1);
450 sc->h1 = rman_get_bushandle(sc->r1);
451 printf("Res1 %x %x\n", sc->t1, sc->h1);
452
453 rid = 0x0;
454 sc->ri = bus_alloc_resource_any(self, SYS_RES_IRQ, &rid,
455 RF_ACTIVE | RF_SHAREABLE);
456 if (sc->ri == NULL)
457 return (ENODEV);
458
459 i = bus_setup_intr(self, sc->ri, INTR_MPSAFE | INTR_TYPE_MISC | INTR_FAST,
460 adlink_intr, sc, &sc->intrhand);
461 if (i) {
462 printf("adlink: Couldn't get FAST intr\n");
463 i = bus_setup_intr(self, sc->ri, INTR_TYPE_MISC,
464 adlink_intr, sc, &sc->intrhand);
465 }
466
467 if (i)
468 return (ENODEV);
469
470 sc->dev = make_dev(&adlink_cdevsw, device_get_unit(self),
471 UID_ROOT, GID_WHEEL, 0444, "adlink%d", device_get_unit(self));
472 sc->dev->si_drv1 = sc;
473
474 return (0);
475 }
476
477 static device_method_t adlink_methods[] = {
478 /* Device interface */
479 DEVMETHOD(device_probe, adlink_probe),
480 DEVMETHOD(device_attach, adlink_attach),
481 DEVMETHOD(device_suspend, bus_generic_suspend),
482 DEVMETHOD(device_resume, bus_generic_resume),
483 DEVMETHOD(device_shutdown, bus_generic_shutdown),
484 {0, 0}
485 };
486
487 static driver_t adlink_driver = {
488 "adlink",
489 adlink_methods,
490 sizeof(struct softc)
491 };
492
493 DRIVER_MODULE(adlink, pci, adlink_driver, adlink_devclass, 0, 0);
494 #endif /* _KERNEL */
Cache object: 37e1ddba7601d9b5f3ddafe782982ec6
|