1 /* $NetBSD: swdmover.c,v 1.11 2007/07/12 14:15:55 he Exp $ */
4 * Copyright (c) 2002, 2003 Wasabi Systems, Inc.
7 * Written by Jason R. Thorpe for Wasabi Systems, Inc.
9 * Redistribution and use in source and binary forms, with or without
10 * modification, are permitted provided that the following conditions
12 * 1. Redistributions of source code must retain the above copyright
13 * notice, this list of conditions and the following disclaimer.
14 * 2. Redistributions in binary form must reproduce the above copyright
15 * notice, this list of conditions and the following disclaimer in the
16 * documentation and/or other materials provided with the distribution.
17 * 3. All advertising materials mentioning features or use of this software
18 * must display the following acknowledgement:
19 * This product includes software developed for the NetBSD Project by
20 * Wasabi Systems, Inc.
21 * 4. The name of Wasabi Systems, Inc. may not be used to endorse
22 * or promote products derived from this software without specific prior
25 * THIS SOFTWARE IS PROVIDED BY WASABI SYSTEMS, INC. ``AS IS'' AND
26 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
27 * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
28 * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL WASABI SYSTEMS, INC
29 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
30 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
31 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
32 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
33 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
34 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
35 * POSSIBILITY OF SUCH DAMAGE.
39 * swdmover.c: Software back-end providing the dmover functions
40 * mentioned in dmover(9).
42 * This module provides a fallback for cases where no hardware
43 * data movers are present in a system, and also serves an an
44 * example of how to write a dmover back-end.
46 * Note that even through the software dmover doesn't require
47 * interrupts to be blocked, we block them anyway to demonstrate
48 * the locking protocol.
51 #include <sys/cdefs.h>
52 __KERNEL_RCSID(0, "$NetBSD: swdmover.c,v 1.11 2007/07/12 14:15:55 he Exp $");
54 #include <sys/param.h>
55 #include <sys/kthread.h>
56 #include <sys/systm.h>
59 #include <dev/dmover/dmovervar.h>
61 struct swdmover_function
{
62 void (*sdf_process
)(struct dmover_request
*);
65 static struct dmover_backend swdmover_backend
;
66 static struct lwp
*swdmover_lwp
;
67 static int swdmover_cv
;
69 void swdmoverattach(int);
74 * Dmover back-end entry point.
77 swdmover_process(struct dmover_backend
*dmb
)
82 * Just wake up the processing thread. This will allow
83 * requests to linger on the middle-end's queue so that
84 * they can be cancelled, if need-be.
88 if (TAILQ_EMPTY(&dmb
->dmb_pendreqs
) == 0)
97 * Request processing thread.
100 swdmover_thread(void *arg
)
102 struct dmover_backend
*dmb
= arg
;
103 struct dmover_request
*dreq
;
104 struct swdmover_function
*sdf
;
111 dreq
= TAILQ_FIRST(&dmb
->dmb_pendreqs
);
114 (void) tsleep(&swdmover_cv
, PRIBIO
, "swdmvr", 0);
118 dmover_backend_remque(dmb
, dreq
);
119 dreq
->dreq_flags
|= DMOVER_REQ_RUNNING
;
124 sdf
= dreq
->dreq_assignment
->das_algdesc
->dad_data
;
125 (*sdf
->sdf_process
)(dreq
);
133 * swdmover_func_zero_process:
135 * Processing routine for the "zero" function.
138 swdmover_func_zero_process(struct dmover_request
*dreq
)
141 switch (dreq
->dreq_outbuf_type
) {
142 case DMOVER_BUF_LINEAR
:
143 memset(dreq
->dreq_outbuf
.dmbuf_linear
.l_addr
, 0,
144 dreq
->dreq_outbuf
.dmbuf_linear
.l_len
);
149 struct uio
*uio
= dreq
->dreq_outbuf
.dmbuf_uio
;
151 size_t count
, buflen
;
154 if (uio
->uio_rw
!= UIO_READ
) {
156 dreq
->dreq_error
= EINVAL
;
157 dreq
->dreq_flags
|= DMOVER_REQ_ERROR
;
162 buflen
= uio
->uio_resid
;
166 memset(cp
, 0, buflen
);
168 while ((count
= uio
->uio_resid
) != 0) {
171 error
= uiomove(cp
, count
, uio
);
174 dreq
->dreq_error
= error
;
175 dreq
->dreq_flags
|= DMOVER_REQ_ERROR
;
185 dreq
->dreq_error
= EINVAL
;
186 dreq
->dreq_flags
|= DMOVER_REQ_ERROR
;
194 * swdmover_func_fill8_process:
196 * Processing routine for the "fill8" function.
199 swdmover_func_fill8_process(struct dmover_request
*dreq
)
202 switch (dreq
->dreq_outbuf_type
) {
203 case DMOVER_BUF_LINEAR
:
204 memset(dreq
->dreq_outbuf
.dmbuf_linear
.l_addr
,
205 dreq
->dreq_immediate
[0],
206 dreq
->dreq_outbuf
.dmbuf_linear
.l_len
);
211 struct uio
*uio
= dreq
->dreq_outbuf
.dmbuf_uio
;
213 size_t count
, buflen
;
216 if (uio
->uio_rw
!= UIO_READ
) {
218 dreq
->dreq_error
= EINVAL
;
219 dreq
->dreq_flags
|= DMOVER_REQ_ERROR
;
224 buflen
= uio
->uio_resid
;
228 memset(cp
, dreq
->dreq_immediate
[0], buflen
);
230 while ((count
= uio
->uio_resid
) != 0) {
233 error
= uiomove(cp
, count
, uio
);
236 dreq
->dreq_error
= error
;
237 dreq
->dreq_flags
|= DMOVER_REQ_ERROR
;
247 dreq
->dreq_error
= EINVAL
;
248 dreq
->dreq_flags
|= DMOVER_REQ_ERROR
;
256 xor2(uint8_t *dst
, uint8_t *src1
, uint8_t *src2
, int cnt
)
260 *dst
++ = *src1
++ ^ *src2
++;
264 * swdmover_func_xor_process:
266 * Processing routine for the "xor" function.
269 swdmover_func_xor_process(struct dmover_request
*dreq
)
271 #define INBUF_L(x) dreq->dreq_inbuf[(x)].dmbuf_linear
272 #define OUTBUF_L dreq->dreq_outbuf.dmbuf_linear
274 uint32_t *dst32
, *src32
;
275 uint8_t *dst8
, *src8
;
276 int i
, ninputs
= dreq
->dreq_assignment
->das_algdesc
->dad_ninputs
;
277 int aligned
, len
, nwords
;
279 /* XXX Currently, both buffers must be of same type. */
280 if (dreq
->dreq_inbuf_type
!= dreq
->dreq_outbuf_type
) {
282 dreq
->dreq_error
= EINVAL
;
283 dreq
->dreq_flags
|= DMOVER_REQ_ERROR
;
288 switch (dreq
->dreq_outbuf_type
) {
289 case DMOVER_BUF_LINEAR
:
291 if ((ulong
) OUTBUF_L
.l_addr
& 0x3)
293 len
= OUTBUF_L
.l_len
;
294 for (i
= 0 ; i
< ninputs
; i
++) {
295 if (len
!= INBUF_L(i
).l_len
) {
297 dreq
->dreq_error
= EINVAL
;
298 dreq
->dreq_flags
|= DMOVER_REQ_ERROR
;
302 if ((ulong
) INBUF_L(i
).l_addr
& 0x3)
306 dst32
= (uint32_t *) OUTBUF_L
.l_addr
;
310 for (i
= 0 ; i
< ninputs
; i
++) {
311 src32
= (uint32_t *) INBUF_L(i
).l_addr
;
319 dst8
= (uint8_t *) OUTBUF_L
.l_addr
;
322 for (i
= 0 ; i
< ninputs
; i
++) {
323 src8
= (uint8_t *) INBUF_L(i
).l_addr
;
334 struct uio
*uio_out
= dreq
->dreq_outbuf
.dmbuf_uio
;
335 struct uio
*uio_in
= dreq
->dreq_inbuf
[0].dmbuf_uio
;
338 size_t count
, buflen
;
341 if (uio_in
->uio_rw
!= UIO_WRITE
||
342 uio_out
->uio_rw
!= UIO_READ
||
343 uio_in
->uio_resid
!= uio_out
->uio_resid
) {
345 dreq
->dreq_error
= EINVAL
;
346 dreq
->dreq_flags
|= DMOVER_REQ_ERROR
;
351 buflen
= uio_in
->uio_resid
;
355 dst
= alloca(buflen
);
358 * For each block, copy first input buffer into the destination
359 * buffer and then read the rest, one by one, into a temporary
360 * buffer and xor into the destination buffer. After all of
361 * the inputs have been xor'd in, move the destination buffer
364 while ((count
= uio_in
->uio_resid
) != 0) {
367 error
= uiomove(dst
, count
, uio_in
);
370 dreq
->dreq_error
= error
;
371 dreq
->dreq_flags
|= DMOVER_REQ_ERROR
;
375 for (i
=1 ; (i
< ninputs
) && (error
== 0) ; i
++) {
376 uio
= dreq
->dreq_inbuf
[i
].dmbuf_uio
;
377 error
= uiomove(cp
, count
, uio
);
379 xor2(dst
, dst
, cp
, count
);
383 error
= uiomove(dst
, count
, uio_out
);
386 dreq
->dreq_error
= error
;
387 dreq
->dreq_flags
|= DMOVER_REQ_ERROR
;
397 dreq
->dreq_error
= EINVAL
;
398 dreq
->dreq_flags
|= DMOVER_REQ_ERROR
;
407 * swdmover_func_copy_process:
409 * Processing routine for the "copy" function.
412 swdmover_func_copy_process(struct dmover_request
*dreq
)
415 /* XXX Currently, both buffers must be of same type. */
416 if (dreq
->dreq_inbuf_type
!= dreq
->dreq_outbuf_type
) {
418 dreq
->dreq_error
= EINVAL
;
419 dreq
->dreq_flags
|= DMOVER_REQ_ERROR
;
424 switch (dreq
->dreq_outbuf_type
) {
425 case DMOVER_BUF_LINEAR
:
426 if (dreq
->dreq_outbuf
.dmbuf_linear
.l_len
!=
427 dreq
->dreq_inbuf
[0].dmbuf_linear
.l_len
) {
429 dreq
->dreq_error
= EINVAL
;
430 dreq
->dreq_flags
|= DMOVER_REQ_ERROR
;
434 memcpy(dreq
->dreq_outbuf
.dmbuf_linear
.l_addr
,
435 dreq
->dreq_inbuf
[0].dmbuf_linear
.l_addr
,
436 dreq
->dreq_outbuf
.dmbuf_linear
.l_len
);
441 struct uio
*uio_out
= dreq
->dreq_outbuf
.dmbuf_uio
;
442 struct uio
*uio_in
= dreq
->dreq_inbuf
[0].dmbuf_uio
;
444 size_t count
, buflen
;
447 if (uio_in
->uio_rw
!= UIO_WRITE
||
448 uio_out
->uio_rw
!= UIO_READ
||
449 uio_in
->uio_resid
!= uio_out
->uio_resid
) {
451 dreq
->dreq_error
= EINVAL
;
452 dreq
->dreq_flags
|= DMOVER_REQ_ERROR
;
457 buflen
= uio_in
->uio_resid
;
462 while ((count
= uio_in
->uio_resid
) != 0) {
465 error
= uiomove(cp
, count
, uio_in
);
467 error
= uiomove(cp
, count
, uio_out
);
470 dreq
->dreq_error
= error
;
471 dreq
->dreq_flags
|= DMOVER_REQ_ERROR
;
481 dreq
->dreq_error
= EINVAL
;
482 dreq
->dreq_flags
|= DMOVER_REQ_ERROR
;
490 static const uint32_t iscsi_crc32c_table
[256] = {
491 0x00000000, 0xf26b8303, 0xe13b70f7, 0x1350f3f4,
492 0xc79a971f, 0x35f1141c, 0x26a1e7e8, 0xd4ca64eb,
493 0x8ad958cf, 0x78b2dbcc, 0x6be22838, 0x9989ab3b,
494 0x4d43cfd0, 0xbf284cd3, 0xac78bf27, 0x5e133c24,
495 0x105ec76f, 0xe235446c, 0xf165b798, 0x030e349b,
496 0xd7c45070, 0x25afd373, 0x36ff2087, 0xc494a384,
497 0x9a879fa0, 0x68ec1ca3, 0x7bbcef57, 0x89d76c54,
498 0x5d1d08bf, 0xaf768bbc, 0xbc267848, 0x4e4dfb4b,
499 0x20bd8ede, 0xd2d60ddd, 0xc186fe29, 0x33ed7d2a,
500 0xe72719c1, 0x154c9ac2, 0x061c6936, 0xf477ea35,
501 0xaa64d611, 0x580f5512, 0x4b5fa6e6, 0xb93425e5,
502 0x6dfe410e, 0x9f95c20d, 0x8cc531f9, 0x7eaeb2fa,
503 0x30e349b1, 0xc288cab2, 0xd1d83946, 0x23b3ba45,
504 0xf779deae, 0x05125dad, 0x1642ae59, 0xe4292d5a,
505 0xba3a117e, 0x4851927d, 0x5b016189, 0xa96ae28a,
506 0x7da08661, 0x8fcb0562, 0x9c9bf696, 0x6ef07595,
507 0x417b1dbc, 0xb3109ebf, 0xa0406d4b, 0x522bee48,
508 0x86e18aa3, 0x748a09a0, 0x67dafa54, 0x95b17957,
509 0xcba24573, 0x39c9c670, 0x2a993584, 0xd8f2b687,
510 0x0c38d26c, 0xfe53516f, 0xed03a29b, 0x1f682198,
511 0x5125dad3, 0xa34e59d0, 0xb01eaa24, 0x42752927,
512 0x96bf4dcc, 0x64d4cecf, 0x77843d3b, 0x85efbe38,
513 0xdbfc821c, 0x2997011f, 0x3ac7f2eb, 0xc8ac71e8,
514 0x1c661503, 0xee0d9600, 0xfd5d65f4, 0x0f36e6f7,
515 0x61c69362, 0x93ad1061, 0x80fde395, 0x72966096,
516 0xa65c047d, 0x5437877e, 0x4767748a, 0xb50cf789,
517 0xeb1fcbad, 0x197448ae, 0x0a24bb5a, 0xf84f3859,
518 0x2c855cb2, 0xdeeedfb1, 0xcdbe2c45, 0x3fd5af46,
519 0x7198540d, 0x83f3d70e, 0x90a324fa, 0x62c8a7f9,
520 0xb602c312, 0x44694011, 0x5739b3e5, 0xa55230e6,
521 0xfb410cc2, 0x092a8fc1, 0x1a7a7c35, 0xe811ff36,
522 0x3cdb9bdd, 0xceb018de, 0xdde0eb2a, 0x2f8b6829,
523 0x82f63b78, 0x709db87b, 0x63cd4b8f, 0x91a6c88c,
524 0x456cac67, 0xb7072f64, 0xa457dc90, 0x563c5f93,
525 0x082f63b7, 0xfa44e0b4, 0xe9141340, 0x1b7f9043,
526 0xcfb5f4a8, 0x3dde77ab, 0x2e8e845f, 0xdce5075c,
527 0x92a8fc17, 0x60c37f14, 0x73938ce0, 0x81f80fe3,
528 0x55326b08, 0xa759e80b, 0xb4091bff, 0x466298fc,
529 0x1871a4d8, 0xea1a27db, 0xf94ad42f, 0x0b21572c,
530 0xdfeb33c7, 0x2d80b0c4, 0x3ed04330, 0xccbbc033,
531 0xa24bb5a6, 0x502036a5, 0x4370c551, 0xb11b4652,
532 0x65d122b9, 0x97baa1ba, 0x84ea524e, 0x7681d14d,
533 0x2892ed69, 0xdaf96e6a, 0xc9a99d9e, 0x3bc21e9d,
534 0xef087a76, 0x1d63f975, 0x0e330a81, 0xfc588982,
535 0xb21572c9, 0x407ef1ca, 0x532e023e, 0xa145813d,
536 0x758fe5d6, 0x87e466d5, 0x94b49521, 0x66df1622,
537 0x38cc2a06, 0xcaa7a905, 0xd9f75af1, 0x2b9cd9f2,
538 0xff56bd19, 0x0d3d3e1a, 0x1e6dcdee, 0xec064eed,
539 0xc38d26c4, 0x31e6a5c7, 0x22b65633, 0xd0ddd530,
540 0x0417b1db, 0xf67c32d8, 0xe52cc12c, 0x1747422f,
541 0x49547e0b, 0xbb3ffd08, 0xa86f0efc, 0x5a048dff,
542 0x8ecee914, 0x7ca56a17, 0x6ff599e3, 0x9d9e1ae0,
543 0xd3d3e1ab, 0x21b862a8, 0x32e8915c, 0xc083125f,
544 0x144976b4, 0xe622f5b7, 0xf5720643, 0x07198540,
545 0x590ab964, 0xab613a67, 0xb831c993, 0x4a5a4a90,
546 0x9e902e7b, 0x6cfbad78, 0x7fab5e8c, 0x8dc0dd8f,
547 0xe330a81a, 0x115b2b19, 0x020bd8ed, 0xf0605bee,
548 0x24aa3f05, 0xd6c1bc06, 0xc5914ff2, 0x37faccf1,
549 0x69e9f0d5, 0x9b8273d6, 0x88d28022, 0x7ab90321,
550 0xae7367ca, 0x5c18e4c9, 0x4f48173d, 0xbd23943e,
551 0xf36e6f75, 0x0105ec76, 0x12551f82, 0xe03e9c81,
552 0x34f4f86a, 0xc69f7b69, 0xd5cf889d, 0x27a40b9e,
553 0x79b737ba, 0x8bdcb4b9, 0x988c474d, 0x6ae7c44e,
554 0xbe2da0a5, 0x4c4623a6, 0x5f16d052, 0xad7d5351,
558 iscsi_crc32c(const uint8_t *buf
, size_t len
, uint32_t last
)
560 uint32_t crc
= 0xffffffffU
^ last
;
563 crc
= iscsi_crc32c_table
[(crc
^ *buf
++) & 0xff] ^ (crc
>> 8);
565 return (crc
^ 0xffffffffU
);
569 * swdmover_func_iscsi_crc32c_process:
571 * Processing routine for the "iscsi-crc32c" function.
574 swdmover_func_iscsi_crc32c_process(struct dmover_request
*dreq
)
578 /* No output buffer; we use the immediate only. */
579 if (dreq
->dreq_outbuf_type
!= DMOVER_BUF_NONE
) {
581 dreq
->dreq_error
= EINVAL
;
582 dreq
->dreq_flags
|= DMOVER_REQ_ERROR
;
587 memcpy(&result
, dreq
->dreq_immediate
, sizeof(result
));
589 switch (dreq
->dreq_inbuf_type
) {
590 case DMOVER_BUF_LINEAR
:
591 result
= iscsi_crc32c(dreq
->dreq_inbuf
[0].dmbuf_linear
.l_addr
,
592 dreq
->dreq_inbuf
[0].dmbuf_linear
.l_len
, result
);
597 struct uio
*uio_in
= dreq
->dreq_inbuf
[0].dmbuf_uio
;
599 size_t count
, buflen
;
602 if (uio_in
->uio_rw
!= UIO_WRITE
) {
604 dreq
->dreq_error
= EINVAL
;
605 dreq
->dreq_flags
|= DMOVER_REQ_ERROR
;
610 buflen
= uio_in
->uio_resid
;
615 while ((count
= uio_in
->uio_resid
) != 0) {
618 error
= uiomove(cp
, count
, uio_in
);
621 dreq
->dreq_error
= error
;
622 dreq
->dreq_flags
|= DMOVER_REQ_ERROR
;
626 result
= iscsi_crc32c(cp
, count
, result
);
633 dreq
->dreq_error
= EINVAL
;
634 dreq
->dreq_flags
|= DMOVER_REQ_ERROR
;
639 memcpy(dreq
->dreq_immediate
, &result
, sizeof(result
));
644 static struct swdmover_function swdmover_func_zero
= {
645 swdmover_func_zero_process
648 static struct swdmover_function swdmover_func_fill8
= {
649 swdmover_func_fill8_process
652 static struct swdmover_function swdmover_func_copy
= {
653 swdmover_func_copy_process
656 static struct swdmover_function swdmover_func_xor
= {
657 swdmover_func_xor_process
660 static struct swdmover_function swdmover_func_iscsi_crc32c
= {
661 swdmover_func_iscsi_crc32c_process
664 const struct dmover_algdesc swdmover_algdescs
[] = {
707 &swdmover_func_fill8
,
716 DMOVER_FUNC_ISCSI_CRC32C
,
717 &swdmover_func_iscsi_crc32c
,
721 #define SWDMOVER_ALGDESC_COUNT \
722 (sizeof(swdmover_algdescs) / sizeof(swdmover_algdescs[0]))
727 * Pesudo-device attach routine.
730 swdmoverattach(int count
)
734 swdmover_backend
.dmb_name
= "swdmover";
735 swdmover_backend
.dmb_speed
= 1; /* XXX */
736 swdmover_backend
.dmb_cookie
= NULL
;
737 swdmover_backend
.dmb_algdescs
= swdmover_algdescs
;
738 swdmover_backend
.dmb_nalgdescs
= SWDMOVER_ALGDESC_COUNT
;
739 swdmover_backend
.dmb_process
= swdmover_process
;
741 error
= kthread_create(PRI_NONE
, 0, NULL
, swdmover_thread
,
742 &swdmover_backend
, &swdmover_lwp
, "swdmover");
744 printf("WARNING: unable to create swdmover thread, "
745 "error = %d\n", error
);
747 /* XXX Should only register this when kthread creation succeeds. */
748 dmover_backend_register(&swdmover_backend
);