diff options
author | Matt Fleming <matt@console-pimps.org> | 2009-10-09 18:20:54 -0400 |
---|---|---|
committer | Matt Fleming <matt@console-pimps.org> | 2009-10-11 11:41:44 -0400 |
commit | a6a2f2ad67506090e332f440457553c0ec011d68 (patch) | |
tree | cfe974784b68cc3c09ed76e449a31d536b2b4589 /arch/sh/kernel/dwarf.c | |
parent | c153a58e715e16ffcd6c4b3da7fc6b4a556bf917 (diff) |
sh: Teach the DWARF unwinder about modules
Pass a module's .eh_frame section to the DWARF unwinder at module load
time so that the section's FDEs and CIEs can be registered with the
DWARF unwinder. This allows us to unwind the stack through module code
when generating backtraces.
Signed-off-by: Matt Fleming <matt@console-pimps.org>
Diffstat (limited to 'arch/sh/kernel/dwarf.c')
-rw-r--r-- | arch/sh/kernel/dwarf.c | 135 |
1 files changed, 105 insertions, 30 deletions
diff --git a/arch/sh/kernel/dwarf.c b/arch/sh/kernel/dwarf.c index 577302f31e6a..981315c6d656 100644 --- a/arch/sh/kernel/dwarf.c +++ b/arch/sh/kernel/dwarf.c | |||
@@ -655,7 +655,7 @@ bail: | |||
655 | } | 655 | } |
656 | 656 | ||
657 | static int dwarf_parse_cie(void *entry, void *p, unsigned long len, | 657 | static int dwarf_parse_cie(void *entry, void *p, unsigned long len, |
658 | unsigned char *end) | 658 | unsigned char *end, struct module *mod) |
659 | { | 659 | { |
660 | struct dwarf_cie *cie; | 660 | struct dwarf_cie *cie; |
661 | unsigned long flags; | 661 | unsigned long flags; |
@@ -751,6 +751,8 @@ static int dwarf_parse_cie(void *entry, void *p, unsigned long len, | |||
751 | cie->initial_instructions = p; | 751 | cie->initial_instructions = p; |
752 | cie->instructions_end = end; | 752 | cie->instructions_end = end; |
753 | 753 | ||
754 | cie->mod = mod; | ||
755 | |||
754 | /* Add to list */ | 756 | /* Add to list */ |
755 | spin_lock_irqsave(&dwarf_cie_lock, flags); | 757 | spin_lock_irqsave(&dwarf_cie_lock, flags); |
756 | list_add_tail(&cie->link, &dwarf_cie_list); | 758 | list_add_tail(&cie->link, &dwarf_cie_list); |
@@ -761,7 +763,7 @@ static int dwarf_parse_cie(void *entry, void *p, unsigned long len, | |||
761 | 763 | ||
762 | static int dwarf_parse_fde(void *entry, u32 entry_type, | 764 | static int dwarf_parse_fde(void *entry, u32 entry_type, |
763 | void *start, unsigned long len, | 765 | void *start, unsigned long len, |
764 | unsigned char *end) | 766 | unsigned char *end, struct module *mod) |
765 | { | 767 | { |
766 | struct dwarf_fde *fde; | 768 | struct dwarf_fde *fde; |
767 | struct dwarf_cie *cie; | 769 | struct dwarf_cie *cie; |
@@ -810,6 +812,8 @@ static int dwarf_parse_fde(void *entry, u32 entry_type, | |||
810 | fde->instructions = p; | 812 | fde->instructions = p; |
811 | fde->end = end; | 813 | fde->end = end; |
812 | 814 | ||
815 | fde->mod = mod; | ||
816 | |||
813 | /* Add to list. */ | 817 | /* Add to list. */ |
814 | spin_lock_irqsave(&dwarf_fde_lock, flags); | 818 | spin_lock_irqsave(&dwarf_fde_lock, flags); |
815 | list_add_tail(&fde->link, &dwarf_fde_list); | 819 | list_add_tail(&fde->link, &dwarf_fde_list); |
@@ -875,15 +879,15 @@ static void dwarf_unwinder_cleanup(void) | |||
875 | } | 879 | } |
876 | 880 | ||
877 | /** | 881 | /** |
878 | * dwarf_unwinder_init - initialise the dwarf unwinder | 882 | * dwarf_parse_section - parse DWARF section |
883 | * @eh_frame_start: start address of the .eh_frame section | ||
884 | * @eh_frame_end: end address of the .eh_frame section | ||
885 | * @mod: the kernel module containing the .eh_frame section | ||
879 | * | 886 | * |
880 | * Build the data structures describing the .dwarf_frame section to | 887 | * Parse the information in a .eh_frame section. |
881 | * make it easier to lookup CIE and FDE entries. Because the | ||
882 | * .eh_frame section is packed as tightly as possible it is not | ||
883 | * easy to lookup the FDE for a given PC, so we build a list of FDE | ||
884 | * and CIE entries that make it easier. | ||
885 | */ | 888 | */ |
886 | static int __init dwarf_unwinder_init(void) | 889 | int dwarf_parse_section(char *eh_frame_start, char *eh_frame_end, |
890 | struct module *mod) | ||
887 | { | 891 | { |
888 | u32 entry_type; | 892 | u32 entry_type; |
889 | void *p, *entry; | 893 | void *p, *entry; |
@@ -891,29 +895,12 @@ static int __init dwarf_unwinder_init(void) | |||
891 | unsigned long len; | 895 | unsigned long len; |
892 | unsigned int c_entries, f_entries; | 896 | unsigned int c_entries, f_entries; |
893 | unsigned char *end; | 897 | unsigned char *end; |
894 | INIT_LIST_HEAD(&dwarf_cie_list); | ||
895 | INIT_LIST_HEAD(&dwarf_fde_list); | ||
896 | 898 | ||
897 | c_entries = 0; | 899 | c_entries = 0; |
898 | f_entries = 0; | 900 | f_entries = 0; |
899 | entry = &__start_eh_frame; | 901 | entry = eh_frame_start; |
900 | |||
901 | dwarf_frame_cachep = kmem_cache_create("dwarf_frames", | ||
902 | sizeof(struct dwarf_frame), 0, SLAB_PANIC, NULL); | ||
903 | dwarf_reg_cachep = kmem_cache_create("dwarf_regs", | ||
904 | sizeof(struct dwarf_reg), 0, SLAB_PANIC, NULL); | ||
905 | |||
906 | dwarf_frame_pool = mempool_create(DWARF_FRAME_MIN_REQ, | ||
907 | mempool_alloc_slab, | ||
908 | mempool_free_slab, | ||
909 | dwarf_frame_cachep); | ||
910 | 902 | ||
911 | dwarf_reg_pool = mempool_create(DWARF_REG_MIN_REQ, | 903 | while ((char *)entry < eh_frame_end) { |
912 | mempool_alloc_slab, | ||
913 | mempool_free_slab, | ||
914 | dwarf_reg_cachep); | ||
915 | |||
916 | while ((char *)entry < __stop_eh_frame) { | ||
917 | p = entry; | 904 | p = entry; |
918 | 905 | ||
919 | count = dwarf_entry_len(p, &len); | 906 | count = dwarf_entry_len(p, &len); |
@@ -925,6 +912,7 @@ static int __init dwarf_unwinder_init(void) | |||
925 | * entry and move to the next one because 'len' | 912 | * entry and move to the next one because 'len' |
926 | * tells us where our next entry is. | 913 | * tells us where our next entry is. |
927 | */ | 914 | */ |
915 | err = -EINVAL; | ||
928 | goto out; | 916 | goto out; |
929 | } else | 917 | } else |
930 | p += count; | 918 | p += count; |
@@ -936,13 +924,14 @@ static int __init dwarf_unwinder_init(void) | |||
936 | p += 4; | 924 | p += 4; |
937 | 925 | ||
938 | if (entry_type == DW_EH_FRAME_CIE) { | 926 | if (entry_type == DW_EH_FRAME_CIE) { |
939 | err = dwarf_parse_cie(entry, p, len, end); | 927 | err = dwarf_parse_cie(entry, p, len, end, mod); |
940 | if (err < 0) | 928 | if (err < 0) |
941 | goto out; | 929 | goto out; |
942 | else | 930 | else |
943 | c_entries++; | 931 | c_entries++; |
944 | } else { | 932 | } else { |
945 | err = dwarf_parse_fde(entry, entry_type, p, len, end); | 933 | err = dwarf_parse_fde(entry, entry_type, p, len, |
934 | end, mod); | ||
946 | if (err < 0) | 935 | if (err < 0) |
947 | goto out; | 936 | goto out; |
948 | else | 937 | else |
@@ -955,6 +944,92 @@ static int __init dwarf_unwinder_init(void) | |||
955 | printk(KERN_INFO "DWARF unwinder initialised: read %u CIEs, %u FDEs\n", | 944 | printk(KERN_INFO "DWARF unwinder initialised: read %u CIEs, %u FDEs\n", |
956 | c_entries, f_entries); | 945 | c_entries, f_entries); |
957 | 946 | ||
947 | return 0; | ||
948 | |||
949 | out: | ||
950 | return err; | ||
951 | } | ||
952 | |||
953 | /** | ||
954 | * dwarf_module_unload - remove FDE/CIEs associated with @mod | ||
955 | * @mod: the module that is being unloaded | ||
956 | * | ||
957 | * Remove any FDEs and CIEs from the global lists that came from | ||
958 | * @mod's .eh_frame section because @mod is being unloaded. | ||
959 | */ | ||
960 | void dwarf_module_unload(struct module *mod) | ||
961 | { | ||
962 | struct dwarf_fde *fde; | ||
963 | struct dwarf_cie *cie; | ||
964 | unsigned long flags; | ||
965 | |||
966 | spin_lock_irqsave(&dwarf_cie_lock, flags); | ||
967 | |||
968 | again_cie: | ||
969 | list_for_each_entry(cie, &dwarf_cie_list, link) { | ||
970 | if (cie->mod == mod) | ||
971 | break; | ||
972 | } | ||
973 | |||
974 | if (&cie->link != &dwarf_cie_list) { | ||
975 | list_del(&cie->link); | ||
976 | kfree(cie); | ||
977 | goto again_cie; | ||
978 | } | ||
979 | |||
980 | spin_unlock_irqrestore(&dwarf_cie_lock, flags); | ||
981 | |||
982 | spin_lock_irqsave(&dwarf_fde_lock, flags); | ||
983 | |||
984 | again_fde: | ||
985 | list_for_each_entry(fde, &dwarf_fde_list, link) { | ||
986 | if (fde->mod == mod) | ||
987 | break; | ||
988 | } | ||
989 | |||
990 | if (&fde->link != &dwarf_fde_list) { | ||
991 | list_del(&fde->link); | ||
992 | kfree(fde); | ||
993 | goto again_fde; | ||
994 | } | ||
995 | |||
996 | spin_unlock_irqrestore(&dwarf_fde_lock, flags); | ||
997 | } | ||
998 | |||
999 | /** | ||
1000 | * dwarf_unwinder_init - initialise the dwarf unwinder | ||
1001 | * | ||
1002 | * Build the data structures describing the .dwarf_frame section to | ||
1003 | * make it easier to lookup CIE and FDE entries. Because the | ||
1004 | * .eh_frame section is packed as tightly as possible it is not | ||
1005 | * easy to lookup the FDE for a given PC, so we build a list of FDE | ||
1006 | * and CIE entries that make it easier. | ||
1007 | */ | ||
1008 | static int __init dwarf_unwinder_init(void) | ||
1009 | { | ||
1010 | int err; | ||
1011 | INIT_LIST_HEAD(&dwarf_cie_list); | ||
1012 | INIT_LIST_HEAD(&dwarf_fde_list); | ||
1013 | |||
1014 | dwarf_frame_cachep = kmem_cache_create("dwarf_frames", | ||
1015 | sizeof(struct dwarf_frame), 0, SLAB_PANIC, NULL); | ||
1016 | dwarf_reg_cachep = kmem_cache_create("dwarf_regs", | ||
1017 | sizeof(struct dwarf_reg), 0, SLAB_PANIC, NULL); | ||
1018 | |||
1019 | dwarf_frame_pool = mempool_create(DWARF_FRAME_MIN_REQ, | ||
1020 | mempool_alloc_slab, | ||
1021 | mempool_free_slab, | ||
1022 | dwarf_frame_cachep); | ||
1023 | |||
1024 | dwarf_reg_pool = mempool_create(DWARF_REG_MIN_REQ, | ||
1025 | mempool_alloc_slab, | ||
1026 | mempool_free_slab, | ||
1027 | dwarf_reg_cachep); | ||
1028 | |||
1029 | err = dwarf_parse_section(__start_eh_frame, __stop_eh_frame, NULL); | ||
1030 | if (err) | ||
1031 | goto out; | ||
1032 | |||
958 | err = unwinder_register(&dwarf_unwinder); | 1033 | err = unwinder_register(&dwarf_unwinder); |
959 | if (err) | 1034 | if (err) |
960 | goto out; | 1035 | goto out; |