kallsyms.c 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722
  1. /*
  2. * from linux/kernel/kallsyms.c
  3. *
  4. * kallsyms.c: in-kernel printing of symbolic oopses and stack traces.
  5. *
  6. * Rewritten and vastly simplified by Rusty Russell for in-kernel
  7. * module loader:
  8. * Copyright 2002 Rusty Russell <rusty@rustcorp.com.au> IBM Corporation
  9. *
  10. * ChangeLog:
  11. *
  12. * (25/Aug/2004) Paulo Marques <pmarques@grupopie.com>
  13. * Changed the compression method from stem compression to "table lookup"
  14. * compression (see scripts/kallsyms.c for a more complete description)
  15. */
  16. #include <kernel.h>
  17. #include <string.h>
  18. #include <stdio.h>
  19. #include <kallsyms.h>
  20. #include <sdfs.h>
  21. #if 0
  22. /*
  23. * These will be re-linked against their real values
  24. * during the second link stage.
  25. */
  26. extern const unsigned long kallsyms_addresses[] __attribute__((weak));
  27. extern const u8_t kallsyms_names[] __attribute__((weak));
  28. /*
  29. * Tell the compiler that the count isn't in the small data section if the arch
  30. * has one (eg: FRV).
  31. */
  32. extern const unsigned long kallsyms_num_syms
  33. __attribute__((weak, section(".rodata")));
  34. extern const u8_t kallsyms_token_table[] __attribute__((weak));
  35. extern const u16_t kallsyms_token_index[] __attribute__((weak));
  36. extern const unsigned long kallsyms_markers[] __attribute__((weak));
  37. extern const unsigned long __text_region_start, __text_region_end;
  38. extern const unsigned long __ramfunc_start, __ramfunc_end;
  39. int is_ksym_addr(unsigned long addr)
  40. {
  41. if ((addr >= (unsigned long)&__text_region_start &&
  42. addr <= (unsigned long)&__text_region_end))
  43. return 1;
  44. if ((addr >= (unsigned long)&__ramfunc_start &&
  45. addr <= (unsigned long)&__ramfunc_end))
  46. return 1;
  47. return 0;
  48. }
  49. /*
  50. * Expand a compressed symbol data into the resulting uncompressed string,
  51. * if uncompressed string is too long (>= maxlen), it will be truncated,
  52. * given the offset to where the symbol is in the compressed stream.
  53. */
  54. static unsigned int kallsyms_expand_symbol(unsigned int off,
  55. char *result, size_t maxlen)
  56. {
  57. int len, skipped_first = 0;
  58. const u8_t *tptr, *data;
  59. /* Get the compressed symbol length from the first symbol byte. */
  60. data = &kallsyms_names[off];
  61. len = *data;
  62. data++;
  63. /*
  64. * Update the offset to return the offset for the next symbol on
  65. * the compressed stream.
  66. */
  67. off += len + 1;
  68. /*
  69. * For every byte on the compressed symbol data, copy the table
  70. * entry for that byte.
  71. */
  72. while (len) {
  73. tptr = &kallsyms_token_table[kallsyms_token_index[*data]];
  74. data++;
  75. len--;
  76. while (*tptr) {
  77. if (skipped_first) {
  78. if (maxlen <= 1)
  79. goto tail;
  80. *result = *tptr;
  81. result++;
  82. maxlen--;
  83. } else
  84. skipped_first = 1;
  85. tptr++;
  86. }
  87. }
  88. tail:
  89. if (maxlen)
  90. *result = '\0';
  91. /* Return to offset to the next symbol. */
  92. return off;
  93. }
  94. /*
  95. * Find the offset on the compressed stream given and index in the
  96. * kallsyms array.
  97. */
  98. static unsigned int get_symbol_offset(unsigned long pos)
  99. {
  100. const u8_t *name;
  101. int i;
  102. /*
  103. * Use the closest marker we have. We have markers every 256 positions,
  104. * so that should be close enough.
  105. */
  106. name = &kallsyms_names[kallsyms_markers[pos >> 8]];
  107. /*
  108. * Sequentially scan all the symbols up to the point we're searching
  109. * for. Every symbol is stored in a [<len>][<len> bytes of data] format,
  110. * so we just need to add the len to the current pointer for every
  111. * symbol we wish to skip.
  112. */
  113. for (i = 0; i < (pos & 0xFF); i++)
  114. name = name + (*name) + 1;
  115. return name - kallsyms_names;
  116. }
  117. static unsigned long get_symbol_pos(unsigned long addr,
  118. unsigned long *symbolsize,
  119. unsigned long *offset)
  120. {
  121. unsigned long symbol_start = 0, symbol_end = 0;
  122. unsigned long i, low, high, mid;
  123. /* Do a binary search on the sorted kallsyms_addresses array. */
  124. low = 0;
  125. high = kallsyms_num_syms;
  126. while (high - low > 1) {
  127. mid = low + (high - low) / 2;
  128. if (kallsyms_addresses[mid] <= addr)
  129. low = mid;
  130. else
  131. high = mid;
  132. }
  133. /*
  134. * Search for the first aliased symbol. Aliased
  135. * symbols are symbols with the same address.
  136. */
  137. while (low && kallsyms_addresses[low-1] == kallsyms_addresses[low])
  138. --low;
  139. symbol_start = kallsyms_addresses[low];
  140. /* Search for next non-aliased symbol. */
  141. for (i = low + 1; i < kallsyms_num_syms; i++) {
  142. if (kallsyms_addresses[i] > symbol_start) {
  143. symbol_end = kallsyms_addresses[i];
  144. break;
  145. }
  146. }
  147. /* If we found no next symbol, we use the end of the section. */
  148. if (!symbol_end) {
  149. symbol_end = (unsigned long)__text_region_end;
  150. }
  151. if (symbolsize)
  152. *symbolsize = symbol_end - symbol_start;
  153. if (offset)
  154. *offset = addr - symbol_start;
  155. return low;
  156. }
  157. /*
  158. * Lookup an address but don't bother to find any names.
  159. */
  160. int kallsyms_lookup_size_offset(unsigned long addr, unsigned long *symbolsize,
  161. unsigned long *offset)
  162. {
  163. if (is_ksym_addr(addr))
  164. return !!get_symbol_pos(addr, symbolsize, offset);
  165. return 0;
  166. }
  167. /*
  168. * Lookup an address
  169. * - modname is set to NULL if it's in the kernel.
  170. * - We guarantee that the returned name is valid until we reschedule even if.
  171. * It resides in a module.
  172. * - We also guarantee that modname will be valid until rescheduled.
  173. */
  174. const char *kallsyms_lookup(unsigned long addr,
  175. unsigned long *symbolsize,
  176. unsigned long *offset,
  177. char **modname, char *namebuf)
  178. {
  179. namebuf[KSYM_NAME_LEN - 1] = 0;
  180. namebuf[0] = 0;
  181. if (is_ksym_addr(addr)) {
  182. unsigned long pos;
  183. pos = get_symbol_pos(addr, symbolsize, offset);
  184. /* Grab name */
  185. kallsyms_expand_symbol(get_symbol_offset(pos),
  186. namebuf, KSYM_NAME_LEN);
  187. if (modname)
  188. *modname = NULL;
  189. return namebuf;
  190. }
  191. return NULL;
  192. }
  193. /* Look up a kernel symbol and return it in a text buffer. */
  194. static int __sprint_symbol(char *buffer, unsigned long address,
  195. int symbol_offset, int add_offset)
  196. {
  197. char *modname;
  198. const char *name;
  199. unsigned long offset, size;
  200. int len;
  201. address += symbol_offset;
  202. name = kallsyms_lookup(address, &size, &offset, &modname, buffer);
  203. if (!name)
  204. return sprintf(buffer, "0x%x", (unsigned int)address);
  205. if (name != buffer)
  206. strcpy(buffer, name);
  207. len = strlen(buffer);
  208. offset -= symbol_offset;
  209. if (add_offset)
  210. len += sprintf(buffer + len, "+%#x/%#x", (unsigned int)offset, (unsigned int)size);
  211. if (modname)
  212. len += sprintf(buffer + len, " [%s]", modname);
  213. return len;
  214. }
  215. /**
  216. * sprint_symbol - Look up a kernel symbol and return it in a text buffer
  217. * @buffer: buffer to be stored
  218. * @address: address to lookup
  219. *
  220. * This function looks up a kernel symbol with @address and stores its name,
  221. * offset, size and module name to @buffer if possible. If no symbol was found,
  222. * just saves its @address as is.
  223. *
  224. * This function returns the number of bytes stored in @buffer.
  225. */
  226. int sprint_symbol(char *buffer, unsigned long address)
  227. {
  228. return __sprint_symbol(buffer, address, 0, 1);
  229. }
  230. /**
  231. * sprint_symbol_no_offset - Look up a kernel symbol and return it in a text buffer
  232. * @buffer: buffer to be stored
  233. * @address: address to lookup
  234. *
  235. * This function looks up a kernel symbol with @address and stores its name
  236. * and module name to @buffer if possible. If no symbol was found, just saves
  237. * its @address as is.
  238. *
  239. * This function returns the number of bytes stored in @buffer.
  240. */
  241. int sprint_symbol_no_offset(char *buffer, unsigned long address)
  242. {
  243. return __sprint_symbol(buffer, address, 0, 0);
  244. }
  245. /**
  246. * sprint_backtrace - Look up a backtrace symbol and return it in a text buffer
  247. * @buffer: buffer to be stored
  248. * @address: address to lookup
  249. *
  250. * This function is for stack backtrace and does the same thing as
  251. * sprint_symbol() but with modified/decreased @address. If there is a
  252. * tail-call to the function marked "noreturn", gcc optimized out code after
  253. * the call so that the stack-saved return address could point outside of the
  254. * caller. This function ensures that kallsyms will find the original caller
  255. * by decreasing @address.
  256. *
  257. * This function returns the number of bytes stored in @buffer.
  258. */
  259. int sprint_backtrace(char *buffer, unsigned long address)
  260. {
  261. return __sprint_symbol(buffer, address, -1, 1);
  262. }
  263. /* Look up a kernel symbol and print it to the kernel messages. */
  264. static void __print_symbol(const char *fmt, unsigned long address)
  265. {
  266. char buffer[KSYM_SYMBOL_LEN];
  267. sprint_symbol(buffer, address);
  268. printk(fmt, buffer);
  269. }
  270. void print_symbol(const char *fmt, unsigned long addr)
  271. {
  272. __print_symbol(fmt, (unsigned long)
  273. __builtin_extract_return_addr((void *)addr));
  274. }
  275. #else
  276. /*
  277. * These will be re-linked against their real values
  278. * during the second link stage.
  279. */
  280. static unsigned long *kallsyms_addresses;
  281. static u8_t *kallsyms_names;
  282. static unsigned long kallsyms_num_syms;
  283. static u8_t *kallsyms_token_table;
  284. static u16_t *kallsyms_token_index;
  285. static unsigned long *kallsyms_markers;
  286. extern const unsigned long __text_region_start, __text_region_end;
  287. extern const unsigned long __ramfunc_start, __ramfunc_end;
  288. int is_ksym_addr(unsigned long addr)
  289. {
  290. if ((addr >= (unsigned long)&__text_region_start &&
  291. addr <= (unsigned long)&__text_region_end))
  292. return 1;
  293. if ((addr >= (unsigned long)&__ramfunc_start &&
  294. addr <= (unsigned long)&__ramfunc_end))
  295. return 1;
  296. return 0;
  297. }
  298. /*
  299. * Expand a compressed symbol data into the resulting uncompressed string,
  300. * if uncompressed string is too long (>= maxlen), it will be truncated,
  301. * given the offset to where the symbol is in the compressed stream.
  302. */
  303. static unsigned int kallsyms_expand_symbol(unsigned int off,
  304. char *result, size_t maxlen)
  305. {
  306. int len, skipped_first = 0;
  307. const u8_t *tptr, *data;
  308. /* Get the compressed symbol length from the first symbol byte. */
  309. data = &kallsyms_names[off];
  310. len = *data;
  311. data++;
  312. /*
  313. * Update the offset to return the offset for the next symbol on
  314. * the compressed stream.
  315. */
  316. off += len + 1;
  317. /*
  318. * For every byte on the compressed symbol data, copy the table
  319. * entry for that byte.
  320. */
  321. while (len) {
  322. tptr = &kallsyms_token_table[kallsyms_token_index[*data]];
  323. data++;
  324. len--;
  325. while (*tptr) {
  326. if (skipped_first) {
  327. if (maxlen <= 1)
  328. goto tail;
  329. *result = *tptr;
  330. result++;
  331. maxlen--;
  332. } else
  333. skipped_first = 1;
  334. tptr++;
  335. }
  336. }
  337. tail:
  338. if (maxlen)
  339. *result = '\0';
  340. /* Return to offset to the next symbol. */
  341. return off;
  342. }
  343. /*
  344. * Find the offset on the compressed stream given and index in the
  345. * kallsyms array.
  346. */
  347. static unsigned int get_symbol_offset(unsigned long pos)
  348. {
  349. const u8_t *name;
  350. int i;
  351. /*
  352. * Use the closest marker we have. We have markers every 256 positions,
  353. * so that should be close enough.
  354. */
  355. name = &kallsyms_names[kallsyms_markers[pos >> 8]];
  356. /*
  357. * Sequentially scan all the symbols up to the point we're searching
  358. * for. Every symbol is stored in a [<len>][<len> bytes of data] format,
  359. * so we just need to add the len to the current pointer for every
  360. * symbol we wish to skip.
  361. */
  362. for (i = 0; i < (pos & 0xFF); i++)
  363. name = name + (*name) + 1;
  364. return name - kallsyms_names;
  365. }
  366. static unsigned long get_symbol_pos(unsigned long addr,
  367. unsigned long *symbolsize,
  368. unsigned long *offset)
  369. {
  370. unsigned long symbol_start = 0, symbol_end = 0;
  371. unsigned long i, low, high, mid;
  372. /* Do a binary search on the sorted kallsyms_addresses array. */
  373. low = 0;
  374. high = kallsyms_num_syms;
  375. while (high - low > 1) {
  376. mid = low + (high - low) / 2;
  377. if (kallsyms_addresses[mid] <= addr)
  378. low = mid;
  379. else
  380. high = mid;
  381. }
  382. /*
  383. * Search for the first aliased symbol. Aliased
  384. * symbols are symbols with the same address.
  385. */
  386. while (low && kallsyms_addresses[low-1] == kallsyms_addresses[low])
  387. --low;
  388. symbol_start = kallsyms_addresses[low];
  389. /* Search for next non-aliased symbol. */
  390. for (i = low + 1; i < kallsyms_num_syms; i++) {
  391. if (kallsyms_addresses[i] > symbol_start) {
  392. symbol_end = kallsyms_addresses[i];
  393. break;
  394. }
  395. }
  396. /* If we found no next symbol, we use the end of the section. */
  397. if (!symbol_end) {
  398. symbol_end = (unsigned long)__text_region_end;
  399. }
  400. if (symbolsize)
  401. *symbolsize = symbol_end - symbol_start;
  402. if (offset)
  403. *offset = addr - symbol_start;
  404. return low;
  405. }
  406. /*
  407. * Lookup an address but don't bother to find any names.
  408. */
  409. int kallsyms_lookup_size_offset(unsigned long addr, unsigned long *symbolsize,
  410. unsigned long *offset)
  411. {
  412. if (is_ksym_addr(addr))
  413. return !!get_symbol_pos(addr, symbolsize, offset);
  414. return 0;
  415. }
  416. /*
  417. * Lookup an address
  418. * - modname is set to NULL if it's in the kernel.
  419. * - We guarantee that the returned name is valid until we reschedule even if.
  420. * It resides in a module.
  421. * - We also guarantee that modname will be valid until rescheduled.
  422. */
  423. const char *kallsyms_lookup(unsigned long addr,
  424. unsigned long *symbolsize,
  425. unsigned long *offset,
  426. char **modname, char *namebuf)
  427. {
  428. namebuf[KSYM_NAME_LEN - 1] = 0;
  429. namebuf[0] = 0;
  430. if (is_ksym_addr(addr)) {
  431. unsigned long pos;
  432. pos = get_symbol_pos(addr, symbolsize, offset);
  433. /* Grab name */
  434. kallsyms_expand_symbol(get_symbol_offset(pos),
  435. namebuf, KSYM_NAME_LEN);
  436. if (modname)
  437. *modname = NULL;
  438. return namebuf;
  439. }
  440. return NULL;
  441. }
  442. /* Look up a kernel symbol and return it in a text buffer. */
  443. static int __sprint_symbol(char *buffer, unsigned long address,
  444. int symbol_offset, int add_offset)
  445. {
  446. char *modname;
  447. const char *name;
  448. unsigned long offset, size;
  449. int len;
  450. if(kallsyms_addresses == NULL){
  451. buffer[0] = '@';
  452. buffer[1] = 0;
  453. return 2;
  454. }
  455. address += symbol_offset;
  456. name = kallsyms_lookup(address, &size, &offset, &modname, buffer);
  457. if (!name)
  458. return sprintf(buffer, "0x%x", (unsigned int)address);
  459. if (name != buffer)
  460. strcpy(buffer, name);
  461. len = strlen(buffer);
  462. offset -= symbol_offset;
  463. if (add_offset)
  464. len += sprintf(buffer + len, "+%#x/%#x", (unsigned int)offset, (unsigned int)size);
  465. if (modname)
  466. len += sprintf(buffer + len, " [%s]", modname);
  467. return len;
  468. }
  469. /**
  470. * sprint_symbol - Look up a kernel symbol and return it in a text buffer
  471. * @buffer: buffer to be stored
  472. * @address: address to lookup
  473. *
  474. * This function looks up a kernel symbol with @address and stores its name,
  475. * offset, size and module name to @buffer if possible. If no symbol was found,
  476. * just saves its @address as is.
  477. *
  478. * This function returns the number of bytes stored in @buffer.
  479. */
  480. int sprint_symbol(char *buffer, unsigned long address)
  481. {
  482. return __sprint_symbol(buffer, address, 0, 1);
  483. }
  484. /**
  485. * sprint_symbol_no_offset - Look up a kernel symbol and return it in a text buffer
  486. * @buffer: buffer to be stored
  487. * @address: address to lookup
  488. *
  489. * This function looks up a kernel symbol with @address and stores its name
  490. * and module name to @buffer if possible. If no symbol was found, just saves
  491. * its @address as is.
  492. *
  493. * This function returns the number of bytes stored in @buffer.
  494. */
  495. int sprint_symbol_no_offset(char *buffer, unsigned long address)
  496. {
  497. return __sprint_symbol(buffer, address, 0, 0);
  498. }
  499. /**
  500. * sprint_backtrace - Look up a backtrace symbol and return it in a text buffer
  501. * @buffer: buffer to be stored
  502. * @address: address to lookup
  503. *
  504. * This function is for stack backtrace and does the same thing as
  505. * sprint_symbol() but with modified/decreased @address. If there is a
  506. * tail-call to the function marked "noreturn", gcc optimized out code after
  507. * the call so that the stack-saved return address could point outside of the
  508. * caller. This function ensures that kallsyms will find the original caller
  509. * by decreasing @address.
  510. *
  511. * This function returns the number of bytes stored in @buffer.
  512. */
  513. int sprint_backtrace(char *buffer, unsigned long address)
  514. {
  515. return __sprint_symbol(buffer, address, -1, 1);
  516. }
  517. /* Look up a kernel symbol and print it to the kernel messages. */
  518. static void __print_symbol(const char *fmt, unsigned long address)
  519. {
  520. char buffer[KSYM_SYMBOL_LEN];
  521. sprint_symbol(buffer, address);
  522. printk(fmt, buffer);
  523. }
  524. void print_symbol(const char *fmt, unsigned long addr)
  525. {
  526. __print_symbol(fmt, (unsigned long)
  527. __builtin_extract_return_addr((void *)addr));
  528. }
  529. //TLV
  530. /*
  531. 0x0- 0x03 magic
  532. 0x04-0x07 len
  533. 0x08-0x0b checksum
  534. 0x0c-0x10 resrve
  535. 0x10---len tlv
  536. */
  537. #define TLV_MAGIC 0x59355935
  538. #define TYPE_KYMS_ADDR 0x01 //kallsyms_addresses
  539. #define TYPE_KYMS_NUM 0x02 //kallsyms_num_syms
  540. #define TYPE_KYMS_NAME 0x03 //kallsyms_names
  541. #define TYPE_KYMS_MARKERS 0x04 //kallsyms_markers
  542. #define TYPE_KYMS_TOKEN_TABEL 0x05 //kallsyms_token_table
  543. #define TYPE_KYMS_TOKEN_INDEX 0x06 //kallsyms_token_index
  544. #include <device.h>
  545. #if 0
  546. static unsigned long *kallsyms_addresses;
  547. static u8_t *kallsyms_names;
  548. static unsigned long kallsyms_num_syms;
  549. static u8_t *kallsyms_token_table;
  550. static u16_t *kallsyms_token_index;
  551. static unsigned long kallsyms_markers;
  552. #endif
  553. static int print_symb_init(const struct device *dev)
  554. {
  555. unsigned int *sym_addr;
  556. unsigned int len, off;
  557. if(sd_fmap("ksym.bin", (void**)&sym_addr, &len)){
  558. printk("ksym.bin not in sdfs\n");
  559. return 0;
  560. }
  561. printk("ksym=%p, len=0x%x\n", sym_addr, len);
  562. if(sym_addr[0] != TLV_MAGIC){
  563. printk("sym:magic=0x%x != 0x59355935\n", sym_addr[0]);
  564. return 0;
  565. }
  566. off = 0x10/4;
  567. printk("1.off=0x%x, type=%d, len=0x%x\n", off*4, sym_addr[off], sym_addr[off+1]);
  568. if(sym_addr[off] != TYPE_KYMS_ADDR){
  569. printk("KYMS_ADDR type=%d invalid\n", sym_addr[off]);
  570. return 0;
  571. }
  572. kallsyms_addresses = (long unsigned int *)&sym_addr[off+2];
  573. off += sym_addr[off+1]/4+2;
  574. printk("2. off=0x%x, type=%d, len=0x%x\n", off*4, sym_addr[off], sym_addr[off+1]);
  575. if(sym_addr[off] != TYPE_KYMS_NUM){
  576. printk("TYPE_KYMS_NUM type=%d invalid\n", sym_addr[off]);
  577. return 0;
  578. }
  579. kallsyms_num_syms = sym_addr[off+2];
  580. off += sym_addr[off+1]/4+2;
  581. printk("3. off=0x%x, type=%d, len=0x%x\n", off*4, sym_addr[off], sym_addr[off+1]);
  582. if(sym_addr[off] != TYPE_KYMS_NAME){
  583. printk("TYPE_KYMS_NAME type=%d invalid\n", sym_addr[off]);
  584. return 0;
  585. }
  586. kallsyms_names =(u8_t *) &sym_addr[off+2];
  587. off += sym_addr[off+1]/4+2;
  588. printk("4. off=0x%x, type=%d, len=0x%x\n", off*4, sym_addr[off], sym_addr[off+1]);
  589. if(sym_addr[off] != TYPE_KYMS_MARKERS){
  590. printk("TYPE_KYMS_MARKERS type=%d invalid\n", sym_addr[off]);
  591. return 0;
  592. }
  593. kallsyms_markers = (long unsigned int *)&sym_addr[off+2];
  594. off += sym_addr[off+1]/4+2;
  595. printk("5. off=0x%x, type=%d, len=0x%x\n", off*4, sym_addr[off], sym_addr[off+1]);
  596. if(sym_addr[off] != TYPE_KYMS_TOKEN_TABEL){
  597. printk("TYPE_KYMS_TOKEN_TABEL type=%d invalid\n", sym_addr[off]);
  598. return 0;
  599. }
  600. kallsyms_token_table = (u8_t *) &sym_addr[off+2];
  601. off += sym_addr[off+1]/4+2;
  602. printk("6. off=0x%x, type=%d, len=0x%x\n", off*4, sym_addr[off], sym_addr[off+1]);
  603. if(sym_addr[off] != TYPE_KYMS_TOKEN_INDEX){
  604. printk("TYPE_KYMS_TOKEN_TABEL type=%d invalid\n", sym_addr[off]);
  605. return 0;
  606. }
  607. kallsyms_token_index = (u16_t *) &sym_addr[off+2];
  608. off += sym_addr[off+1]/4+2;
  609. return 0;
  610. }
  611. SYS_INIT(print_symb_init, PRE_KERNEL_1, 81);
  612. #endif