2 * Copyright (c) 2002-2003
3 * Hidetoshi Shimokawa. All rights reserved.
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
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. All advertising materials mentioning features or use of this software
14 * must display the following acknowledgement:
16 * This product includes software developed by Hidetoshi Shimokawa.
18 * 4. Neither the name of the author nor the names of its contributors
19 * may be used to endorse or promote products derived from this software
20 * without specific prior written permission.
22 * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
23 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
24 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
25 * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
26 * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
27 * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
28 * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
29 * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
30 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
31 * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
36 #include <sys/cdefs.h>
37 __FBSDID("$FreeBSD: src/sys/dev/firewire/fwcrom.c,v 1.14 2006/02/04 21:37:39 imp Exp $");
40 #include <sys/param.h>
44 #include <bootstrap.h>
46 #if defined(_KERNEL) || defined(TEST)
50 #include <sys/queue.h>
55 #include <sys/systm.h>
56 #include <sys/kernel.h>
59 #include <KernelExport.h>
60 #include <ByteOrder.h>
66 #include <netinet/in.h>
78 #if defined( __DragonFly__) || defined(__HAIKU__)
81 #define vm_offset_t uint32_t
83 #include <dev/firewire/firewire.h>
84 #include <dev/firewire/iec13213.h>
87 #define MAX_ROM (1024 - sizeof(uint32_t) * 5)
88 #define CROM_END(cc) ((vm_offset_t)(cc)->stack[0].dir + MAX_ROM - 1)
91 crom_init_context(struct crom_context
*cc
, uint32_t *p
)
95 hdr
= (struct csrhdr
*)p
;
96 if (hdr
->info_len
<= 1) {
97 /* minimum or invalid ROM */
101 p
+= 1 + hdr
->info_len
;
103 /* check size of root directory */
104 if (((struct csrdirectory
*)p
)->crc_len
== 0) {
109 cc
->stack
[0].dir
= (struct csrdirectory
*)p
;
110 cc
->stack
[0].index
= 0;
114 crom_get(struct crom_context
*cc
)
116 struct crom_ptr
*ptr
;
118 ptr
= &cc
->stack
[cc
->depth
];
119 return (&ptr
->dir
->entry
[ptr
->index
]);
123 crom_next(struct crom_context
*cc
)
125 struct crom_ptr
*ptr
;
131 if ((reg
->key
& CSRTYPE_MASK
) == CSRTYPE_D
) {
132 if (cc
->depth
>= CROM_MAX_DEPTH
) {
133 printf("crom_next: too deep\n");
138 ptr
= &cc
->stack
[cc
->depth
];
139 ptr
->dir
= (struct csrdirectory
*) (reg
+ reg
->val
);
144 ptr
= &cc
->stack
[cc
->depth
];
147 if (ptr
->index
< ptr
->dir
->crc_len
&&
148 (vm_offset_t
)crom_get(cc
) <= CROM_END(cc
))
151 if (ptr
->index
< ptr
->dir
->crc_len
)
152 printf("crom_next: bound check failed\n");
164 crom_search_key(struct crom_context
*cc
, uint8_t key
)
168 while(cc
->depth
>= 0) {
178 crom_has_specver(uint32_t *p
, uint32_t spec
, uint32_t ver
)
181 struct crom_context c
, *cc
;
185 crom_init_context(cc
, p
);
186 while(cc
->depth
>= 0) {
189 if (reg
->key
== CSRKEY_SPEC
&& reg
->val
== spec
)
194 if (reg
->key
== CSRKEY_VER
&& reg
->val
== ver
)
205 crom_parse_text(struct crom_context
*cc
, char *buf
, int len
)
208 struct csrtext
*textleaf
;
211 static char *nullstr
= "(null)";
217 if (reg
->key
!= CROM_TEXTLEAF
||
218 (vm_offset_t
)(reg
+ reg
->val
) > CROM_END(cc
)) {
219 strncpy(buf
, nullstr
, len
);
222 textleaf
= (struct csrtext
*)(reg
+ reg
->val
);
224 if ((vm_offset_t
)textleaf
+ textleaf
->crc_len
> CROM_END(cc
)) {
225 strncpy(buf
, nullstr
, len
);
229 /* XXX should check spec and type */
231 bp
= (uint32_t *)&buf
[0];
232 qlen
= textleaf
->crc_len
- 2;
235 for (i
= 0; i
< qlen
; i
++)
236 *bp
++ = ntohl(textleaf
->text
[i
]);
237 /* make sure to terminate the string */
245 crom_crc(uint32_t *ptr
, int len
)
248 uint32_t data
, sum
, crc
= 0;
250 for (i
= 0; i
< len
; i
++) {
252 for (shift
= 28; shift
>= 0; shift
-= 4) {
253 sum
= ((crc
>> 12) ^ (data
>> shift
)) & 0xf;
254 crc
= (crc
<< 4) ^ (sum
<< 12) ^ (sum
<< 5) ^ sum
;
258 return((uint16_t) crc
);
261 #if !defined(_KERNEL) && !defined(_BOOT)
263 crom_desc_specver(uint32_t spec
, uint32_t ver
, char *buf
, int len
)
267 if (spec
== CSRVAL_ANSIT10
|| spec
== 0) {
274 s
= "unknown ANSIT10";
277 if (spec
== CSRVAL_1394TA
|| spec
== 0) {
301 s
= "1394 Direct print";
304 s
= "Industrial & Instrument";
308 s
= "unknown 1394TA";
312 snprintf(buf
, len
, "%s", s
);
316 crom_desc(struct crom_context
*cc
, char *buf
, int len
)
319 struct csrdirectory
*dir
;
324 switch (reg
->key
& CSRTYPE_MASK
) {
327 len
-= snprintf(buf
, len
, "%d", reg
->val
);
334 len
-= snprintf(buf
, len
, "offset=0x%04x(%d)",
339 /* XXX fall through */
341 dir
= (struct csrdirectory
*) (reg
+ reg
->val
);
342 crc
= crom_crc((uint32_t *)&dir
->entry
[0], dir
->crc_len
);
343 len
-= snprintf(buf
, len
, "len=%d crc=0x%04x(%s) ",
344 dir
->crc_len
, dir
->crc
,
345 (crc
== dir
->crc
) ? "OK" : "NG");
350 desc
= "module_vendor_ID";
353 desc
= "hardware_version";
356 desc
= "node_capabilities";
359 desc
= "unit_spec_ID";
362 desc
= "unit_sw_version";
363 crom_desc_specver(0, reg
->val
, buf
, len
);
366 desc
= "logical_unit_number";
372 desc
= "command_set_spec_ID";
375 desc
= "command_set";
378 desc
= "unit_characteristics";
381 desc
= "command_set_revision";
384 desc
= "firmware_revision";
387 desc
= "reconnect_timeout";
390 desc
= "management_agent";
394 crom_parse_text(cc
, buf
+ strlen(buf
), len
);
397 desc
= "unit_directory";
400 desc
= "logical_unit_directory";
409 #if defined(_KERNEL) || defined(_BOOT) || defined(TEST)
412 crom_add_quad(struct crom_chunk
*chunk
, uint32_t entry
)
416 index
= chunk
->data
.crc_len
;
417 if (index
>= CROM_MAX_CHUNK_LEN
- 1) {
418 printf("too large chunk %d\n", index
);
421 chunk
->data
.buf
[index
] = entry
;
422 chunk
->data
.crc_len
++;
427 crom_add_entry(struct crom_chunk
*chunk
, int key
, int val
)
437 return (crom_add_quad(chunk
, foo
.i
));
441 crom_add_chunk(struct crom_src
*src
, struct crom_chunk
*parent
,
442 struct crom_chunk
*child
, int key
)
446 if (parent
== NULL
) {
447 STAILQ_INSERT_TAIL(&src
->chunk_list
, child
, link
);
451 index
= crom_add_entry(parent
, key
, 0);
455 child
->ref_chunk
= parent
;
456 child
->ref_index
= index
;
457 STAILQ_INSERT_TAIL(&src
->chunk_list
, child
, link
);
461 #define MAX_TEXT ((CROM_MAX_CHUNK_LEN + 1) * 4 - sizeof(struct csrtext))
463 crom_add_simple_text(struct crom_src
*src
, struct crom_chunk
*parent
,
464 struct crom_chunk
*chunk
, const char *buf
)
472 if (len
> MAX_TEXT
) {
473 #if defined(__DragonFly__) || __FreeBSD_version < 500000
474 printf("text(%d) truncated to %zd.\n", len
, MAX_TEXT
);
476 printf("text(%d) truncated to %zd.\n", len
, MAX_TEXT
);
481 tl
= (struct csrtext
*) &chunk
->data
;
482 tl
->crc_len
= howmany(sizeof(struct csrtext
) + len
, sizeof(uint32_t));
486 bzero(&t
[0], roundup2(len
, sizeof(uint32_t)));
487 bcopy(buf
, &t
[0], len
);
488 p
= (uint32_t *)&t
[0];
489 for (i
= 0; i
< howmany(len
, sizeof(uint32_t)); i
++)
490 tl
->text
[i
] = ntohl(*p
++);
491 return (crom_add_chunk(src
, parent
, chunk
, CROM_TEXTLEAF
));
495 crom_copy(uint32_t *src
, uint32_t *dst
, int *offset
, int len
, int maxlen
)
497 if (*offset
+ len
> maxlen
) {
498 printf("Config. ROM is too large for the buffer\n");
501 bcopy(src
, (char *)(dst
+ *offset
), len
* sizeof(uint32_t));
507 crom_load(struct crom_src
*src
, uint32_t *buf
, int maxlen
)
509 struct crom_chunk
*chunk
, *parent
;
511 #if defined(_KERNEL) || defined(_BOOT)
519 /* Determine offset */
520 STAILQ_FOREACH(chunk
, &src
->chunk_list
, link
) {
521 chunk
->offset
= offset
;
522 /* Assume the offset of the parent is already known */
523 parent
= chunk
->ref_chunk
;
524 if (parent
!= NULL
) {
526 reg
= (struct csrreg
*)
527 &parent
->data
.buf
[chunk
->ref_index
];
529 (parent
->offset
+ 1 + chunk
->ref_index
);
531 offset
+= 1 + chunk
->data
.crc_len
;
534 /* Calculate CRC and dump to the buffer */
535 len
= 1 + src
->hdr
.info_len
;
537 if (crom_copy((uint32_t *)&src
->hdr
, buf
, &count
, len
, maxlen
) < 0)
539 STAILQ_FOREACH(chunk
, &src
->chunk_list
, link
) {
541 crom_crc(&chunk
->data
.buf
[0], chunk
->data
.crc_len
);
543 len
= 1 + chunk
->data
.crc_len
;
544 if (crom_copy((uint32_t *)&chunk
->data
, buf
,
545 &count
, len
, maxlen
) < 0)
548 hdr
= (struct csrhdr
*)buf
;
549 hdr
->crc_len
= count
- 1;
550 hdr
->crc
= crom_crc(&buf
[1], hdr
->crc_len
);
552 #if defined(_KERNEL) || defined(_BOOT)
555 for (i
= 0; i
< count
; i
++) {
569 struct crom_chunk root
,unit1
,unit2
,unit3
;
570 struct crom_chunk text1
,text2
,text3
,text4
,text5
,text6
,text7
;
571 uint32_t buf
[256], *p
;
574 bzero(&src
, sizeof(src
));
575 bzero(&root
, sizeof(root
));
576 bzero(&unit1
, sizeof(unit1
));
577 bzero(&unit2
, sizeof(unit2
));
578 bzero(&unit3
, sizeof(unit3
));
579 bzero(&text1
, sizeof(text1
));
580 bzero(&text2
, sizeof(text2
));
581 bzero(&text3
, sizeof(text3
));
582 bzero(&text3
, sizeof(text4
));
583 bzero(&text3
, sizeof(text5
));
584 bzero(&text3
, sizeof(text6
));
585 bzero(&text3
, sizeof(text7
));
586 bzero(buf
, sizeof(buf
));
588 /* BUS info sample */
589 src
.hdr
.info_len
= 4;
590 src
.businfo
.bus_name
= CSR_BUS_NAME_IEEE1394
;
591 src
.businfo
.eui64
.hi
= 0x11223344;
592 src
.businfo
.eui64
.lo
= 0x55667788;
593 src
.businfo
.link_spd
= FWSPD_S400
;
594 src
.businfo
.generation
= 0;
595 src
.businfo
.max_rom
= MAXROM_4
;
596 src
.businfo
.max_rec
= 10;
597 src
.businfo
.cyc_clk_acc
= 100;
602 src
.businfo
.irmc
= 1;
603 STAILQ_INIT(&src
.chunk_list
);
606 crom_add_chunk(&src
, NULL
, &root
, 0);
607 crom_add_entry(&root
, CSRKEY_NCAP
, 0x123456);
608 /* private company_id */
609 crom_add_entry(&root
, CSRKEY_VENDOR
, 0xacde48);
612 crom_add_simple_text(&src
, &root
, &text1
, "DragonFly");
613 crom_add_entry(&root
, CSRKEY_HW
, __DragonFly_cc_version
);
614 crom_add_simple_text(&src
, &root
, &text2
, "DragonFly-1");
616 crom_add_simple_text(&src
, &root
, &text1
, "FreeBSD");
617 crom_add_entry(&root
, CSRKEY_HW
, __FreeBSD_version
);
618 crom_add_simple_text(&src
, &root
, &text2
, "FreeBSD-5");
621 /* SBP unit directory */
622 crom_add_chunk(&src
, &root
, &unit1
, CROM_UDIR
);
623 crom_add_entry(&unit1
, CSRKEY_SPEC
, CSRVAL_ANSIT10
);
624 crom_add_entry(&unit1
, CSRKEY_VER
, CSRVAL_T10SBP2
);
625 crom_add_entry(&unit1
, CSRKEY_COM_SPEC
, CSRVAL_ANSIT10
);
626 crom_add_entry(&unit1
, CSRKEY_COM_SET
, CSRVAL_SCSI
);
627 /* management_agent */
628 crom_add_entry(&unit1
, CROM_MGM
, 0x1000);
629 crom_add_entry(&unit1
, CSRKEY_UNIT_CH
, (10<<8) | 8);
630 /* Device type and LUN */
631 crom_add_entry(&unit1
, CROM_LUN
, 0);
632 crom_add_entry(&unit1
, CSRKEY_MODEL
, 1);
633 crom_add_simple_text(&src
, &unit1
, &text3
, "scsi_target");
635 /* RFC2734 IPv4 over IEEE1394 */
636 crom_add_chunk(&src
, &root
, &unit2
, CROM_UDIR
);
637 crom_add_entry(&unit2
, CSRKEY_SPEC
, CSRVAL_IETF
);
638 crom_add_simple_text(&src
, &unit2
, &text4
, "IANA");
639 crom_add_entry(&unit2
, CSRKEY_VER
, 1);
640 crom_add_simple_text(&src
, &unit2
, &text5
, "IPv4");
642 /* RFC3146 IPv6 over IEEE1394 */
643 crom_add_chunk(&src
, &root
, &unit3
, CROM_UDIR
);
644 crom_add_entry(&unit3
, CSRKEY_SPEC
, CSRVAL_IETF
);
645 crom_add_simple_text(&src
, &unit3
, &text6
, "IANA");
646 crom_add_entry(&unit3
, CSRKEY_VER
, 2);
647 crom_add_simple_text(&src
, &unit3
, &text7
, "IPv6");
649 crom_load(&src
, buf
, 256);
651 #define DUMP_FORMAT "%08x %08x %08x %08x %08x %08x %08x %08x\n"
652 for (i
= 0; i
< 256/8; i
++) {
654 p
[0], p
[1], p
[2], p
[3], p
[4], p
[5], p
[6], p
[7]);