Print this page
10476 file(1) could be smatch clean
10366 ld(1) should support GNU-style linker sets
10581 ld(1) should know kernel modules are a thing

@@ -77,10 +77,11 @@
 #include <procfs.h>
 #include <sys/param.h>
 #include <sys/types.h>
 #include <sys/stat.h>
 #include <sys/elf.h>
+#include <sys/link.h>
 #include <elfcap.h>
 #include "file.h"
 #include "elf_read.h"
 
 extern const char *File;

@@ -139,11 +140,11 @@
         format = get_format();
 
         /* will convert only these types */
         if (type != ELF_T_EHDR && type != ELF_T_PHDR &&
             type != ELF_T_SHDR && type != ELF_T_WORD &&
-            type != ELF_T_CAP)
+            type != ELF_T_CAP && type != ELF_T_DYN)
                 return (ELF_READ_FAIL);
 
         src.d_buf = (Elf_Void *)hdr;
         src.d_type = type;
         src.d_version = version;

@@ -413,21 +414,17 @@
  *                      in Elf_Info with corresponding flags.
  */
 static int
 process_shdr(Elf_Info *EI)
 {
-        int             capn, mac;
-        int             i, j, idx;
-        FILE_ELF_OFF_T  cap_off;
-        FILE_ELF_SIZE_T csize;
+        int             mac;
+        int             i, idx;
         char            *strtab;
         size_t          strtab_sz;
-        Elf_Cap         Chdr;
+        uint64_t        j;
         Elf_Shdr        *shdr = &EI_Shdr;
 
-
-        csize = sizeof (Elf_Cap);
         mac = EI_Ehdr.e_machine;
 
         /* if there are no sections, return success anyway */
         if (EI_Ehdr.e_shoff == 0 && EI_Ehdr_shnum == 0)
                 return (ELF_READ_OKAY);

@@ -455,13 +452,19 @@
                 if (shdr->sh_type == SHT_NULL) {
                         idx--;
                         continue;
                 }
 
-                cap_off = shdr->sh_offset;
                 if (shdr->sh_type == SHT_SUNW_cap) {
                         char capstr[128];
+                        Elf_Cap         Chdr;
+                        FILE_ELF_OFF_T  cap_off;
+                        FILE_ELF_SIZE_T csize;
+                        uint64_t capn;
+
+                        cap_off = shdr->sh_offset;
+                        csize = sizeof (Elf_Cap);
 
                         if (shdr->sh_size == 0 || shdr->sh_entsize == 0) {
                                 (void) fprintf(stderr, ELF_ERR_ELFCAP1,
                                     File, EI->file);
                                 return (ELF_READ_FAIL);

@@ -469,12 +472,12 @@
                         capn = (shdr->sh_size / shdr->sh_entsize);
                         for (j = 0; j < capn; j++) {
                                 /*
                                  * read cap and xlate the values
                                  */
-                                if (pread64(EI->elffd, &Chdr, csize, cap_off)
-                                    != csize ||
+                                if ((pread64(EI->elffd, &Chdr, csize, cap_off)
+                                    != csize) ||
                                     file_xlatetom(ELF_T_CAP, (char *)&Chdr)
                                     == 0) {
                                         (void) fprintf(stderr, ELF_ERR_ELFCAP2,
                                             File, EI->file);
                                         return (ELF_READ_FAIL);

@@ -501,10 +504,43 @@
                                             sizeof (EI->cap_str));
 
                                 (void) strlcat(EI->cap_str, capstr,
                                     sizeof (EI->cap_str));
                         }
+                } else if (shdr->sh_type == SHT_DYNAMIC) {
+                        Elf_Dyn dyn;
+                        FILE_ELF_SIZE_T dsize;
+                        FILE_ELF_OFF_T doff;
+                        uint64_t dynn;
+
+                        doff = shdr->sh_offset;
+                        dsize = sizeof (Elf_Dyn);
+
+                        if (shdr->sh_size == 0 || shdr->sh_entsize == 0) {
+                                (void) fprintf(stderr, ELF_ERR_DYNAMIC1,
+                                    File, EI->file);
+                                return (ELF_READ_FAIL);
+                        }
+
+                        dynn = (shdr->sh_size / shdr->sh_entsize);
+                        for (j = 0; j < dynn; j++) {
+                                if (pread64(EI->elffd, &dyn, dsize, doff)
+                                    != dsize ||
+                                    file_xlatetom(ELF_T_DYN, (char *)&dyn)
+                                    == 0) {
+                                        (void) fprintf(stderr, ELF_ERR_DYNAMIC2,
+                                            File, EI->file);
+                                        return (ELF_READ_FAIL);
+                                }
+
+                                doff += dsize;
+
+                                if ((dyn.d_tag == DT_SUNW_KMOD) &&
+                                    (dyn.d_un.d_val == 1)) {
+                                        EI->kmod = B_TRUE;
+                                }
+                        }
                 }
 
                 /*
                  * Definition time:
                  *      - "not stripped" means that an executable file