1 /*
2 * Driver O/S-independent utility routines
3 *
4 * Copyright (C) 1999-2013, Broadcom Corporation
5 *
6 * Permission to use, copy, modify, and/or distribute this software for any
7 * purpose with or without fee is hereby granted, provided that the above
8 * copyright notice and this permission notice appear in all copies.
9 *
10 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
11 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
12 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY
13 * SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
14 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION
15 * OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF OR IN
16 * CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
17 *
18 * $Id: bcmutils.c 380908 2013-01-24 12:26:18Z $
19 */
20
21 #include <bcm_cfg.h>
22 #include <typedefs.h>
23 #include <bcmdefs.h>
24 #include <stdarg.h>
25 #ifdef BCMDRIVER
26
27 #include <osl.h>
28 #include <bcmutils.h>
29
30 #else /* !BCMDRIVER */
31
32 #include <stdio.h>
33 #include <string.h>
34 #include <bcmutils.h>
35
36 #if defined(BCMEXTSUP)
37 #include <bcm_osl.h>
38 #endif
39
40
41 #endif /* !BCMDRIVER */
42
43 #include <bcmendian.h>
44 #include <bcmdevs.h>
45 #include <proto/ethernet.h>
46 #include <proto/vlan.h>
47 #include <proto/bcmip.h>
48 #include <proto/802.1d.h>
49 #include <proto/802.11.h>
50 void *_bcmutils_dummy_fn = NULL;
51
52
53 #ifdef BCMDRIVER
54
55
56
57 /* copy a pkt buffer chain into a buffer */
58 uint
pktcopy(osl_t * osh,void * p,uint offset,int len,uchar * buf)59 pktcopy(osl_t *osh, void *p, uint offset, int len, uchar *buf)
60 {
61 uint n, ret = 0;
62
63 if (len < 0)
64 len = 4096; /* "infinite" */
65
66 /* skip 'offset' bytes */
67 for (; p && offset; p = PKTNEXT(osh, p)) {
68 if (offset < (uint)PKTLEN(osh, p))
69 break;
70 offset -= PKTLEN(osh, p);
71 }
72
73 if (!p)
74 return 0;
75
76 /* copy the data */
77 for (; p && len; p = PKTNEXT(osh, p)) {
78 n = MIN((uint)PKTLEN(osh, p) - offset, (uint)len);
79 bcopy(PKTDATA(osh, p) + offset, buf, n);
80 buf += n;
81 len -= n;
82 ret += n;
83 offset = 0;
84 }
85
86 return ret;
87 }
88
89 /* copy a buffer into a pkt buffer chain */
90 uint
pktfrombuf(osl_t * osh,void * p,uint offset,int len,uchar * buf)91 pktfrombuf(osl_t *osh, void *p, uint offset, int len, uchar *buf)
92 {
93 uint n, ret = 0;
94
95 /* skip 'offset' bytes */
96 for (; p && offset; p = PKTNEXT(osh, p)) {
97 if (offset < (uint)PKTLEN(osh, p))
98 break;
99 offset -= PKTLEN(osh, p);
100 }
101
102 if (!p)
103 return 0;
104
105 /* copy the data */
106 for (; p && len; p = PKTNEXT(osh, p)) {
107 n = MIN((uint)PKTLEN(osh, p) - offset, (uint)len);
108 bcopy(buf, PKTDATA(osh, p) + offset, n);
109 buf += n;
110 len -= n;
111 ret += n;
112 offset = 0;
113 }
114
115 return ret;
116 }
117
118
119
120 /* return total length of buffer chain */
121 uint BCMFASTPATH
pkttotlen(osl_t * osh,void * p)122 pkttotlen(osl_t *osh, void *p)
123 {
124 uint total;
125 int len;
126
127 total = 0;
128 for (; p; p = PKTNEXT(osh, p)) {
129 len = PKTLEN(osh, p);
130 total += len;
131 }
132
133 return (total);
134 }
135
136 /* return the last buffer of chained pkt */
137 void *
pktlast(osl_t * osh,void * p)138 pktlast(osl_t *osh, void *p)
139 {
140 for (; PKTNEXT(osh, p); p = PKTNEXT(osh, p))
141 ;
142
143 return (p);
144 }
145
146 /* count segments of a chained packet */
147 uint BCMFASTPATH
pktsegcnt(osl_t * osh,void * p)148 pktsegcnt(osl_t *osh, void *p)
149 {
150 uint cnt;
151
152 for (cnt = 0; p; p = PKTNEXT(osh, p))
153 cnt++;
154
155 return cnt;
156 }
157
158
159 /* count segments of a chained packet */
160 uint BCMFASTPATH
pktsegcnt_war(osl_t * osh,void * p)161 pktsegcnt_war(osl_t *osh, void *p)
162 {
163 uint cnt;
164 uint8 *pktdata;
165 uint len, remain, align64;
166
167 for (cnt = 0; p; p = PKTNEXT(osh, p)) {
168 cnt++;
169 len = PKTLEN(osh, p);
170 if (len > 128) {
171 pktdata = (uint8 *)PKTDATA(osh, p); /* starting address of data */
172 /* Check for page boundary straddle (2048B) */
173 if (((uintptr)pktdata & ~0x7ff) != ((uintptr)(pktdata+len) & ~0x7ff))
174 cnt++;
175
176 align64 = (uint)((uintptr)pktdata & 0x3f); /* aligned to 64B */
177 align64 = (64 - align64) & 0x3f;
178 len -= align64; /* bytes from aligned 64B to end */
179 /* if aligned to 128B, check for MOD 128 between 1 to 4B */
180 remain = len % 128;
181 if (remain > 0 && remain <= 4)
182 cnt++; /* add extra seg */
183 }
184 }
185
186 return cnt;
187 }
188
189 uint8 * BCMFASTPATH
pktdataoffset(osl_t * osh,void * p,uint offset)190 pktdataoffset(osl_t *osh, void *p, uint offset)
191 {
192 uint total = pkttotlen(osh, p);
193 uint pkt_off = 0, len = 0;
194 uint8 *pdata = (uint8 *) PKTDATA(osh, p);
195
196 if (offset > total)
197 return NULL;
198
199 for (; p; p = PKTNEXT(osh, p)) {
200 pdata = (uint8 *) PKTDATA(osh, p);
201 pkt_off = offset - len;
202 len += PKTLEN(osh, p);
203 if (len > offset)
204 break;
205 }
206 return (uint8*) (pdata+pkt_off);
207 }
208
209
210 /* given a offset in pdata, find the pkt seg hdr */
211 void *
pktoffset(osl_t * osh,void * p,uint offset)212 pktoffset(osl_t *osh, void *p, uint offset)
213 {
214 uint total = pkttotlen(osh, p);
215 uint len = 0;
216
217 if (offset > total)
218 return NULL;
219
220 for (; p; p = PKTNEXT(osh, p)) {
221 len += PKTLEN(osh, p);
222 if (len > offset)
223 break;
224 }
225 return p;
226 }
227
228 /*
229 * osl multiple-precedence packet queue
230 * hi_prec is always >= the number of the highest non-empty precedence
231 */
232 void * BCMFASTPATH
pktq_penq(struct pktq * pq,int prec,void * p)233 pktq_penq(struct pktq *pq, int prec, void *p)
234 {
235 struct pktq_prec *q;
236
237 ASSERT(prec >= 0 && prec < pq->num_prec);
238 ASSERT(PKTLINK(p) == NULL); /* queueing chains not allowed */
239
240 ASSERT(!pktq_full(pq));
241 ASSERT(!pktq_pfull(pq, prec));
242
243 q = &pq->q[prec];
244
245 if (q->head)
246 PKTSETLINK(q->tail, p);
247 else
248 q->head = p;
249
250 q->tail = p;
251 q->len++;
252
253 pq->len++;
254
255 if (pq->hi_prec < prec)
256 pq->hi_prec = (uint8)prec;
257
258 return p;
259 }
260
261 void * BCMFASTPATH
pktq_penq_head(struct pktq * pq,int prec,void * p)262 pktq_penq_head(struct pktq *pq, int prec, void *p)
263 {
264 struct pktq_prec *q;
265
266 ASSERT(prec >= 0 && prec < pq->num_prec);
267 ASSERT(PKTLINK(p) == NULL); /* queueing chains not allowed */
268
269 ASSERT(!pktq_full(pq));
270 ASSERT(!pktq_pfull(pq, prec));
271
272 q = &pq->q[prec];
273
274 if (q->head == NULL)
275 q->tail = p;
276
277 PKTSETLINK(p, q->head);
278 q->head = p;
279 q->len++;
280
281 pq->len++;
282
283 if (pq->hi_prec < prec)
284 pq->hi_prec = (uint8)prec;
285
286 return p;
287 }
288
289 void * BCMFASTPATH
pktq_pdeq(struct pktq * pq,int prec)290 pktq_pdeq(struct pktq *pq, int prec)
291 {
292 struct pktq_prec *q;
293 void *p;
294
295 ASSERT(prec >= 0 && prec < pq->num_prec);
296
297 q = &pq->q[prec];
298
299 if ((p = q->head) == NULL)
300 return NULL;
301
302 if ((q->head = PKTLINK(p)) == NULL)
303 q->tail = NULL;
304
305 q->len--;
306
307 pq->len--;
308
309 PKTSETLINK(p, NULL);
310
311 return p;
312 }
313
314 void * BCMFASTPATH
pktq_pdeq_prev(struct pktq * pq,int prec,void * prev_p)315 pktq_pdeq_prev(struct pktq *pq, int prec, void *prev_p)
316 {
317 struct pktq_prec *q;
318 void *p;
319
320 ASSERT(prec >= 0 && prec < pq->num_prec);
321
322 q = &pq->q[prec];
323
324 if (prev_p == NULL)
325 return NULL;
326
327 if ((p = PKTLINK(prev_p)) == NULL)
328 return NULL;
329
330 q->len--;
331
332 pq->len--;
333
334 PKTSETLINK(prev_p, PKTLINK(p));
335 PKTSETLINK(p, NULL);
336
337 return p;
338 }
339
340 void * BCMFASTPATH
pktq_pdeq_with_fn(struct pktq * pq,int prec,ifpkt_cb_t fn,int arg)341 pktq_pdeq_with_fn(struct pktq *pq, int prec, ifpkt_cb_t fn, int arg)
342 {
343 struct pktq_prec *q;
344 void *p, *prev = NULL;
345
346 ASSERT(prec >= 0 && prec < pq->num_prec);
347
348 q = &pq->q[prec];
349 p = q->head;
350
351 while (p) {
352 if (fn == NULL || (*fn)(p, arg)) {
353 break;
354 } else {
355 prev = p;
356 p = PKTLINK(p);
357 }
358 }
359 if (p == NULL)
360 return NULL;
361
362 if (prev == NULL) {
363 if ((q->head = PKTLINK(p)) == NULL)
364 q->tail = NULL;
365 } else {
366 PKTSETLINK(prev, PKTLINK(p));
367 }
368
369 q->len--;
370
371 pq->len--;
372
373 PKTSETLINK(p, NULL);
374
375 return p;
376 }
377
378 void * BCMFASTPATH
pktq_pdeq_tail(struct pktq * pq,int prec)379 pktq_pdeq_tail(struct pktq *pq, int prec)
380 {
381 struct pktq_prec *q;
382 void *p, *prev;
383
384 ASSERT(prec >= 0 && prec < pq->num_prec);
385
386 q = &pq->q[prec];
387
388 if ((p = q->head) == NULL)
389 return NULL;
390
391 for (prev = NULL; p != q->tail; p = PKTLINK(p))
392 prev = p;
393
394 if (prev)
395 PKTSETLINK(prev, NULL);
396 else
397 q->head = NULL;
398
399 q->tail = prev;
400 q->len--;
401
402 pq->len--;
403
404 return p;
405 }
406
407 void
pktq_pflush(osl_t * osh,struct pktq * pq,int prec,bool dir,ifpkt_cb_t fn,int arg)408 pktq_pflush(osl_t *osh, struct pktq *pq, int prec, bool dir, ifpkt_cb_t fn, int arg)
409 {
410 struct pktq_prec *q;
411 void *p, *prev = NULL;
412
413 q = &pq->q[prec];
414 p = q->head;
415 while (p) {
416 if (fn == NULL || (*fn)(p, arg)) {
417 bool head = (p == q->head);
418 if (head)
419 q->head = PKTLINK(p);
420 else
421 PKTSETLINK(prev, PKTLINK(p));
422 PKTSETLINK(p, NULL);
423 PKTFREE(osh, p, dir);
424 q->len--;
425 pq->len--;
426 p = (head ? q->head : PKTLINK(prev));
427 } else {
428 prev = p;
429 p = PKTLINK(p);
430 }
431 }
432
433 if (q->head == NULL) {
434 ASSERT(q->len == 0);
435 q->tail = NULL;
436 }
437 }
438
439 bool BCMFASTPATH
pktq_pdel(struct pktq * pq,void * pktbuf,int prec)440 pktq_pdel(struct pktq *pq, void *pktbuf, int prec)
441 {
442 struct pktq_prec *q;
443 void *p;
444
445 ASSERT(prec >= 0 && prec < pq->num_prec);
446
447 if (!pktbuf)
448 return FALSE;
449
450 q = &pq->q[prec];
451
452 if (q->head == pktbuf) {
453 if ((q->head = PKTLINK(pktbuf)) == NULL)
454 q->tail = NULL;
455 } else {
456 for (p = q->head; p && PKTLINK(p) != pktbuf; p = PKTLINK(p))
457 ;
458 if (p == NULL)
459 return FALSE;
460
461 PKTSETLINK(p, PKTLINK(pktbuf));
462 if (q->tail == pktbuf)
463 q->tail = p;
464 }
465
466 q->len--;
467 pq->len--;
468 PKTSETLINK(pktbuf, NULL);
469 return TRUE;
470 }
471
472 void
pktq_init(struct pktq * pq,int num_prec,int max_len)473 pktq_init(struct pktq *pq, int num_prec, int max_len)
474 {
475 int prec;
476
477 ASSERT(num_prec > 0 && num_prec <= PKTQ_MAX_PREC);
478
479 /* pq is variable size; only zero out what's requested */
480 bzero(pq, OFFSETOF(struct pktq, q) + (sizeof(struct pktq_prec) * num_prec));
481
482 pq->num_prec = (uint16)num_prec;
483
484 pq->max = (uint16)max_len;
485
486 for (prec = 0; prec < num_prec; prec++)
487 pq->q[prec].max = pq->max;
488 }
489
490 void
pktq_set_max_plen(struct pktq * pq,int prec,int max_len)491 pktq_set_max_plen(struct pktq *pq, int prec, int max_len)
492 {
493 ASSERT(prec >= 0 && prec < pq->num_prec);
494
495 if (prec < pq->num_prec)
496 pq->q[prec].max = (uint16)max_len;
497 }
498
499 void * BCMFASTPATH
pktq_deq(struct pktq * pq,int * prec_out)500 pktq_deq(struct pktq *pq, int *prec_out)
501 {
502 struct pktq_prec *q;
503 void *p;
504 int prec;
505
506 if (pq->len == 0)
507 return NULL;
508
509 while ((prec = pq->hi_prec) > 0 && pq->q[prec].head == NULL)
510 pq->hi_prec--;
511
512 q = &pq->q[prec];
513
514 if ((p = q->head) == NULL)
515 return NULL;
516
517 if ((q->head = PKTLINK(p)) == NULL)
518 q->tail = NULL;
519
520 q->len--;
521
522 pq->len--;
523
524 if (prec_out)
525 *prec_out = prec;
526
527 PKTSETLINK(p, NULL);
528
529 return p;
530 }
531
532 void * BCMFASTPATH
pktq_deq_tail(struct pktq * pq,int * prec_out)533 pktq_deq_tail(struct pktq *pq, int *prec_out)
534 {
535 struct pktq_prec *q;
536 void *p, *prev;
537 int prec;
538
539 if (pq->len == 0)
540 return NULL;
541
542 for (prec = 0; prec < pq->hi_prec; prec++)
543 if (pq->q[prec].head)
544 break;
545
546 q = &pq->q[prec];
547
548 if ((p = q->head) == NULL)
549 return NULL;
550
551 for (prev = NULL; p != q->tail; p = PKTLINK(p))
552 prev = p;
553
554 if (prev)
555 PKTSETLINK(prev, NULL);
556 else
557 q->head = NULL;
558
559 q->tail = prev;
560 q->len--;
561
562 pq->len--;
563
564 if (prec_out)
565 *prec_out = prec;
566
567 PKTSETLINK(p, NULL);
568
569 return p;
570 }
571
572 void *
pktq_peek(struct pktq * pq,int * prec_out)573 pktq_peek(struct pktq *pq, int *prec_out)
574 {
575 int prec;
576
577 if (pq->len == 0)
578 return NULL;
579
580 while ((prec = pq->hi_prec) > 0 && pq->q[prec].head == NULL)
581 pq->hi_prec--;
582
583 if (prec_out)
584 *prec_out = prec;
585
586 return (pq->q[prec].head);
587 }
588
589 void *
pktq_peek_tail(struct pktq * pq,int * prec_out)590 pktq_peek_tail(struct pktq *pq, int *prec_out)
591 {
592 int prec;
593
594 if (pq->len == 0)
595 return NULL;
596
597 for (prec = 0; prec < pq->hi_prec; prec++)
598 if (pq->q[prec].head)
599 break;
600
601 if (prec_out)
602 *prec_out = prec;
603
604 return (pq->q[prec].tail);
605 }
606
607 void
pktq_flush(osl_t * osh,struct pktq * pq,bool dir,ifpkt_cb_t fn,int arg)608 pktq_flush(osl_t *osh, struct pktq *pq, bool dir, ifpkt_cb_t fn, int arg)
609 {
610 int prec;
611
612 /* Optimize flush, if pktq len = 0, just return.
613 * pktq len of 0 means pktq's prec q's are all empty.
614 */
615 if (pq->len == 0) {
616 return;
617 }
618
619 for (prec = 0; prec < pq->num_prec; prec++)
620 pktq_pflush(osh, pq, prec, dir, fn, arg);
621 if (fn == NULL)
622 ASSERT(pq->len == 0);
623 }
624
625 /* Return sum of lengths of a specific set of precedences */
626 int
pktq_mlen(struct pktq * pq,uint prec_bmp)627 pktq_mlen(struct pktq *pq, uint prec_bmp)
628 {
629 int prec, len;
630
631 len = 0;
632
633 for (prec = 0; prec <= pq->hi_prec; prec++)
634 if (prec_bmp & (1 << prec))
635 len += pq->q[prec].len;
636
637 return len;
638 }
639
640 /* Priority peek from a specific set of precedences */
641 void * BCMFASTPATH
pktq_mpeek(struct pktq * pq,uint prec_bmp,int * prec_out)642 pktq_mpeek(struct pktq *pq, uint prec_bmp, int *prec_out)
643 {
644 struct pktq_prec *q;
645 void *p;
646 int prec;
647
648 if (pq->len == 0)
649 {
650 return NULL;
651 }
652 while ((prec = pq->hi_prec) > 0 && pq->q[prec].head == NULL)
653 pq->hi_prec--;
654
655 while ((prec_bmp & (1 << prec)) == 0 || pq->q[prec].head == NULL)
656 if (prec-- == 0)
657 return NULL;
658
659 q = &pq->q[prec];
660
661 if ((p = q->head) == NULL)
662 return NULL;
663
664 if (prec_out)
665 *prec_out = prec;
666
667 return p;
668 }
669 /* Priority dequeue from a specific set of precedences */
670 void * BCMFASTPATH
pktq_mdeq(struct pktq * pq,uint prec_bmp,int * prec_out)671 pktq_mdeq(struct pktq *pq, uint prec_bmp, int *prec_out)
672 {
673 struct pktq_prec *q;
674 void *p;
675 int prec;
676
677 if (pq->len == 0)
678 return NULL;
679
680 while ((prec = pq->hi_prec) > 0 && pq->q[prec].head == NULL)
681 pq->hi_prec--;
682
683 while ((pq->q[prec].head == NULL) || ((prec_bmp & (1 << prec)) == 0))
684 if (prec-- == 0)
685 return NULL;
686
687 q = &pq->q[prec];
688
689 if ((p = q->head) == NULL)
690 return NULL;
691
692 if ((q->head = PKTLINK(p)) == NULL)
693 q->tail = NULL;
694
695 q->len--;
696
697 if (prec_out)
698 *prec_out = prec;
699
700 pq->len--;
701
702 PKTSETLINK(p, NULL);
703
704 return p;
705 }
706
707 #endif /* BCMDRIVER */
708
709 #if !defined(BCMROMOFFLOAD_EXCLUDE_BCMUTILS_FUNCS)
710 const unsigned char bcm_ctype[] = {
711
712 _BCM_C,_BCM_C,_BCM_C,_BCM_C,_BCM_C,_BCM_C,_BCM_C,_BCM_C, /* 0-7 */
713 _BCM_C, _BCM_C|_BCM_S, _BCM_C|_BCM_S, _BCM_C|_BCM_S, _BCM_C|_BCM_S, _BCM_C|_BCM_S, _BCM_C,
714 _BCM_C, /* 8-15 */
715 _BCM_C,_BCM_C,_BCM_C,_BCM_C,_BCM_C,_BCM_C,_BCM_C,_BCM_C, /* 16-23 */
716 _BCM_C,_BCM_C,_BCM_C,_BCM_C,_BCM_C,_BCM_C,_BCM_C,_BCM_C, /* 24-31 */
717 _BCM_S|_BCM_SP,_BCM_P,_BCM_P,_BCM_P,_BCM_P,_BCM_P,_BCM_P,_BCM_P, /* 32-39 */
718 _BCM_P,_BCM_P,_BCM_P,_BCM_P,_BCM_P,_BCM_P,_BCM_P,_BCM_P, /* 40-47 */
719 _BCM_D,_BCM_D,_BCM_D,_BCM_D,_BCM_D,_BCM_D,_BCM_D,_BCM_D, /* 48-55 */
720 _BCM_D,_BCM_D,_BCM_P,_BCM_P,_BCM_P,_BCM_P,_BCM_P,_BCM_P, /* 56-63 */
721 _BCM_P, _BCM_U|_BCM_X, _BCM_U|_BCM_X, _BCM_U|_BCM_X, _BCM_U|_BCM_X, _BCM_U|_BCM_X,
722 _BCM_U|_BCM_X, _BCM_U, /* 64-71 */
723 _BCM_U,_BCM_U,_BCM_U,_BCM_U,_BCM_U,_BCM_U,_BCM_U,_BCM_U, /* 72-79 */
724 _BCM_U,_BCM_U,_BCM_U,_BCM_U,_BCM_U,_BCM_U,_BCM_U,_BCM_U, /* 80-87 */
725 _BCM_U,_BCM_U,_BCM_U,_BCM_P,_BCM_P,_BCM_P,_BCM_P,_BCM_P, /* 88-95 */
726 _BCM_P, _BCM_L|_BCM_X, _BCM_L|_BCM_X, _BCM_L|_BCM_X, _BCM_L|_BCM_X, _BCM_L|_BCM_X,
727 _BCM_L|_BCM_X, _BCM_L, /* 96-103 */
728 _BCM_L,_BCM_L,_BCM_L,_BCM_L,_BCM_L,_BCM_L,_BCM_L,_BCM_L, /* 104-111 */
729 _BCM_L,_BCM_L,_BCM_L,_BCM_L,_BCM_L,_BCM_L,_BCM_L,_BCM_L, /* 112-119 */
730 _BCM_L,_BCM_L,_BCM_L,_BCM_P,_BCM_P,_BCM_P,_BCM_P,_BCM_C, /* 120-127 */
731 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, /* 128-143 */
732 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, /* 144-159 */
733 _BCM_S|_BCM_SP, _BCM_P, _BCM_P, _BCM_P, _BCM_P, _BCM_P, _BCM_P, _BCM_P, _BCM_P, _BCM_P,
734 _BCM_P, _BCM_P, _BCM_P, _BCM_P, _BCM_P, _BCM_P, /* 160-175 */
735 _BCM_P, _BCM_P, _BCM_P, _BCM_P, _BCM_P, _BCM_P, _BCM_P, _BCM_P, _BCM_P, _BCM_P, _BCM_P,
736 _BCM_P, _BCM_P, _BCM_P, _BCM_P, _BCM_P, /* 176-191 */
737 _BCM_U, _BCM_U, _BCM_U, _BCM_U, _BCM_U, _BCM_U, _BCM_U, _BCM_U, _BCM_U, _BCM_U, _BCM_U,
738 _BCM_U, _BCM_U, _BCM_U, _BCM_U, _BCM_U, /* 192-207 */
739 _BCM_U, _BCM_U, _BCM_U, _BCM_U, _BCM_U, _BCM_U, _BCM_U, _BCM_P, _BCM_U, _BCM_U, _BCM_U,
740 _BCM_U, _BCM_U, _BCM_U, _BCM_U, _BCM_L, /* 208-223 */
741 _BCM_L, _BCM_L, _BCM_L, _BCM_L, _BCM_L, _BCM_L, _BCM_L, _BCM_L, _BCM_L, _BCM_L, _BCM_L,
742 _BCM_L, _BCM_L, _BCM_L, _BCM_L, _BCM_L, /* 224-239 */
743 _BCM_L, _BCM_L, _BCM_L, _BCM_L, _BCM_L, _BCM_L, _BCM_L, _BCM_P, _BCM_L, _BCM_L, _BCM_L,
744 _BCM_L, _BCM_L, _BCM_L, _BCM_L, _BCM_L /* 240-255 */
745 };
746
747 ulong
bcm_strtoul(const char * cp,char ** endp,uint base)748 bcm_strtoul(const char *cp, char **endp, uint base)
749 {
750 ulong result, last_result = 0, value;
751 bool minus;
752
753 minus = FALSE;
754
755 while (bcm_isspace(*cp))
756 cp++;
757
758 if (cp[0] == '+')
759 cp++;
760 else if (cp[0] == '-') {
761 minus = TRUE;
762 cp++;
763 }
764
765 if (base == 0) {
766 if (cp[0] == '0') {
767 if ((cp[1] == 'x') || (cp[1] == 'X')) {
768 base = 16;
769 cp = &cp[2];
770 } else {
771 base = 8;
772 cp = &cp[1];
773 }
774 } else
775 base = 10;
776 } else if (base == 16 && (cp[0] == '0') && ((cp[1] == 'x') || (cp[1] == 'X'))) {
777 cp = &cp[2];
778 }
779
780 result = 0;
781
782 while (bcm_isxdigit(*cp) &&
783 (value = bcm_isdigit(*cp) ? *cp-'0' : bcm_toupper(*cp)-'A'+10) < base) {
784 result = result*base + value;
785 /* Detected overflow */
786 if (result < last_result && !minus)
787 return (ulong)-1;
788 last_result = result;
789 cp++;
790 }
791
792 if (minus)
793 result = (ulong)(-(long)result);
794
795 if (endp)
796 *endp = DISCARD_QUAL(cp, char);
797
798 return (result);
799 }
800
801 int
bcm_atoi(const char * s)802 bcm_atoi(const char *s)
803 {
804 return (int)bcm_strtoul(s, NULL, 10);
805 }
806
807 /* return pointer to location of substring 'needle' in 'haystack' */
808 char *
bcmstrstr(const char * haystack,const char * needle)809 bcmstrstr(const char *haystack, const char *needle)
810 {
811 int len, nlen;
812 int i;
813
814 if ((haystack == NULL) || (needle == NULL))
815 return DISCARD_QUAL(haystack, char);
816
817 nlen = strlen(needle);
818 len = strlen(haystack) - nlen + 1;
819
820 for (i = 0; i < len; i++)
821 if (memcmp(needle, &haystack[i], nlen) == 0)
822 return DISCARD_QUAL(&haystack[i], char);
823 return (NULL);
824 }
825
826 char *
bcmstrcat(char * dest,const char * src)827 bcmstrcat(char *dest, const char *src)
828 {
829 char *p;
830
831 p = dest + strlen(dest);
832
833 while ((*p++ = *src++) != '\0')
834 ;
835
836 return (dest);
837 }
838
839 char *
bcmstrncat(char * dest,const char * src,uint size)840 bcmstrncat(char *dest, const char *src, uint size)
841 {
842 char *endp;
843 char *p;
844
845 p = dest + strlen(dest);
846 endp = p + size;
847
848 while (p != endp && (*p++ = *src++) != '\0')
849 ;
850
851 return (dest);
852 }
853
854
855 /****************************************************************************
856 * Function: bcmstrtok
857 *
858 * Purpose:
859 * Tokenizes a string. This function is conceptually similiar to ANSI C strtok(),
860 * but allows strToken() to be used by different strings or callers at the same
861 * time. Each call modifies '*string' by substituting a NULL character for the
862 * first delimiter that is encountered, and updates 'string' to point to the char
863 * after the delimiter. Leading delimiters are skipped.
864 *
865 * Parameters:
866 * string (mod) Ptr to string ptr, updated by token.
867 * delimiters (in) Set of delimiter characters.
868 * tokdelim (out) Character that delimits the returned token. (May
869 * be set to NULL if token delimiter is not required).
870 *
871 * Returns: Pointer to the next token found. NULL when no more tokens are found.
872 *****************************************************************************
873 */
874 char *
bcmstrtok(char ** string,const char * delimiters,char * tokdelim)875 bcmstrtok(char **string, const char *delimiters, char *tokdelim)
876 {
877 unsigned char *str;
878 unsigned long map[8];
879 int count;
880 char *nextoken;
881
882 if (tokdelim != NULL) {
883 /* Prime the token delimiter */
884 *tokdelim = '\0';
885 }
886
887 /* Clear control map */
888 for (count = 0; count < 8; count++) {
889 map[count] = 0;
890 }
891
892 /* Set bits in delimiter table */
893 do {
894 map[*delimiters >> 5] |= (1 << (*delimiters & 31));
895 }
896 while (*delimiters++);
897
898 str = (unsigned char*)*string;
899
900 /* Find beginning of token (skip over leading delimiters). Note that
901 * there is no token iff this loop sets str to point to the terminal
902 * null (*str == '\0')
903 */
904 while (((map[*str >> 5] & (1 << (*str & 31))) && *str) || (*str == ' ')) {
905 str++;
906 }
907
908 nextoken = (char*)str;
909
910 /* Find the end of the token. If it is not the end of the string,
911 * put a null there.
912 */
913 for (; *str; str++) {
914 if (map[*str >> 5] & (1 << (*str & 31))) {
915 if (tokdelim != NULL) {
916 *tokdelim = *str;
917 }
918
919 *str++ = '\0';
920 break;
921 }
922 }
923
924 *string = (char*)str;
925
926 /* Determine if a token has been found. */
927 if (nextoken == (char *) str) {
928 return NULL;
929 }
930 else {
931 return nextoken;
932 }
933 }
934
935
936 #define xToLower(C) \
937 ((C >= 'A' && C <= 'Z') ? (char)((int)C - (int)'A' + (int)'a') : C)
938
939
940 /****************************************************************************
941 * Function: bcmstricmp
942 *
943 * Purpose: Compare to strings case insensitively.
944 *
945 * Parameters: s1 (in) First string to compare.
946 * s2 (in) Second string to compare.
947 *
948 * Returns: Return 0 if the two strings are equal, -1 if t1 < t2 and 1 if
949 * t1 > t2, when ignoring case sensitivity.
950 *****************************************************************************
951 */
952 int
bcmstricmp(const char * s1,const char * s2)953 bcmstricmp(const char *s1, const char *s2)
954 {
955 char dc, sc;
956
957 while (*s2 && *s1) {
958 dc = xToLower(*s1);
959 sc = xToLower(*s2);
960 if (dc < sc) return -1;
961 if (dc > sc) return 1;
962 s1++;
963 s2++;
964 }
965
966 if (*s1 && !*s2) return 1;
967 if (!*s1 && *s2) return -1;
968 return 0;
969 }
970
971
972 /****************************************************************************
973 * Function: bcmstrnicmp
974 *
975 * Purpose: Compare to strings case insensitively, upto a max of 'cnt'
976 * characters.
977 *
978 * Parameters: s1 (in) First string to compare.
979 * s2 (in) Second string to compare.
980 * cnt (in) Max characters to compare.
981 *
982 * Returns: Return 0 if the two strings are equal, -1 if t1 < t2 and 1 if
983 * t1 > t2, when ignoring case sensitivity.
984 *****************************************************************************
985 */
986 int
bcmstrnicmp(const char * s1,const char * s2,int cnt)987 bcmstrnicmp(const char* s1, const char* s2, int cnt)
988 {
989 char dc, sc;
990
991 while (*s2 && *s1 && cnt) {
992 dc = xToLower(*s1);
993 sc = xToLower(*s2);
994 if (dc < sc) return -1;
995 if (dc > sc) return 1;
996 s1++;
997 s2++;
998 cnt--;
999 }
1000
1001 if (!cnt) return 0;
1002 if (*s1 && !*s2) return 1;
1003 if (!*s1 && *s2) return -1;
1004 return 0;
1005 }
1006
1007 /* parse a xx:xx:xx:xx:xx:xx format ethernet address */
1008 int
bcm_ether_atoe(const char * p,struct ether_addr * ea)1009 bcm_ether_atoe(const char *p, struct ether_addr *ea)
1010 {
1011 int i = 0;
1012 char *ep;
1013
1014 for (;;) {
1015 ea->octet[i++] = (char) bcm_strtoul(p, &ep, 16);
1016 p = ep;
1017 if (!*p++ || i == 6)
1018 break;
1019 }
1020
1021 return (i == 6);
1022 }
1023 #endif /* !BCMROMOFFLOAD_EXCLUDE_BCMUTILS_FUNCS */
1024
1025
1026 #if defined(CONFIG_USBRNDIS_RETAIL) || defined(NDIS_MINIPORT_DRIVER)
1027 /* registry routine buffer preparation utility functions:
1028 * parameter order is like strncpy, but returns count
1029 * of bytes copied. Minimum bytes copied is null char(1)/wchar(2)
1030 */
1031 ulong
wchar2ascii(char * abuf,ushort * wbuf,ushort wbuflen,ulong abuflen)1032 wchar2ascii(char *abuf, ushort *wbuf, ushort wbuflen, ulong abuflen)
1033 {
1034 ulong copyct = 1;
1035 ushort i;
1036
1037 if (abuflen == 0)
1038 return 0;
1039
1040 /* wbuflen is in bytes */
1041 wbuflen /= sizeof(ushort);
1042
1043 for (i = 0; i < wbuflen; ++i) {
1044 if (--abuflen == 0)
1045 break;
1046 *abuf++ = (char) *wbuf++;
1047 ++copyct;
1048 }
1049 *abuf = '\0';
1050
1051 return copyct;
1052 }
1053 #endif /* CONFIG_USBRNDIS_RETAIL || NDIS_MINIPORT_DRIVER */
1054
1055 char *
bcm_ether_ntoa(const struct ether_addr * ea,char * buf)1056 bcm_ether_ntoa(const struct ether_addr *ea, char *buf)
1057 {
1058 static const char hex[] =
1059 {
1060 '0', '1', '2', '3', '4', '5', '6', '7',
1061 '8', '9', 'a', 'b', 'c', 'd', 'e', 'f'
1062 };
1063 const uint8 *octet = ea->octet;
1064 char *p = buf;
1065 int i;
1066
1067 for (i = 0; i < 6; i++, octet++) {
1068 *p++ = hex[(*octet >> 4) & 0xf];
1069 *p++ = hex[*octet & 0xf];
1070 *p++ = ':';
1071 }
1072
1073 *(p-1) = '\0';
1074
1075 return (buf);
1076 }
1077
1078 char *
bcm_ip_ntoa(struct ipv4_addr * ia,char * buf)1079 bcm_ip_ntoa(struct ipv4_addr *ia, char *buf)
1080 {
1081 snprintf(buf, 16, "%d.%d.%d.%d",
1082 ia->addr[0], ia->addr[1], ia->addr[2], ia->addr[3]);
1083 return (buf);
1084 }
1085
1086 #ifdef BCMDRIVER
1087
1088 void
bcm_mdelay(uint ms)1089 bcm_mdelay(uint ms)
1090 {
1091 uint i;
1092
1093 for (i = 0; i < ms; i++) {
1094 OSL_DELAY(1000);
1095 }
1096 }
1097
1098
1099
1100
1101
1102 #if defined(DHD_DEBUG)
1103 /* pretty hex print a pkt buffer chain */
1104 void
prpkt(const char * msg,osl_t * osh,void * p0)1105 prpkt(const char *msg, osl_t *osh, void *p0)
1106 {
1107 void *p;
1108
1109 if (msg && (msg[0] != '\0'))
1110 printf("%s:\n", msg);
1111
1112 for (p = p0; p; p = PKTNEXT(osh, p))
1113 prhex(NULL, PKTDATA(osh, p), PKTLEN(osh, p));
1114 }
1115 #endif
1116
1117 /* Takes an Ethernet frame and sets out-of-bound PKTPRIO.
1118 * Also updates the inplace vlan tag if requested.
1119 * For debugging, it returns an indication of what it did.
1120 */
1121 uint BCMFASTPATH
pktsetprio(void * pkt,bool update_vtag)1122 pktsetprio(void *pkt, bool update_vtag)
1123 {
1124 struct ether_header *eh;
1125 struct ethervlan_header *evh;
1126 uint8 *pktdata;
1127 int priority = 0;
1128 int rc = 0;
1129
1130 pktdata = (uint8 *)PKTDATA(NULL, pkt);
1131 ASSERT(ISALIGNED((uintptr)pktdata, sizeof(uint16)));
1132
1133 eh = (struct ether_header *) pktdata;
1134
1135 if (eh->ether_type == hton16(ETHER_TYPE_8021Q)) {
1136 uint16 vlan_tag;
1137 int vlan_prio, dscp_prio = 0;
1138
1139 evh = (struct ethervlan_header *)eh;
1140
1141 vlan_tag = ntoh16(evh->vlan_tag);
1142 vlan_prio = (int) (vlan_tag >> VLAN_PRI_SHIFT) & VLAN_PRI_MASK;
1143
1144 if (evh->ether_type == hton16(ETHER_TYPE_IP)) {
1145 uint8 *ip_body = pktdata + sizeof(struct ethervlan_header);
1146 uint8 tos_tc = IP_TOS46(ip_body);
1147 dscp_prio = (int)(tos_tc >> IPV4_TOS_PREC_SHIFT);
1148 }
1149
1150 /* DSCP priority gets precedence over 802.1P (vlan tag) */
1151 if (dscp_prio != 0) {
1152 priority = dscp_prio;
1153 rc |= PKTPRIO_VDSCP;
1154 } else {
1155 priority = vlan_prio;
1156 rc |= PKTPRIO_VLAN;
1157 }
1158 /*
1159 * If the DSCP priority is not the same as the VLAN priority,
1160 * then overwrite the priority field in the vlan tag, with the
1161 * DSCP priority value. This is required for Linux APs because
1162 * the VLAN driver on Linux, overwrites the skb->priority field
1163 * with the priority value in the vlan tag
1164 */
1165 if (update_vtag && (priority != vlan_prio)) {
1166 vlan_tag &= ~(VLAN_PRI_MASK << VLAN_PRI_SHIFT);
1167 vlan_tag |= (uint16)priority << VLAN_PRI_SHIFT;
1168 evh->vlan_tag = hton16(vlan_tag);
1169 rc |= PKTPRIO_UPD;
1170 }
1171 } else if (eh->ether_type == hton16(ETHER_TYPE_IP)) {
1172 uint8 *ip_body = pktdata + sizeof(struct ether_header);
1173 uint8 tos_tc = IP_TOS46(ip_body);
1174 priority = (int)(tos_tc >> IPV4_TOS_PREC_SHIFT);
1175 rc |= PKTPRIO_DSCP;
1176 }
1177
1178 ASSERT(priority >= 0 && priority <= MAXPRIO);
1179 PKTSETPRIO(pkt, priority);
1180 return (rc | priority);
1181 }
1182
1183
1184 static char bcm_undeferrstr[32];
1185 static const char *bcmerrorstrtable[] = BCMERRSTRINGTABLE;
1186
1187 /* Convert the error codes into related error strings */
1188 const char *
bcmerrorstr(int bcmerror)1189 bcmerrorstr(int bcmerror)
1190 {
1191 /* check if someone added a bcmerror code but forgot to add errorstring */
1192 ASSERT(ABS(BCME_LAST) == (ARRAYSIZE(bcmerrorstrtable) - 1));
1193
1194 if (bcmerror > 0 || bcmerror < BCME_LAST) {
1195 snprintf(bcm_undeferrstr, sizeof(bcm_undeferrstr), "Undefined error %d", bcmerror);
1196 return bcm_undeferrstr;
1197 }
1198
1199 ASSERT(strlen(bcmerrorstrtable[-bcmerror]) < BCME_STRLEN);
1200
1201 return bcmerrorstrtable[-bcmerror];
1202 }
1203
1204
1205
1206 /* iovar table lookup */
1207 const bcm_iovar_t*
bcm_iovar_lookup(const bcm_iovar_t * table,const char * name)1208 bcm_iovar_lookup(const bcm_iovar_t *table, const char *name)
1209 {
1210 const bcm_iovar_t *vi;
1211 const char *lookup_name;
1212
1213 /* skip any ':' delimited option prefixes */
1214 lookup_name = strrchr(name, ':');
1215 if (lookup_name != NULL)
1216 lookup_name++;
1217 else
1218 lookup_name = name;
1219
1220 ASSERT(table != NULL);
1221
1222 for (vi = table; vi->name; vi++) {
1223 if (!strcmp(vi->name, lookup_name))
1224 return vi;
1225 }
1226 /* ran to end of table */
1227
1228 return NULL; /* var name not found */
1229 }
1230
1231 int
bcm_iovar_lencheck(const bcm_iovar_t * vi,void * arg,int len,bool set)1232 bcm_iovar_lencheck(const bcm_iovar_t *vi, void *arg, int len, bool set)
1233 {
1234 int bcmerror = 0;
1235
1236 /* length check on io buf */
1237 switch (vi->type) {
1238 case IOVT_BOOL:
1239 case IOVT_INT8:
1240 case IOVT_INT16:
1241 case IOVT_INT32:
1242 case IOVT_UINT8:
1243 case IOVT_UINT16:
1244 case IOVT_UINT32:
1245 /* all integers are int32 sized args at the ioctl interface */
1246 if (len < (int)sizeof(int)) {
1247 bcmerror = BCME_BUFTOOSHORT;
1248 }
1249 break;
1250
1251 case IOVT_BUFFER:
1252 /* buffer must meet minimum length requirement */
1253 if (len < vi->minlen) {
1254 bcmerror = BCME_BUFTOOSHORT;
1255 }
1256 break;
1257
1258 case IOVT_VOID:
1259 if (!set) {
1260 /* Cannot return nil... */
1261 bcmerror = BCME_UNSUPPORTED;
1262 } else if (len) {
1263 /* Set is an action w/o parameters */
1264 bcmerror = BCME_BUFTOOLONG;
1265 }
1266 break;
1267
1268 default:
1269 /* unknown type for length check in iovar info */
1270 ASSERT(0);
1271 bcmerror = BCME_UNSUPPORTED;
1272 }
1273
1274 return bcmerror;
1275 }
1276
1277 #endif /* BCMDRIVER */
1278
1279
1280 #if !defined(BCMROMOFFLOAD_EXCLUDE_BCMUTILS_FUNCS)
1281 /*******************************************************************************
1282 * crc8
1283 *
1284 * Computes a crc8 over the input data using the polynomial:
1285 *
1286 * x^8 + x^7 +x^6 + x^4 + x^2 + 1
1287 *
1288 * The caller provides the initial value (either CRC8_INIT_VALUE
1289 * or the previous returned value) to allow for processing of
1290 * discontiguous blocks of data. When generating the CRC the
1291 * caller is responsible for complementing the final return value
1292 * and inserting it into the byte stream. When checking, a final
1293 * return value of CRC8_GOOD_VALUE indicates a valid CRC.
1294 *
1295 * Reference: Dallas Semiconductor Application Note 27
1296 * Williams, Ross N., "A Painless Guide to CRC Error Detection Algorithms",
1297 * ver 3, Aug 1993, ross@guest.adelaide.edu.au, Rocksoft Pty Ltd.,
1298 * ftp://ftp.rocksoft.com/clients/rocksoft/papers/crc_v3.txt
1299 *
1300 * ****************************************************************************
1301 */
1302
1303 static const uint8 crc8_table[256] = {
1304 0x00, 0xF7, 0xB9, 0x4E, 0x25, 0xD2, 0x9C, 0x6B,
1305 0x4A, 0xBD, 0xF3, 0x04, 0x6F, 0x98, 0xD6, 0x21,
1306 0x94, 0x63, 0x2D, 0xDA, 0xB1, 0x46, 0x08, 0xFF,
1307 0xDE, 0x29, 0x67, 0x90, 0xFB, 0x0C, 0x42, 0xB5,
1308 0x7F, 0x88, 0xC6, 0x31, 0x5A, 0xAD, 0xE3, 0x14,
1309 0x35, 0xC2, 0x8C, 0x7B, 0x10, 0xE7, 0xA9, 0x5E,
1310 0xEB, 0x1C, 0x52, 0xA5, 0xCE, 0x39, 0x77, 0x80,
1311 0xA1, 0x56, 0x18, 0xEF, 0x84, 0x73, 0x3D, 0xCA,
1312 0xFE, 0x09, 0x47, 0xB0, 0xDB, 0x2C, 0x62, 0x95,
1313 0xB4, 0x43, 0x0D, 0xFA, 0x91, 0x66, 0x28, 0xDF,
1314 0x6A, 0x9D, 0xD3, 0x24, 0x4F, 0xB8, 0xF6, 0x01,
1315 0x20, 0xD7, 0x99, 0x6E, 0x05, 0xF2, 0xBC, 0x4B,
1316 0x81, 0x76, 0x38, 0xCF, 0xA4, 0x53, 0x1D, 0xEA,
1317 0xCB, 0x3C, 0x72, 0x85, 0xEE, 0x19, 0x57, 0xA0,
1318 0x15, 0xE2, 0xAC, 0x5B, 0x30, 0xC7, 0x89, 0x7E,
1319 0x5F, 0xA8, 0xE6, 0x11, 0x7A, 0x8D, 0xC3, 0x34,
1320 0xAB, 0x5C, 0x12, 0xE5, 0x8E, 0x79, 0x37, 0xC0,
1321 0xE1, 0x16, 0x58, 0xAF, 0xC4, 0x33, 0x7D, 0x8A,
1322 0x3F, 0xC8, 0x86, 0x71, 0x1A, 0xED, 0xA3, 0x54,
1323 0x75, 0x82, 0xCC, 0x3B, 0x50, 0xA7, 0xE9, 0x1E,
1324 0xD4, 0x23, 0x6D, 0x9A, 0xF1, 0x06, 0x48, 0xBF,
1325 0x9E, 0x69, 0x27, 0xD0, 0xBB, 0x4C, 0x02, 0xF5,
1326 0x40, 0xB7, 0xF9, 0x0E, 0x65, 0x92, 0xDC, 0x2B,
1327 0x0A, 0xFD, 0xB3, 0x44, 0x2F, 0xD8, 0x96, 0x61,
1328 0x55, 0xA2, 0xEC, 0x1B, 0x70, 0x87, 0xC9, 0x3E,
1329 0x1F, 0xE8, 0xA6, 0x51, 0x3A, 0xCD, 0x83, 0x74,
1330 0xC1, 0x36, 0x78, 0x8F, 0xE4, 0x13, 0x5D, 0xAA,
1331 0x8B, 0x7C, 0x32, 0xC5, 0xAE, 0x59, 0x17, 0xE0,
1332 0x2A, 0xDD, 0x93, 0x64, 0x0F, 0xF8, 0xB6, 0x41,
1333 0x60, 0x97, 0xD9, 0x2E, 0x45, 0xB2, 0xFC, 0x0B,
1334 0xBE, 0x49, 0x07, 0xF0, 0x9B, 0x6C, 0x22, 0xD5,
1335 0xF4, 0x03, 0x4D, 0xBA, 0xD1, 0x26, 0x68, 0x9F
1336 };
1337
1338 #define CRC_INNER_LOOP(n, c, x) \
1339 (c) = ((c) >> 8) ^ crc##n##_table[((c) ^ (x)) & 0xff]
1340
1341 uint8
hndcrc8(uint8 * pdata,uint nbytes,uint8 crc)1342 hndcrc8(
1343 uint8 *pdata, /* pointer to array of data to process */
1344 uint nbytes, /* number of input data bytes to process */
1345 uint8 crc /* either CRC8_INIT_VALUE or previous return value */
1346 )
1347 {
1348 /* hard code the crc loop instead of using CRC_INNER_LOOP macro
1349 * to avoid the undefined and unnecessary (uint8 >> 8) operation.
1350 */
1351 while (nbytes-- > 0)
1352 crc = crc8_table[(crc ^ *pdata++) & 0xff];
1353
1354 return crc;
1355 }
1356
1357 /*******************************************************************************
1358 * crc16
1359 *
1360 * Computes a crc16 over the input data using the polynomial:
1361 *
1362 * x^16 + x^12 +x^5 + 1
1363 *
1364 * The caller provides the initial value (either CRC16_INIT_VALUE
1365 * or the previous returned value) to allow for processing of
1366 * discontiguous blocks of data. When generating the CRC the
1367 * caller is responsible for complementing the final return value
1368 * and inserting it into the byte stream. When checking, a final
1369 * return value of CRC16_GOOD_VALUE indicates a valid CRC.
1370 *
1371 * Reference: Dallas Semiconductor Application Note 27
1372 * Williams, Ross N., "A Painless Guide to CRC Error Detection Algorithms",
1373 * ver 3, Aug 1993, ross@guest.adelaide.edu.au, Rocksoft Pty Ltd.,
1374 * ftp://ftp.rocksoft.com/clients/rocksoft/papers/crc_v3.txt
1375 *
1376 * ****************************************************************************
1377 */
1378
1379 static const uint16 crc16_table[256] = {
1380 0x0000, 0x1189, 0x2312, 0x329B, 0x4624, 0x57AD, 0x6536, 0x74BF,
1381 0x8C48, 0x9DC1, 0xAF5A, 0xBED3, 0xCA6C, 0xDBE5, 0xE97E, 0xF8F7,
1382 0x1081, 0x0108, 0x3393, 0x221A, 0x56A5, 0x472C, 0x75B7, 0x643E,
1383 0x9CC9, 0x8D40, 0xBFDB, 0xAE52, 0xDAED, 0xCB64, 0xF9FF, 0xE876,
1384 0x2102, 0x308B, 0x0210, 0x1399, 0x6726, 0x76AF, 0x4434, 0x55BD,
1385 0xAD4A, 0xBCC3, 0x8E58, 0x9FD1, 0xEB6E, 0xFAE7, 0xC87C, 0xD9F5,
1386 0x3183, 0x200A, 0x1291, 0x0318, 0x77A7, 0x662E, 0x54B5, 0x453C,
1387 0xBDCB, 0xAC42, 0x9ED9, 0x8F50, 0xFBEF, 0xEA66, 0xD8FD, 0xC974,
1388 0x4204, 0x538D, 0x6116, 0x709F, 0x0420, 0x15A9, 0x2732, 0x36BB,
1389 0xCE4C, 0xDFC5, 0xED5E, 0xFCD7, 0x8868, 0x99E1, 0xAB7A, 0xBAF3,
1390 0x5285, 0x430C, 0x7197, 0x601E, 0x14A1, 0x0528, 0x37B3, 0x263A,
1391 0xDECD, 0xCF44, 0xFDDF, 0xEC56, 0x98E9, 0x8960, 0xBBFB, 0xAA72,
1392 0x6306, 0x728F, 0x4014, 0x519D, 0x2522, 0x34AB, 0x0630, 0x17B9,
1393 0xEF4E, 0xFEC7, 0xCC5C, 0xDDD5, 0xA96A, 0xB8E3, 0x8A78, 0x9BF1,
1394 0x7387, 0x620E, 0x5095, 0x411C, 0x35A3, 0x242A, 0x16B1, 0x0738,
1395 0xFFCF, 0xEE46, 0xDCDD, 0xCD54, 0xB9EB, 0xA862, 0x9AF9, 0x8B70,
1396 0x8408, 0x9581, 0xA71A, 0xB693, 0xC22C, 0xD3A5, 0xE13E, 0xF0B7,
1397 0x0840, 0x19C9, 0x2B52, 0x3ADB, 0x4E64, 0x5FED, 0x6D76, 0x7CFF,
1398 0x9489, 0x8500, 0xB79B, 0xA612, 0xD2AD, 0xC324, 0xF1BF, 0xE036,
1399 0x18C1, 0x0948, 0x3BD3, 0x2A5A, 0x5EE5, 0x4F6C, 0x7DF7, 0x6C7E,
1400 0xA50A, 0xB483, 0x8618, 0x9791, 0xE32E, 0xF2A7, 0xC03C, 0xD1B5,
1401 0x2942, 0x38CB, 0x0A50, 0x1BD9, 0x6F66, 0x7EEF, 0x4C74, 0x5DFD,
1402 0xB58B, 0xA402, 0x9699, 0x8710, 0xF3AF, 0xE226, 0xD0BD, 0xC134,
1403 0x39C3, 0x284A, 0x1AD1, 0x0B58, 0x7FE7, 0x6E6E, 0x5CF5, 0x4D7C,
1404 0xC60C, 0xD785, 0xE51E, 0xF497, 0x8028, 0x91A1, 0xA33A, 0xB2B3,
1405 0x4A44, 0x5BCD, 0x6956, 0x78DF, 0x0C60, 0x1DE9, 0x2F72, 0x3EFB,
1406 0xD68D, 0xC704, 0xF59F, 0xE416, 0x90A9, 0x8120, 0xB3BB, 0xA232,
1407 0x5AC5, 0x4B4C, 0x79D7, 0x685E, 0x1CE1, 0x0D68, 0x3FF3, 0x2E7A,
1408 0xE70E, 0xF687, 0xC41C, 0xD595, 0xA12A, 0xB0A3, 0x8238, 0x93B1,
1409 0x6B46, 0x7ACF, 0x4854, 0x59DD, 0x2D62, 0x3CEB, 0x0E70, 0x1FF9,
1410 0xF78F, 0xE606, 0xD49D, 0xC514, 0xB1AB, 0xA022, 0x92B9, 0x8330,
1411 0x7BC7, 0x6A4E, 0x58D5, 0x495C, 0x3DE3, 0x2C6A, 0x1EF1, 0x0F78
1412 };
1413
1414 uint16
hndcrc16(uint8 * pdata,uint nbytes,uint16 crc)1415 hndcrc16(
1416 uint8 *pdata, /* pointer to array of data to process */
1417 uint nbytes, /* number of input data bytes to process */
1418 uint16 crc /* either CRC16_INIT_VALUE or previous return value */
1419 )
1420 {
1421 while (nbytes-- > 0)
1422 CRC_INNER_LOOP(16, crc, *pdata++);
1423 return crc;
1424 }
1425
1426 static const uint32 crc32_table[256] = {
1427 0x00000000, 0x77073096, 0xEE0E612C, 0x990951BA,
1428 0x076DC419, 0x706AF48F, 0xE963A535, 0x9E6495A3,
1429 0x0EDB8832, 0x79DCB8A4, 0xE0D5E91E, 0x97D2D988,
1430 0x09B64C2B, 0x7EB17CBD, 0xE7B82D07, 0x90BF1D91,
1431 0x1DB71064, 0x6AB020F2, 0xF3B97148, 0x84BE41DE,
1432 0x1ADAD47D, 0x6DDDE4EB, 0xF4D4B551, 0x83D385C7,
1433 0x136C9856, 0x646BA8C0, 0xFD62F97A, 0x8A65C9EC,
1434 0x14015C4F, 0x63066CD9, 0xFA0F3D63, 0x8D080DF5,
1435 0x3B6E20C8, 0x4C69105E, 0xD56041E4, 0xA2677172,
1436 0x3C03E4D1, 0x4B04D447, 0xD20D85FD, 0xA50AB56B,
1437 0x35B5A8FA, 0x42B2986C, 0xDBBBC9D6, 0xACBCF940,
1438 0x32D86CE3, 0x45DF5C75, 0xDCD60DCF, 0xABD13D59,
1439 0x26D930AC, 0x51DE003A, 0xC8D75180, 0xBFD06116,
1440 0x21B4F4B5, 0x56B3C423, 0xCFBA9599, 0xB8BDA50F,
1441 0x2802B89E, 0x5F058808, 0xC60CD9B2, 0xB10BE924,
1442 0x2F6F7C87, 0x58684C11, 0xC1611DAB, 0xB6662D3D,
1443 0x76DC4190, 0x01DB7106, 0x98D220BC, 0xEFD5102A,
1444 0x71B18589, 0x06B6B51F, 0x9FBFE4A5, 0xE8B8D433,
1445 0x7807C9A2, 0x0F00F934, 0x9609A88E, 0xE10E9818,
1446 0x7F6A0DBB, 0x086D3D2D, 0x91646C97, 0xE6635C01,
1447 0x6B6B51F4, 0x1C6C6162, 0x856530D8, 0xF262004E,
1448 0x6C0695ED, 0x1B01A57B, 0x8208F4C1, 0xF50FC457,
1449 0x65B0D9C6, 0x12B7E950, 0x8BBEB8EA, 0xFCB9887C,
1450 0x62DD1DDF, 0x15DA2D49, 0x8CD37CF3, 0xFBD44C65,
1451 0x4DB26158, 0x3AB551CE, 0xA3BC0074, 0xD4BB30E2,
1452 0x4ADFA541, 0x3DD895D7, 0xA4D1C46D, 0xD3D6F4FB,
1453 0x4369E96A, 0x346ED9FC, 0xAD678846, 0xDA60B8D0,
1454 0x44042D73, 0x33031DE5, 0xAA0A4C5F, 0xDD0D7CC9,
1455 0x5005713C, 0x270241AA, 0xBE0B1010, 0xC90C2086,
1456 0x5768B525, 0x206F85B3, 0xB966D409, 0xCE61E49F,
1457 0x5EDEF90E, 0x29D9C998, 0xB0D09822, 0xC7D7A8B4,
1458 0x59B33D17, 0x2EB40D81, 0xB7BD5C3B, 0xC0BA6CAD,
1459 0xEDB88320, 0x9ABFB3B6, 0x03B6E20C, 0x74B1D29A,
1460 0xEAD54739, 0x9DD277AF, 0x04DB2615, 0x73DC1683,
1461 0xE3630B12, 0x94643B84, 0x0D6D6A3E, 0x7A6A5AA8,
1462 0xE40ECF0B, 0x9309FF9D, 0x0A00AE27, 0x7D079EB1,
1463 0xF00F9344, 0x8708A3D2, 0x1E01F268, 0x6906C2FE,
1464 0xF762575D, 0x806567CB, 0x196C3671, 0x6E6B06E7,
1465 0xFED41B76, 0x89D32BE0, 0x10DA7A5A, 0x67DD4ACC,
1466 0xF9B9DF6F, 0x8EBEEFF9, 0x17B7BE43, 0x60B08ED5,
1467 0xD6D6A3E8, 0xA1D1937E, 0x38D8C2C4, 0x4FDFF252,
1468 0xD1BB67F1, 0xA6BC5767, 0x3FB506DD, 0x48B2364B,
1469 0xD80D2BDA, 0xAF0A1B4C, 0x36034AF6, 0x41047A60,
1470 0xDF60EFC3, 0xA867DF55, 0x316E8EEF, 0x4669BE79,
1471 0xCB61B38C, 0xBC66831A, 0x256FD2A0, 0x5268E236,
1472 0xCC0C7795, 0xBB0B4703, 0x220216B9, 0x5505262F,
1473 0xC5BA3BBE, 0xB2BD0B28, 0x2BB45A92, 0x5CB36A04,
1474 0xC2D7FFA7, 0xB5D0CF31, 0x2CD99E8B, 0x5BDEAE1D,
1475 0x9B64C2B0, 0xEC63F226, 0x756AA39C, 0x026D930A,
1476 0x9C0906A9, 0xEB0E363F, 0x72076785, 0x05005713,
1477 0x95BF4A82, 0xE2B87A14, 0x7BB12BAE, 0x0CB61B38,
1478 0x92D28E9B, 0xE5D5BE0D, 0x7CDCEFB7, 0x0BDBDF21,
1479 0x86D3D2D4, 0xF1D4E242, 0x68DDB3F8, 0x1FDA836E,
1480 0x81BE16CD, 0xF6B9265B, 0x6FB077E1, 0x18B74777,
1481 0x88085AE6, 0xFF0F6A70, 0x66063BCA, 0x11010B5C,
1482 0x8F659EFF, 0xF862AE69, 0x616BFFD3, 0x166CCF45,
1483 0xA00AE278, 0xD70DD2EE, 0x4E048354, 0x3903B3C2,
1484 0xA7672661, 0xD06016F7, 0x4969474D, 0x3E6E77DB,
1485 0xAED16A4A, 0xD9D65ADC, 0x40DF0B66, 0x37D83BF0,
1486 0xA9BCAE53, 0xDEBB9EC5, 0x47B2CF7F, 0x30B5FFE9,
1487 0xBDBDF21C, 0xCABAC28A, 0x53B39330, 0x24B4A3A6,
1488 0xBAD03605, 0xCDD70693, 0x54DE5729, 0x23D967BF,
1489 0xB3667A2E, 0xC4614AB8, 0x5D681B02, 0x2A6F2B94,
1490 0xB40BBE37, 0xC30C8EA1, 0x5A05DF1B, 0x2D02EF8D
1491 };
1492
1493 /*
1494 * crc input is CRC32_INIT_VALUE for a fresh start, or previous return value if
1495 * accumulating over multiple pieces.
1496 */
1497 uint32
hndcrc32(uint8 * pdata,uint nbytes,uint32 crc)1498 hndcrc32(uint8 *pdata, uint nbytes, uint32 crc)
1499 {
1500 uint8 *pend;
1501 pend = pdata + nbytes;
1502 while (pdata < pend)
1503 CRC_INNER_LOOP(32, crc, *pdata++);
1504
1505 return crc;
1506 }
1507
1508 #ifdef notdef
1509 #define CLEN 1499 /* CRC Length */
1510 #define CBUFSIZ (CLEN+4)
1511 #define CNBUFS 5 /* # of bufs */
1512
1513 void
testcrc32(void)1514 testcrc32(void)
1515 {
1516 uint j, k, l;
1517 uint8 *buf;
1518 uint len[CNBUFS];
1519 uint32 crcr;
1520 uint32 crc32tv[CNBUFS] =
1521 {0xd2cb1faa, 0xd385c8fa, 0xf5b4f3f3, 0x55789e20, 0x00343110};
1522
1523 ASSERT((buf = MALLOC(CBUFSIZ*CNBUFS)) != NULL);
1524
1525 /* step through all possible alignments */
1526 for (l = 0; l <= 4; l++) {
1527 for (j = 0; j < CNBUFS; j++) {
1528 len[j] = CLEN;
1529 for (k = 0; k < len[j]; k++)
1530 *(buf + j*CBUFSIZ + (k+l)) = (j+k) & 0xff;
1531 }
1532
1533 for (j = 0; j < CNBUFS; j++) {
1534 crcr = crc32(buf + j*CBUFSIZ + l, len[j], CRC32_INIT_VALUE);
1535 ASSERT(crcr == crc32tv[j]);
1536 }
1537 }
1538
1539 MFREE(buf, CBUFSIZ*CNBUFS);
1540 return;
1541 }
1542 #endif /* notdef */
1543
1544 /*
1545 * Advance from the current 1-byte tag/1-byte length/variable-length value
1546 * triple, to the next, returning a pointer to the next.
1547 * If the current or next TLV is invalid (does not fit in given buffer length),
1548 * NULL is returned.
1549 * *buflen is not modified if the TLV elt parameter is invalid, or is decremented
1550 * by the TLV parameter's length if it is valid.
1551 */
1552 bcm_tlv_t *
bcm_next_tlv(bcm_tlv_t * elt,int * buflen)1553 bcm_next_tlv(bcm_tlv_t *elt, int *buflen)
1554 {
1555 int len;
1556
1557 /* validate current elt */
1558 if (!bcm_valid_tlv(elt, *buflen))
1559 return NULL;
1560
1561 /* advance to next elt */
1562 len = elt->len;
1563 elt = (bcm_tlv_t*)(elt->data + len);
1564 *buflen -= (TLV_HDR_LEN + len);
1565
1566 /* validate next elt */
1567 if (!bcm_valid_tlv(elt, *buflen))
1568 return NULL;
1569
1570 return elt;
1571 }
1572
1573 /*
1574 * Traverse a string of 1-byte tag/1-byte length/variable-length value
1575 * triples, returning a pointer to the substring whose first element
1576 * matches tag
1577 */
1578 bcm_tlv_t *
bcm_parse_tlvs(void * buf,int buflen,uint key)1579 bcm_parse_tlvs(void *buf, int buflen, uint key)
1580 {
1581 bcm_tlv_t *elt;
1582 int totlen;
1583
1584 elt = (bcm_tlv_t*)buf;
1585 totlen = buflen;
1586
1587 /* find tagged parameter */
1588 while (totlen >= TLV_HDR_LEN) {
1589 int len = elt->len;
1590
1591 /* validate remaining totlen */
1592 if ((elt->id == key) &&
1593 (totlen >= (len + TLV_HDR_LEN)))
1594 return (elt);
1595
1596 elt = (bcm_tlv_t*)((uint8*)elt + (len + TLV_HDR_LEN));
1597 totlen -= (len + TLV_HDR_LEN);
1598 }
1599
1600 return NULL;
1601 }
1602
1603 /*
1604 * Traverse a string of 1-byte tag/1-byte length/variable-length value
1605 * triples, returning a pointer to the substring whose first element
1606 * matches tag. Stop parsing when we see an element whose ID is greater
1607 * than the target key.
1608 */
1609 bcm_tlv_t *
bcm_parse_ordered_tlvs(void * buf,int buflen,uint key)1610 bcm_parse_ordered_tlvs(void *buf, int buflen, uint key)
1611 {
1612 bcm_tlv_t *elt;
1613 int totlen;
1614
1615 elt = (bcm_tlv_t*)buf;
1616 totlen = buflen;
1617
1618 /* find tagged parameter */
1619 while (totlen >= TLV_HDR_LEN) {
1620 uint id = elt->id;
1621 int len = elt->len;
1622
1623 /* Punt if we start seeing IDs > than target key */
1624 if (id > key)
1625 return (NULL);
1626
1627 /* validate remaining totlen */
1628 if ((id == key) &&
1629 (totlen >= (len + TLV_HDR_LEN)))
1630 return (elt);
1631
1632 elt = (bcm_tlv_t*)((uint8*)elt + (len + TLV_HDR_LEN));
1633 totlen -= (len + TLV_HDR_LEN);
1634 }
1635 return NULL;
1636 }
1637 #endif /* !BCMROMOFFLOAD_EXCLUDE_BCMUTILS_FUNCS */
1638
1639 #if defined(WLMSG_PRHDRS) || defined(WLMSG_PRPKT) || defined(WLMSG_ASSOC) || \
1640 defined(DHD_DEBUG)
1641 int
bcm_format_field(const bcm_bit_desc_ex_t * bd,uint32 flags,char * buf,int len)1642 bcm_format_field(const bcm_bit_desc_ex_t *bd, uint32 flags, char* buf, int len)
1643 {
1644 int i, slen = 0;
1645 uint32 bit, mask;
1646 const char *name;
1647 mask = bd->mask;
1648 if (len < 2 || !buf)
1649 return 0;
1650
1651 buf[0] = '\0';
1652
1653 for (i = 0; (name = bd->bitfield[i].name) != NULL; i++) {
1654 bit = bd->bitfield[i].bit;
1655 if ((flags & mask) == bit) {
1656 if (len > (int)strlen(name)) {
1657 slen = strlen(name);
1658 strncpy(buf, name, slen+1);
1659 }
1660 break;
1661 }
1662 }
1663 return slen;
1664 }
1665
1666 int
bcm_format_flags(const bcm_bit_desc_t * bd,uint32 flags,char * buf,int len)1667 bcm_format_flags(const bcm_bit_desc_t *bd, uint32 flags, char* buf, int len)
1668 {
1669 int i;
1670 char* p = buf;
1671 char hexstr[16];
1672 int slen = 0, nlen = 0;
1673 uint32 bit;
1674 const char* name;
1675
1676 if (len < 2 || !buf)
1677 return 0;
1678
1679 buf[0] = '\0';
1680
1681 for (i = 0; flags != 0; i++) {
1682 bit = bd[i].bit;
1683 name = bd[i].name;
1684 if (bit == 0 && flags != 0) {
1685 /* print any unnamed bits */
1686 snprintf(hexstr, 16, "0x%X", flags);
1687 name = hexstr;
1688 flags = 0; /* exit loop */
1689 } else if ((flags & bit) == 0)
1690 continue;
1691 flags &= ~bit;
1692 nlen = strlen(name);
1693 slen += nlen;
1694 /* count btwn flag space */
1695 if (flags != 0)
1696 slen += 1;
1697 /* need NULL char as well */
1698 if (len <= slen)
1699 break;
1700 /* copy NULL char but don't count it */
1701 strncpy(p, name, nlen + 1);
1702 p += nlen;
1703 /* copy btwn flag space and NULL char */
1704 if (flags != 0)
1705 p += snprintf(p, 2, " ");
1706 }
1707
1708 /* indicate the str was too short */
1709 if (flags != 0) {
1710 if (len < 2)
1711 p -= 2 - len; /* overwrite last char */
1712 p += snprintf(p, 2, ">");
1713 }
1714
1715 return (int)(p - buf);
1716 }
1717
1718 /* print bytes formatted as hex to a string. return the resulting string length */
1719 int
bcm_format_hex(char * str,const void * bytes,int len)1720 bcm_format_hex(char *str, const void *bytes, int len)
1721 {
1722 int i;
1723 char *p = str;
1724 const uint8 *src = (const uint8*)bytes;
1725
1726 for (i = 0; i < len; i++) {
1727 p += snprintf(p, 3, "%02X", *src);
1728 src++;
1729 }
1730 return (int)(p - str);
1731 }
1732 #endif
1733
1734 /* pretty hex print a contiguous buffer */
1735 void
prhex(const char * msg,uchar * buf,uint nbytes)1736 prhex(const char *msg, uchar *buf, uint nbytes)
1737 {
1738 char line[128], *p;
1739 int len = sizeof(line);
1740 int nchar;
1741 uint i;
1742
1743 if (msg && (msg[0] != '\0'))
1744 printf("%s:\n", msg);
1745
1746 p = line;
1747 for (i = 0; i < nbytes; i++) {
1748 if (i % 16 == 0) {
1749 nchar = snprintf(p, len, " %04d: ", i); /* line prefix */
1750 p += nchar;
1751 len -= nchar;
1752 }
1753 if (len > 0) {
1754 nchar = snprintf(p, len, "%02x ", buf[i]);
1755 p += nchar;
1756 len -= nchar;
1757 }
1758
1759 if (i % 16 == 15) {
1760 printf("%s\n", line); /* flush line */
1761 p = line;
1762 len = sizeof(line);
1763 }
1764 }
1765
1766 /* flush last partial line */
1767 if (p != line)
1768 printf("%s\n", line);
1769 }
1770
1771 static const char *crypto_algo_names[] = {
1772 "NONE",
1773 "WEP1",
1774 "TKIP",
1775 "WEP128",
1776 "AES_CCM",
1777 "AES_OCB_MSDU",
1778 "AES_OCB_MPDU",
1779 "NALG",
1780 "UNDEF",
1781 "UNDEF",
1782 "UNDEF",
1783 "UNDEF"
1784 };
1785
1786 const char *
bcm_crypto_algo_name(uint algo)1787 bcm_crypto_algo_name(uint algo)
1788 {
1789 return (algo < ARRAYSIZE(crypto_algo_names)) ? crypto_algo_names[algo] : "ERR";
1790 }
1791
1792
1793 char *
bcm_chipname(uint chipid,char * buf,uint len)1794 bcm_chipname(uint chipid, char *buf, uint len)
1795 {
1796 const char *fmt;
1797
1798 fmt = ((chipid > 0xa000) || (chipid < 0x4000)) ? "%d" : "%x";
1799 snprintf(buf, len, fmt, chipid);
1800 return buf;
1801 }
1802
1803 /* Produce a human-readable string for boardrev */
1804 char *
bcm_brev_str(uint32 brev,char * buf)1805 bcm_brev_str(uint32 brev, char *buf)
1806 {
1807 if (brev < 0x100)
1808 snprintf(buf, 8, "%d.%d", (brev & 0xf0) >> 4, brev & 0xf);
1809 else
1810 snprintf(buf, 8, "%c%03x", ((brev & 0xf000) == 0x1000) ? 'P' : 'A', brev & 0xfff);
1811
1812 return (buf);
1813 }
1814
1815 #define BUFSIZE_TODUMP_ATONCE 512 /* Buffer size */
1816
1817 /* dump large strings to console */
1818 void
printbig(char * buf)1819 printbig(char *buf)
1820 {
1821 uint len, max_len;
1822 char c;
1823
1824 len = strlen(buf);
1825
1826 max_len = BUFSIZE_TODUMP_ATONCE;
1827
1828 while (len > max_len) {
1829 c = buf[max_len];
1830 buf[max_len] = '\0';
1831 printf("%s", buf);
1832 buf[max_len] = c;
1833
1834 buf += max_len;
1835 len -= max_len;
1836 }
1837 /* print the remaining string */
1838 printf("%s\n", buf);
1839 return;
1840 }
1841
1842 /* routine to dump fields in a fileddesc structure */
1843 uint
bcmdumpfields(bcmutl_rdreg_rtn read_rtn,void * arg0,uint arg1,struct fielddesc * fielddesc_array,char * buf,uint32 bufsize)1844 bcmdumpfields(bcmutl_rdreg_rtn read_rtn, void *arg0, uint arg1, struct fielddesc *fielddesc_array,
1845 char *buf, uint32 bufsize)
1846 {
1847 uint filled_len;
1848 int len;
1849 struct fielddesc *cur_ptr;
1850
1851 filled_len = 0;
1852 cur_ptr = fielddesc_array;
1853
1854 while (bufsize > 1) {
1855 if (cur_ptr->nameandfmt == NULL)
1856 break;
1857 len = snprintf(buf, bufsize, cur_ptr->nameandfmt,
1858 read_rtn(arg0, arg1, cur_ptr->offset));
1859 /* check for snprintf overflow or error */
1860 if (len < 0 || (uint32)len >= bufsize)
1861 len = bufsize - 1;
1862 buf += len;
1863 bufsize -= len;
1864 filled_len += len;
1865 cur_ptr++;
1866 }
1867 return filled_len;
1868 }
1869
1870 uint
bcm_mkiovar(char * name,char * data,uint datalen,char * buf,uint buflen)1871 bcm_mkiovar(char *name, char *data, uint datalen, char *buf, uint buflen)
1872 {
1873 uint len;
1874
1875 len = strlen(name) + 1;
1876
1877 if ((len + datalen) > buflen)
1878 return 0;
1879
1880 strncpy(buf, name, buflen);
1881
1882 /* append data onto the end of the name string */
1883 memcpy(&buf[len], data, datalen);
1884 len += datalen;
1885
1886 return len;
1887 }
1888
1889 /* Quarter dBm units to mW
1890 * Table starts at QDBM_OFFSET, so the first entry is mW for qdBm=153
1891 * Table is offset so the last entry is largest mW value that fits in
1892 * a uint16.
1893 */
1894
1895 #define QDBM_OFFSET 153 /* Offset for first entry */
1896 #define QDBM_TABLE_LEN 40 /* Table size */
1897
1898 /* Smallest mW value that will round up to the first table entry, QDBM_OFFSET.
1899 * Value is ( mW(QDBM_OFFSET - 1) + mW(QDBM_OFFSET) ) / 2
1900 */
1901 #define QDBM_TABLE_LOW_BOUND 6493 /* Low bound */
1902
1903 /* Largest mW value that will round down to the last table entry,
1904 * QDBM_OFFSET + QDBM_TABLE_LEN-1.
1905 * Value is ( mW(QDBM_OFFSET + QDBM_TABLE_LEN - 1) + mW(QDBM_OFFSET + QDBM_TABLE_LEN) ) / 2.
1906 */
1907 #define QDBM_TABLE_HIGH_BOUND 64938 /* High bound */
1908
1909 static const uint16 nqdBm_to_mW_map[QDBM_TABLE_LEN] = {
1910 /* qdBm: +0 +1 +2 +3 +4 +5 +6 +7 */
1911 /* 153: */ 6683, 7079, 7499, 7943, 8414, 8913, 9441, 10000,
1912 /* 161: */ 10593, 11220, 11885, 12589, 13335, 14125, 14962, 15849,
1913 /* 169: */ 16788, 17783, 18836, 19953, 21135, 22387, 23714, 25119,
1914 /* 177: */ 26607, 28184, 29854, 31623, 33497, 35481, 37584, 39811,
1915 /* 185: */ 42170, 44668, 47315, 50119, 53088, 56234, 59566, 63096
1916 };
1917
1918 uint16
bcm_qdbm_to_mw(uint8 qdbm)1919 bcm_qdbm_to_mw(uint8 qdbm)
1920 {
1921 uint factor = 1;
1922 int idx = qdbm - QDBM_OFFSET;
1923
1924 if (idx >= QDBM_TABLE_LEN) {
1925 /* clamp to max uint16 mW value */
1926 return 0xFFFF;
1927 }
1928
1929 /* scale the qdBm index up to the range of the table 0-40
1930 * where an offset of 40 qdBm equals a factor of 10 mW.
1931 */
1932 while (idx < 0) {
1933 idx += 40;
1934 factor *= 10;
1935 }
1936
1937 /* return the mW value scaled down to the correct factor of 10,
1938 * adding in factor/2 to get proper rounding.
1939 */
1940 return ((nqdBm_to_mW_map[idx] + factor/2) / factor);
1941 }
1942
1943 uint8
bcm_mw_to_qdbm(uint16 mw)1944 bcm_mw_to_qdbm(uint16 mw)
1945 {
1946 uint8 qdbm;
1947 int offset;
1948 uint mw_uint = mw;
1949 uint boundary;
1950
1951 /* handle boundary case */
1952 if (mw_uint <= 1)
1953 return 0;
1954
1955 offset = QDBM_OFFSET;
1956
1957 /* move mw into the range of the table */
1958 while (mw_uint < QDBM_TABLE_LOW_BOUND) {
1959 mw_uint *= 10;
1960 offset -= 40;
1961 }
1962
1963 for (qdbm = 0; qdbm < QDBM_TABLE_LEN-1; qdbm++) {
1964 boundary = nqdBm_to_mW_map[qdbm] + (nqdBm_to_mW_map[qdbm+1] -
1965 nqdBm_to_mW_map[qdbm])/2;
1966 if (mw_uint < boundary) break;
1967 }
1968
1969 qdbm += (uint8)offset;
1970
1971 return (qdbm);
1972 }
1973
1974
1975 uint
bcm_bitcount(uint8 * bitmap,uint length)1976 bcm_bitcount(uint8 *bitmap, uint length)
1977 {
1978 uint bitcount = 0, i;
1979 uint8 tmp;
1980 for (i = 0; i < length; i++) {
1981 tmp = bitmap[i];
1982 while (tmp) {
1983 bitcount++;
1984 tmp &= (tmp - 1);
1985 }
1986 }
1987 return bitcount;
1988 }
1989
1990 #ifdef BCMDRIVER
1991
1992 /* Initialization of bcmstrbuf structure */
1993 void
bcm_binit(struct bcmstrbuf * b,char * buf,uint size)1994 bcm_binit(struct bcmstrbuf *b, char *buf, uint size)
1995 {
1996 b->origsize = b->size = size;
1997 b->origbuf = b->buf = buf;
1998 }
1999
2000 /* Buffer sprintf wrapper to guard against buffer overflow */
2001 int
bcm_bprintf(struct bcmstrbuf * b,const char * fmt,...)2002 bcm_bprintf(struct bcmstrbuf *b, const char *fmt, ...)
2003 {
2004 va_list ap;
2005 int r;
2006
2007 va_start(ap, fmt);
2008
2009 r = vsnprintf(b->buf, b->size, fmt, ap);
2010
2011 /* Non Ansi C99 compliant returns -1,
2012 * Ansi compliant return r >= b->size,
2013 * bcmstdlib returns 0, handle all
2014 */
2015 /* r == 0 is also the case when strlen(fmt) is zero.
2016 * typically the case when "" is passed as argument.
2017 */
2018 if ((r == -1) || (r >= (int)b->size)) {
2019 b->size = 0;
2020 } else {
2021 b->size -= r;
2022 b->buf += r;
2023 }
2024
2025 va_end(ap);
2026
2027 return r;
2028 }
2029
2030 void
bcm_bprhex(struct bcmstrbuf * b,const char * msg,bool newline,uint8 * buf,int len)2031 bcm_bprhex(struct bcmstrbuf *b, const char *msg, bool newline, uint8 *buf, int len)
2032 {
2033 int i;
2034
2035 if (msg != NULL && msg[0] != '\0')
2036 bcm_bprintf(b, "%s", msg);
2037 for (i = 0; i < len; i ++)
2038 bcm_bprintf(b, "%02X", buf[i]);
2039 if (newline)
2040 bcm_bprintf(b, "\n");
2041 }
2042
2043 void
bcm_inc_bytes(uchar * num,int num_bytes,uint8 amount)2044 bcm_inc_bytes(uchar *num, int num_bytes, uint8 amount)
2045 {
2046 int i;
2047
2048 for (i = 0; i < num_bytes; i++) {
2049 num[i] += amount;
2050 if (num[i] >= amount)
2051 break;
2052 amount = 1;
2053 }
2054 }
2055
2056 int
bcm_cmp_bytes(const uchar * arg1,const uchar * arg2,uint8 nbytes)2057 bcm_cmp_bytes(const uchar *arg1, const uchar *arg2, uint8 nbytes)
2058 {
2059 int i;
2060
2061 for (i = nbytes - 1; i >= 0; i--) {
2062 if (arg1[i] != arg2[i])
2063 return (arg1[i] - arg2[i]);
2064 }
2065 return 0;
2066 }
2067
2068 void
bcm_print_bytes(const char * name,const uchar * data,int len)2069 bcm_print_bytes(const char *name, const uchar *data, int len)
2070 {
2071 int i;
2072 int per_line = 0;
2073
2074 printf("%s: %d \n", name ? name : "", len);
2075 for (i = 0; i < len; i++) {
2076 printf("%02x ", *data++);
2077 per_line++;
2078 if (per_line == 16) {
2079 per_line = 0;
2080 printf("\n");
2081 }
2082 }
2083 printf("\n");
2084 }
2085
2086 /* Look for vendor-specific IE with specified OUI and optional type */
2087 bcm_tlv_t *
find_vendor_ie(void * tlvs,int tlvs_len,const char * voui,uint8 * type,int type_len)2088 find_vendor_ie(void *tlvs, int tlvs_len, const char *voui, uint8 *type, int type_len)
2089 {
2090 bcm_tlv_t *ie;
2091 uint8 ie_len;
2092
2093 ie = (bcm_tlv_t*)tlvs;
2094
2095 /* make sure we are looking at a valid IE */
2096 if (ie == NULL ||
2097 !bcm_valid_tlv(ie, tlvs_len))
2098 return NULL;
2099
2100 /* Walk through the IEs looking for an OUI match */
2101 do {
2102 ie_len = ie->len;
2103 if ((ie->id == DOT11_MNG_PROPR_ID) &&
2104 (ie_len >= (DOT11_OUI_LEN + type_len)) &&
2105 !bcmp(ie->data, voui, DOT11_OUI_LEN))
2106 {
2107 /* compare optional type */
2108 if (type_len == 0 ||
2109 !bcmp(&ie->data[DOT11_OUI_LEN], type, type_len)) {
2110 return (ie); /* a match */
2111 }
2112 }
2113 } while ((ie = bcm_next_tlv(ie, &tlvs_len)) != NULL);
2114
2115 return NULL;
2116 }
2117
2118 #if defined(WLTINYDUMP) || defined(WLMSG_INFORM) || defined(WLMSG_ASSOC) || \
2119 defined(WLMSG_PRPKT) || defined(WLMSG_WSEC)
2120 #define SSID_FMT_BUF_LEN ((4 * DOT11_MAX_SSID_LEN) + 1)
2121
2122 int
bcm_format_ssid(char * buf,const uchar ssid[],uint ssid_len)2123 bcm_format_ssid(char* buf, const uchar ssid[], uint ssid_len)
2124 {
2125 uint i, c;
2126 char *p = buf;
2127 char *endp = buf + SSID_FMT_BUF_LEN;
2128
2129 if (ssid_len > DOT11_MAX_SSID_LEN) ssid_len = DOT11_MAX_SSID_LEN;
2130
2131 for (i = 0; i < ssid_len; i++) {
2132 c = (uint)ssid[i];
2133 if (c == '\\') {
2134 *p++ = '\\';
2135 *p++ = '\\';
2136 } else if (bcm_isprint((uchar)c)) {
2137 *p++ = (char)c;
2138 } else {
2139 p += snprintf(p, (endp - p), "\\x%02X", c);
2140 }
2141 }
2142 *p = '\0';
2143 ASSERT(p < endp);
2144
2145 return (int)(p - buf);
2146 }
2147 #endif
2148
2149 #endif /* BCMDRIVER */
2150
2151 /*
2152 * ProcessVars:Takes a buffer of "<var>=<value>\n" lines read from a file and ending in a NUL.
2153 * also accepts nvram files which are already in the format of <var1>=<value>\0\<var2>=<value2>\0
2154 * Removes carriage returns, empty lines, comment lines, and converts newlines to NULs.
2155 * Shortens buffer as needed and pads with NULs. End of buffer is marked by two NULs.
2156 */
2157
2158 unsigned int
process_nvram_vars(char * varbuf,unsigned int len)2159 process_nvram_vars(char *varbuf, unsigned int len)
2160 {
2161 char *dp;
2162 bool findNewline;
2163 int column;
2164 unsigned int buf_len, n;
2165 unsigned int pad = 0;
2166
2167 dp = varbuf;
2168
2169 findNewline = FALSE;
2170 column = 0;
2171
2172 for (n = 0; n < len; n++) {
2173 if (varbuf[n] == '\r')
2174 continue;
2175 if (findNewline && varbuf[n] != '\n')
2176 continue;
2177 findNewline = FALSE;
2178 if (varbuf[n] == '#') {
2179 findNewline = TRUE;
2180 continue;
2181 }
2182 if (varbuf[n] == '\n') {
2183 if (column == 0)
2184 continue;
2185 *dp++ = 0;
2186 column = 0;
2187 continue;
2188 }
2189 *dp++ = varbuf[n];
2190 column++;
2191 }
2192 buf_len = (unsigned int)(dp - varbuf);
2193 if (buf_len % 4) {
2194 pad = 4 - buf_len % 4;
2195 if (pad && (buf_len + pad <= len)) {
2196 buf_len += pad;
2197 }
2198 }
2199
2200 while (dp < varbuf + n)
2201 *dp++ = 0;
2202
2203 return buf_len;
2204 }
2205
2206 /* calculate a * b + c */
2207 void
bcm_uint64_multiple_add(uint32 * r_high,uint32 * r_low,uint32 a,uint32 b,uint32 c)2208 bcm_uint64_multiple_add(uint32* r_high, uint32* r_low, uint32 a, uint32 b, uint32 c)
2209 {
2210 #define FORMALIZE(var) {cc += (var & 0x80000000) ? 1 : 0; var &= 0x7fffffff;}
2211 uint32 r1, r0;
2212 uint32 a1, a0, b1, b0, t, cc = 0;
2213
2214 a1 = a >> 16;
2215 a0 = a & 0xffff;
2216 b1 = b >> 16;
2217 b0 = b & 0xffff;
2218
2219 r0 = a0 * b0;
2220 FORMALIZE(r0);
2221
2222 t = (a1 * b0) << 16;
2223 FORMALIZE(t);
2224
2225 r0 += t;
2226 FORMALIZE(r0);
2227
2228 t = (a0 * b1) << 16;
2229 FORMALIZE(t);
2230
2231 r0 += t;
2232 FORMALIZE(r0);
2233
2234 FORMALIZE(c);
2235
2236 r0 += c;
2237 FORMALIZE(r0);
2238
2239 r0 |= (cc % 2) ? 0x80000000 : 0;
2240 r1 = a1 * b1 + ((a1 * b0) >> 16) + ((b1 * a0) >> 16) + (cc / 2);
2241
2242 *r_high = r1;
2243 *r_low = r0;
2244 }
2245
2246 /* calculate a / b */
2247 void
bcm_uint64_divide(uint32 * r,uint32 a_high,uint32 a_low,uint32 b)2248 bcm_uint64_divide(uint32* r, uint32 a_high, uint32 a_low, uint32 b)
2249 {
2250 uint32 a1 = a_high, a0 = a_low, r0 = 0;
2251
2252 if (b < 2)
2253 return;
2254
2255 while (a1 != 0) {
2256 r0 += (0xffffffff / b) * a1;
2257 bcm_uint64_multiple_add(&a1, &a0, ((0xffffffff % b) + 1) % b, a1, a0);
2258 }
2259
2260 r0 += a0 / b;
2261 *r = r0;
2262 }
2263
2264 #ifndef setbit /* As in the header file */
2265 #ifdef BCMUTILS_BIT_MACROS_USE_FUNCS
2266 /* Set bit in byte array. */
2267 void
setbit(void * array,uint bit)2268 setbit(void *array, uint bit)
2269 {
2270 ((uint8 *)array)[bit / NBBY] |= 1 << (bit % NBBY);
2271 }
2272
2273 /* Clear bit in byte array. */
2274 void
clrbit(void * array,uint bit)2275 clrbit(void *array, uint bit)
2276 {
2277 ((uint8 *)array)[bit / NBBY] &= ~(1 << (bit % NBBY));
2278 }
2279
2280 /* Test if bit is set in byte array. */
2281 bool
isset(const void * array,uint bit)2282 isset(const void *array, uint bit)
2283 {
2284 return (((const uint8 *)array)[bit / NBBY] & (1 << (bit % NBBY)));
2285 }
2286
2287 /* Test if bit is clear in byte array. */
2288 bool
isclr(const void * array,uint bit)2289 isclr(const void *array, uint bit)
2290 {
2291 return ((((const uint8 *)array)[bit / NBBY] & (1 << (bit % NBBY))) == 0);
2292 }
2293 #endif /* BCMUTILS_BIT_MACROS_USE_FUNCS */
2294 #endif /* setbit */
2295