#define __always_unused                __attribute__((unused))
 #define __mode(x)               __attribute__((mode(x)))
 
-#ifdef CONFIG_STACK_VALIDATION
-#define annotate_unreachable() ({                                      \
-       asm("%c0:\n\t"                                                  \
-           ".pushsection .discard.unreachable\n\t"                     \
-           ".long %c0b - .\n\t"                                        \
-           ".popsection\n\t" : : "i" (__LINE__));                      \
-})
-#define ASM_UNREACHABLE                                                        \
-       "999:\n\t"                                                      \
-       ".pushsection .discard.unreachable\n\t"                         \
-       ".long 999b - .\n\t"                                            \
-       ".popsection\n\t"
-#else
-#define annotate_unreachable()
-#endif
-
 /* gcc version specific checks */
 
 #if GCC_VERSION < 30200
 
 #endif
 
 /* Unreachable code */
+#ifdef CONFIG_STACK_VALIDATION
+#define annotate_reachable() ({                                                \
+       asm("%c0:\n\t"                                                  \
+           ".pushsection .discard.reachable\n\t"                       \
+           ".long %c0b - .\n\t"                                        \
+           ".popsection\n\t" : : "i" (__LINE__));                      \
+})
+#define annotate_unreachable() ({                                      \
+       asm("%c0:\n\t"                                                  \
+           ".pushsection .discard.unreachable\n\t"                     \
+           ".long %c0b - .\n\t"                                        \
+           ".popsection\n\t" : : "i" (__LINE__));                      \
+})
+#define ASM_UNREACHABLE                                                        \
+       "999:\n\t"                                                      \
+       ".pushsection .discard.unreachable\n\t"                         \
+       ".long 999b - .\n\t"                                            \
+       ".popsection\n\t"
+#else
+#define annotate_reachable()
+#define annotate_unreachable()
+#endif
+
 #ifndef ASM_UNREACHABLE
 # define ASM_UNREACHABLE
 #endif
 #ifndef unreachable
-# define unreachable() do { } while (1)
+# define unreachable() do { annotate_reachable(); do { } while (1); } while (0)
 #endif
 
 /*
 
 
        case 0x0f:
 
-               if (op2 >= 0x80 && op2 <= 0x8f)
+               if (op2 >= 0x80 && op2 <= 0x8f) {
+
                        *type = INSN_JUMP_CONDITIONAL;
-               else if (op2 == 0x05 || op2 == 0x07 || op2 == 0x34 ||
-                        op2 == 0x35)
+
+               } else if (op2 == 0x05 || op2 == 0x07 || op2 == 0x34 ||
+                          op2 == 0x35) {
 
                        /* sysenter, sysret */
                        *type = INSN_CONTEXT_SWITCH;
 
-               else if (op2 == 0x0d || op2 == 0x1f)
+               } else if (op2 == 0x0b || op2 == 0xb9) {
+
+                       /* ud2 */
+                       *type = INSN_BUG;
+
+               } else if (op2 == 0x0d || op2 == 0x1f) {
 
                        /* nopl/nopw */
                        *type = INSN_NOP;
 
-               else if (op2 == 0xa0 || op2 == 0xa8) {
+               } else if (op2 == 0xa0 || op2 == 0xa8) {
 
                        /* push fs/gs */
                        *type = INSN_STACK;
 
 }
 
 /*
- * Find all uses of the unreachable() macro, which are code path dead ends.
+ * Mark "ud2" instructions and manually annotated dead ends.
  */
 static int add_dead_ends(struct objtool_file *file)
 {
        struct instruction *insn;
        bool found;
 
+       /*
+        * By default, "ud2" is a dead end unless otherwise annotated, because
+        * GCC 7 inserts it for certain divide-by-zero cases.
+        */
+       for_each_insn(file, insn)
+               if (insn->type == INSN_BUG)
+                       insn->dead_end = true;
+
+       /*
+        * Check for manually annotated dead ends.
+        */
        sec = find_section_by_name(file->elf, ".rela.discard.unreachable");
        if (!sec)
-               return 0;
+               goto reachable;
 
        list_for_each_entry(rela, &sec->rela_list, list) {
                if (rela->sym->type != STT_SECTION) {
                insn->dead_end = true;
        }
 
+reachable:
+       /*
+        * These manually annotated reachable checks are needed for GCC 4.4,
+        * where the Linux unreachable() macro isn't supported.  In that case
+        * GCC doesn't know the "ud2" is fatal, so it generates code as if it's
+        * not a dead end.
+        */
+       sec = find_section_by_name(file->elf, ".rela.discard.reachable");
+       if (!sec)
+               return 0;
+
+       list_for_each_entry(rela, &sec->rela_list, list) {
+               if (rela->sym->type != STT_SECTION) {
+                       WARN("unexpected relocation symbol type in %s", sec->name);
+                       return -1;
+               }
+               insn = find_insn(file, rela->sym->sec, rela->addend);
+               if (insn)
+                       insn = list_prev_entry(insn, list);
+               else if (rela->addend == rela->sym->sec->len) {
+                       found = false;
+                       list_for_each_entry_reverse(insn, &file->insn_list, list) {
+                               if (insn->sec == rela->sym->sec) {
+                                       found = true;
+                                       break;
+                               }
+                       }
+
+                       if (!found) {
+                               WARN("can't find reachable insn at %s+0x%x",
+                                    rela->sym->sec->name, rela->addend);
+                               return -1;
+                       }
+               } else {
+                       WARN("can't find reachable insn at %s+0x%x",
+                            rela->sym->sec->name, rela->addend);
+                       return -1;
+               }
+
+               insn->dead_end = false;
+       }
+
        return 0;
 }