b1a53d2
Upstream commits since 0.172:
b1a53d2
b1a53d2
  b4dced3 readelf: While printing .debug_loc make sure that next_off doesn't overflow.
b1a53d2
  bb11e36 libdw: Make __libdw_dieabbrev more robust on failure.
b1a53d2
  3647b5b readelf: Make sure print_form_data always consumes DW_FORM_strx[1234] data.
b1a53d2
  eaaf908 readelf: Check there are at least 4 bytes available for DWARF_FORM_block4.
b1a53d2
  0d93f49 libdw, readelf: Don't handle DW_FORM_data16 as expression block/location.
b1a53d2
  9495c26 libdw: aggregate_size check NULL result from get_type.
b1a53d2
  e636112 libdw: dwarf_peel_type break long chains/cycles.
b1a53d2
  5956b8a libdw: Break dwarf_aggregate_size recursion because of type cycles.
b1a53d2
  ca0d831 libelf: Don't return unaligned data returned from elf_getdata[_rawchunk].
b1a53d2
b1a53d2
diff --git a/libdw/dwarf_aggregate_size.c b/libdw/dwarf_aggregate_size.c
b1a53d2
index 6e50185..75105e4 100644
b1a53d2
--- a/libdw/dwarf_aggregate_size.c
b1a53d2
+++ b/libdw/dwarf_aggregate_size.c
b1a53d2
@@ -46,13 +46,17 @@ get_type (Dwarf_Die *die, Dwarf_Attribute *attr_mem, Dwarf_Die *type_mem)
b1a53d2
   return type;
b1a53d2
 }
b1a53d2
 
b1a53d2
+static int aggregate_size (Dwarf_Die *die, Dwarf_Word *size,
b1a53d2
+			   Dwarf_Die *type_mem, int depth);
b1a53d2
+
b1a53d2
 static int
b1a53d2
 array_size (Dwarf_Die *die, Dwarf_Word *size,
b1a53d2
-	    Dwarf_Attribute *attr_mem, Dwarf_Die *type_mem)
b1a53d2
+	    Dwarf_Attribute *attr_mem, int depth)
b1a53d2
 {
b1a53d2
   Dwarf_Word eltsize;
b1a53d2
-  if (INTUSE(dwarf_aggregate_size) (get_type (die, attr_mem, type_mem),
b1a53d2
-				    &eltsize) != 0)
b1a53d2
+  Dwarf_Die type_mem, aggregate_type_mem;
b1a53d2
+  if (aggregate_size (get_type (die, attr_mem, &type_mem), &eltsize,
b1a53d2
+		      &aggregate_type_mem, depth) != 0)
b1a53d2
       return -1;
b1a53d2
 
b1a53d2
   /* An array can have DW_TAG_subrange_type or DW_TAG_enumeration_type
b1a53d2
@@ -167,21 +171,30 @@ array_size (Dwarf_Die *die, Dwarf_Word *size,
b1a53d2
 }
b1a53d2
 
b1a53d2
 static int
b1a53d2
-aggregate_size (Dwarf_Die *die, Dwarf_Word *size, Dwarf_Die *type_mem)
b1a53d2
+aggregate_size (Dwarf_Die *die, Dwarf_Word *size,
b1a53d2
+		Dwarf_Die *type_mem, int depth)
b1a53d2
 {
b1a53d2
   Dwarf_Attribute attr_mem;
b1a53d2
 
b1a53d2
+/* Arrays of arrays of subrange types of arrays... Don't recurse too deep.  */
b1a53d2
+#define MAX_DEPTH 256
b1a53d2
+  if (die == NULL || depth++ >= MAX_DEPTH)
b1a53d2
+    return -1;
b1a53d2
+
b1a53d2
   if (INTUSE(dwarf_attr_integrate) (die, DW_AT_byte_size, &attr_mem) != NULL)
b1a53d2
     return INTUSE(dwarf_formudata) (&attr_mem, size);
b1a53d2
 
b1a53d2
   switch (INTUSE(dwarf_tag) (die))
b1a53d2
     {
b1a53d2
     case DW_TAG_subrange_type:
b1a53d2
-      return aggregate_size (get_type (die, &attr_mem, type_mem),
b1a53d2
-			     size, type_mem); /* Tail call.  */
b1a53d2
+      {
b1a53d2
+	Dwarf_Die aggregate_type_mem;
b1a53d2
+	return aggregate_size (get_type (die, &attr_mem, type_mem),
b1a53d2
+			       size, &aggregate_type_mem, depth);
b1a53d2
+      }
b1a53d2
 
b1a53d2
     case DW_TAG_array_type:
b1a53d2
-      return array_size (die, size, &attr_mem, type_mem);
b1a53d2
+      return array_size (die, size, &attr_mem, depth);
b1a53d2
 
b1a53d2
     /* Assume references and pointers have pointer size if not given an
b1a53d2
        explicit DW_AT_byte_size.  */
b1a53d2
@@ -204,7 +217,7 @@ dwarf_aggregate_size (Dwarf_Die *die, Dwarf_Word *size)
b1a53d2
   if (INTUSE (dwarf_peel_type) (die, &die_mem) != 0)
b1a53d2
     return -1;
b1a53d2
 
b1a53d2
-  return aggregate_size (&die_mem, size, &type_mem);
b1a53d2
+  return aggregate_size (&die_mem, size, &type_mem, 0);
b1a53d2
 }
b1a53d2
 INTDEF (dwarf_aggregate_size)
b1a53d2
 OLD_VERSION (dwarf_aggregate_size, ELFUTILS_0.144)
b1a53d2
diff --git a/libdw/dwarf_getlocation.c b/libdw/dwarf_getlocation.c
b1a53d2
index 7f294fe..fc59a2a 100644
b1a53d2
--- a/libdw/dwarf_getlocation.c
b1a53d2
+++ b/libdw/dwarf_getlocation.c
b1a53d2
@@ -174,6 +174,8 @@ check_constant_offset (Dwarf_Attribute *attr,
b1a53d2
     default:
b1a53d2
       return 1;
b1a53d2
 
b1a53d2
+      /* Note, we don't regard DW_FORM_data16 as a constant form,
b1a53d2
+	 even though technically it is according to the standard.  */
b1a53d2
     case DW_FORM_data1:
b1a53d2
     case DW_FORM_data2:
b1a53d2
     case DW_FORM_data4:
b1a53d2
@@ -665,7 +667,13 @@ dwarf_getlocation (Dwarf_Attribute *attr, Dwarf_Op **llbuf, size_t *listlen)
b1a53d2
   if (result != 1)
b1a53d2
     return result;
b1a53d2
 
b1a53d2
-  /* If it has a block form, it's a single location expression.  */
b1a53d2
+  /* If it has a block form, it's a single location expression.
b1a53d2
+     Except for DW_FORM_data16, which is a 128bit constant.  */
b1a53d2
+  if (attr->form == DW_FORM_data16)
b1a53d2
+    {
b1a53d2
+      __libdw_seterrno (DWARF_E_NO_BLOCK);
b1a53d2
+      return -1;
b1a53d2
+    }
b1a53d2
   Dwarf_Block block;
b1a53d2
   if (INTUSE(dwarf_formblock) (attr, &block) != 0)
b1a53d2
     return -1;
b1a53d2
@@ -863,9 +871,11 @@ dwarf_getlocation_addr (Dwarf_Attribute *attr, Dwarf_Addr address,
b1a53d2
   if (llbufs == NULL)
b1a53d2
     maxlocs = SIZE_MAX;
b1a53d2
 
b1a53d2
-  /* If it has a block form, it's a single location expression.  */
b1a53d2
+  /* If it has a block form, it's a single location expression.
b1a53d2
+     Except for DW_FORM_data16, which is a 128bit constant.  */
b1a53d2
   Dwarf_Block block;
b1a53d2
-  if (INTUSE(dwarf_formblock) (attr, &block) == 0)
b1a53d2
+  if (attr->form != DW_FORM_data16
b1a53d2
+      && INTUSE(dwarf_formblock) (attr, &block) == 0)
b1a53d2
     {
b1a53d2
       if (maxlocs == 0)
b1a53d2
 	return 0;
b1a53d2
@@ -876,11 +886,14 @@ dwarf_getlocation_addr (Dwarf_Attribute *attr, Dwarf_Addr address,
b1a53d2
       return listlens[0] == 0 ? 0 : 1;
b1a53d2
     }
b1a53d2
 
b1a53d2
-  int error = INTUSE(dwarf_errno) ();
b1a53d2
-  if (unlikely (error != DWARF_E_NO_BLOCK))
b1a53d2
+  if (attr->form != DW_FORM_data16)
b1a53d2
     {
b1a53d2
-      __libdw_seterrno (error);
b1a53d2
-      return -1;
b1a53d2
+      int error = INTUSE(dwarf_errno) ();
b1a53d2
+      if (unlikely (error != DWARF_E_NO_BLOCK))
b1a53d2
+	{
b1a53d2
+	  __libdw_seterrno (error);
b1a53d2
+	  return -1;
b1a53d2
+	}
b1a53d2
     }
b1a53d2
 
b1a53d2
   int result = check_constant_offset (attr, &llbufs[0], &listlens[0]);
b1a53d2
@@ -938,9 +951,11 @@ dwarf_getlocations (Dwarf_Attribute *attr, ptrdiff_t offset, Dwarf_Addr *basep,
b1a53d2
 
b1a53d2
   if (offset == 0)
b1a53d2
     {
b1a53d2
-      /* If it has a block form, it's a single location expression.  */
b1a53d2
+      /* If it has a block form, it's a single location expression.
b1a53d2
+	 Except for DW_FORM_data16, which is a 128bit constant.  */
b1a53d2
       Dwarf_Block block;
b1a53d2
-      if (INTUSE(dwarf_formblock) (attr, &block) == 0)
b1a53d2
+      if (attr->form != DW_FORM_data16
b1a53d2
+	  && INTUSE(dwarf_formblock) (attr, &block) == 0)
b1a53d2
 	{
b1a53d2
 	  if (getlocation (attr->cu, &block, expr, exprlen,
b1a53d2
 			   cu_sec_idx (attr->cu)) != 0)
b1a53d2
@@ -952,11 +967,14 @@ dwarf_getlocations (Dwarf_Attribute *attr, ptrdiff_t offset, Dwarf_Addr *basep,
b1a53d2
 	  return 1;
b1a53d2
 	}
b1a53d2
 
b1a53d2
-      int error = INTUSE(dwarf_errno) ();
b1a53d2
-      if (unlikely (error != DWARF_E_NO_BLOCK))
b1a53d2
+      if (attr->form != DW_FORM_data16)
b1a53d2
 	{
b1a53d2
-	  __libdw_seterrno (error);
b1a53d2
-	  return -1;
b1a53d2
+	  int error = INTUSE(dwarf_errno) ();
b1a53d2
+	  if (unlikely (error != DWARF_E_NO_BLOCK))
b1a53d2
+	    {
b1a53d2
+	      __libdw_seterrno (error);
b1a53d2
+	      return -1;
b1a53d2
+	    }
b1a53d2
 	}
b1a53d2
 
b1a53d2
       int result = check_constant_offset (attr, expr, exprlen);
b1a53d2
diff --git a/libdw/dwarf_peel_type.c b/libdw/dwarf_peel_type.c
b1a53d2
index 6bbfd42..59fc6f1 100644
b1a53d2
--- a/libdw/dwarf_peel_type.c
b1a53d2
+++ b/libdw/dwarf_peel_type.c
b1a53d2
@@ -46,14 +46,19 @@ dwarf_peel_type (Dwarf_Die *die, Dwarf_Die *result)
b1a53d2
 
b1a53d2
   *result = *die;
b1a53d2
   tag = INTUSE (dwarf_tag) (result);
b1a53d2
-  while (tag == DW_TAG_typedef
b1a53d2
-	 || tag == DW_TAG_const_type
b1a53d2
-	 || tag == DW_TAG_volatile_type
b1a53d2
-	 || tag == DW_TAG_restrict_type
b1a53d2
-	 || tag == DW_TAG_atomic_type
b1a53d2
-	 || tag == DW_TAG_immutable_type
b1a53d2
-	 || tag == DW_TAG_packed_type
b1a53d2
-	 || tag == DW_TAG_shared_type)
b1a53d2
+
b1a53d2
+/* Stack 8 of all these modifiers, after that it gets silly.  */
b1a53d2
+#define MAX_DEPTH (8 * 8)
b1a53d2
+  int max_depth = MAX_DEPTH;
b1a53d2
+  while ((tag == DW_TAG_typedef
b1a53d2
+	  || tag == DW_TAG_const_type
b1a53d2
+	  || tag == DW_TAG_volatile_type
b1a53d2
+	  || tag == DW_TAG_restrict_type
b1a53d2
+	  || tag == DW_TAG_atomic_type
b1a53d2
+	  || tag == DW_TAG_immutable_type
b1a53d2
+	  || tag == DW_TAG_packed_type
b1a53d2
+	  || tag == DW_TAG_shared_type)
b1a53d2
+	&& max_depth-- > 0)
b1a53d2
     {
b1a53d2
       Dwarf_Attribute attr_mem;
b1a53d2
       Dwarf_Attribute *attr = INTUSE (dwarf_attr_integrate) (result, DW_AT_type,
b1a53d2
@@ -67,7 +72,7 @@ dwarf_peel_type (Dwarf_Die *die, Dwarf_Die *result)
b1a53d2
       tag = INTUSE (dwarf_tag) (result);
b1a53d2
     }
b1a53d2
 
b1a53d2
-  if (tag == DW_TAG_invalid)
b1a53d2
+  if (tag == DW_TAG_invalid || max_depth <= 0)
b1a53d2
     return -1;
b1a53d2
 
b1a53d2
   return 0;
b1a53d2
diff --git a/libdw/libdwP.h b/libdw/libdwP.h
b1a53d2
index 3d8e145..eebb7d1 100644
b1a53d2
--- a/libdw/libdwP.h
b1a53d2
+++ b/libdw/libdwP.h
b1a53d2
@@ -653,8 +653,9 @@ __libdw_dieabbrev (Dwarf_Die *die, const unsigned char **readp)
b1a53d2
       /* Get the abbreviation code.  */
b1a53d2
       unsigned int code;
b1a53d2
       const unsigned char *addr = die->addr;
b1a53d2
-      if (die->cu == NULL || addr >= (const unsigned char *) die->cu->endp)
b1a53d2
-	return DWARF_END_ABBREV;
b1a53d2
+      if (unlikely (die->cu == NULL
b1a53d2
+		    || addr >= (const unsigned char *) die->cu->endp))
b1a53d2
+	return die->abbrev = DWARF_END_ABBREV;
b1a53d2
       get_uleb128 (code, addr, die->cu->endp);
b1a53d2
       if (readp != NULL)
b1a53d2
 	*readp = addr;
b1a53d2
diff --git a/libdw/memory-access.h b/libdw/memory-access.h
b1a53d2
index 22918cb..a39ad6d 100644
b1a53d2
--- a/libdw/memory-access.h
b1a53d2
+++ b/libdw/memory-access.h
b1a53d2
@@ -362,6 +362,11 @@ read_3ubyte_unaligned (Dwarf *dbg, const unsigned char *p)
b1a53d2
 }
b1a53d2
 
b1a53d2
 
b1a53d2
+#define read_3ubyte_unaligned_inc(Dbg, Addr) \
b1a53d2
+  ({ uint32_t t_ = read_2ubyte_unaligned (Dbg, Addr);			      \
b1a53d2
+     Addr = (__typeof (Addr)) (((uintptr_t) (Addr)) + 3);		      \
b1a53d2
+     t_; })
b1a53d2
+
b1a53d2
 #define read_addr_unaligned_inc(Nbytes, Dbg, Addr)			\
b1a53d2
   (assert ((Nbytes) == 4 || (Nbytes) == 8),				\
b1a53d2
     ((Nbytes) == 4 ? read_4ubyte_unaligned_inc (Dbg, Addr)		\
b1a53d2
diff --git a/libelf/elf_getdata.c b/libelf/elf_getdata.c
b1a53d2
index 97c503b..278dfa8 100644
b1a53d2
--- a/libelf/elf_getdata.c
b1a53d2
+++ b/libelf/elf_getdata.c
b1a53d2
@@ -76,7 +76,6 @@ static const Elf_Type shtype_map[EV_NUM - 1][TYPEIDX (SHT_HISUNW) + 1] =
b1a53d2
     }
b1a53d2
   };
b1a53d2
 
b1a53d2
-#if !ALLOW_UNALIGNED
b1a53d2
 /* Associate libelf types with their internal alignment requirements.  */
b1a53d2
 const uint_fast8_t __libelf_type_aligns[EV_NUM - 1][ELFCLASSNUM - 1][ELF_T_NUM] =
b1a53d2
   {
b1a53d2
@@ -115,7 +114,6 @@ const uint_fast8_t __libelf_type_aligns[EV_NUM - 1][ELFCLASSNUM - 1][ELF_T_NUM]
b1a53d2
     }
b1a53d2
 # undef TYPE_ALIGNS
b1a53d2
   };
b1a53d2
-#endif
b1a53d2
 
b1a53d2
 
b1a53d2
 Elf_Type
b1a53d2
@@ -173,8 +171,7 @@ convert_data (Elf_Scn *scn, int version __attribute__ ((unused)), int eclass,
b1a53d2
       /* Make sure the source is correctly aligned for the conversion
b1a53d2
 	 function to directly access the data elements.  */
b1a53d2
       char *rawdata_source;
b1a53d2
-      if (ALLOW_UNALIGNED ||
b1a53d2
-	  ((((size_t) (char *) scn->rawdata_base)) & (align - 1)) == 0)
b1a53d2
+      if (((((size_t) (char *) scn->rawdata_base)) & (align - 1)) == 0)
b1a53d2
 	rawdata_source = scn->rawdata_base;
b1a53d2
       else
b1a53d2
 	{
b1a53d2
diff --git a/libelf/elf_getdata_rawchunk.c b/libelf/elf_getdata_rawchunk.c
b1a53d2
index 31b2fe7..d0c0b75 100644
b1a53d2
--- a/libelf/elf_getdata_rawchunk.c
b1a53d2
+++ b/libelf/elf_getdata_rawchunk.c
b1a53d2
@@ -80,8 +80,7 @@ elf_getdata_rawchunk (Elf *elf, off_t offset, size_t size, Elf_Type type)
b1a53d2
     {
b1a53d2
     /* If the file is mmap'ed we can use it directly, if aligned for type.  */
b1a53d2
       char *rawdata = elf->map_address + elf->start_offset + offset;
b1a53d2
-      if (ALLOW_UNALIGNED ||
b1a53d2
-	  ((uintptr_t) rawdata & (align - 1)) == 0)
b1a53d2
+      if (((uintptr_t) rawdata & (align - 1)) == 0)
b1a53d2
 	rawchunk = rawdata;
b1a53d2
       else
b1a53d2
 	{
b1a53d2
diff --git a/libelf/libelfP.h b/libelf/libelfP.h
b1a53d2
index ca805ac..ed216c8 100644
b1a53d2
--- a/libelf/libelfP.h
b1a53d2
+++ b/libelf/libelfP.h
b1a53d2
@@ -443,15 +443,11 @@ extern int __libelf_version_initialized attribute_hidden;
b1a53d2
 # define LIBELF_EV_IDX	(__libelf_version - 1)
b1a53d2
 #endif
b1a53d2
 
b1a53d2
-#if !ALLOW_UNALIGNED
b1a53d2
 /* Array with alignment requirements of the internal types indexed by ELF
b1a53d2
    version, binary class, and type. */
b1a53d2
 extern const uint_fast8_t __libelf_type_aligns[EV_NUM - 1][ELFCLASSNUM - 1][ELF_T_NUM] attribute_hidden;
b1a53d2
 # define __libelf_type_align(class, type)	\
b1a53d2
     (__libelf_type_aligns[LIBELF_EV_IDX][class - 1][type] ?: 1)
b1a53d2
-#else
b1a53d2
-# define __libelf_type_align(class, type)	1
b1a53d2
-#endif
b1a53d2
 
b1a53d2
 /* Given an Elf handle and a section type returns the Elf_Data d_type.
b1a53d2
    Should not be called when SHF_COMPRESSED is set, the d_type should
b1a53d2
diff --git a/src/readelf.c b/src/readelf.c
b1a53d2
index f185897..313f940 100644
b1a53d2
--- a/src/readelf.c
b1a53d2
+++ b/src/readelf.c
b1a53d2
@@ -7403,11 +7403,16 @@ attr_callback (Dwarf_Attribute *attrp, void *arg)
b1a53d2
 	case DW_AT_GNU_call_site_data_value:
b1a53d2
 	case DW_AT_GNU_call_site_target:
b1a53d2
 	case DW_AT_GNU_call_site_target_clobbered:
b1a53d2
-	  putchar ('\n');
b1a53d2
-	  print_ops (cbargs->dwflmod, cbargs->dbg,
b1a53d2
-		     12 + level * 2, 12 + level * 2,
b1a53d2
-		     cbargs->version, cbargs->addrsize, cbargs->offset_size,
b1a53d2
-		     attrp->cu, block.length, block.data);
b1a53d2
+	  if (form != DW_FORM_data16)
b1a53d2
+	    {
b1a53d2
+	      putchar ('\n');
b1a53d2
+	      print_ops (cbargs->dwflmod, cbargs->dbg,
b1a53d2
+			 12 + level * 2, 12 + level * 2,
b1a53d2
+			 cbargs->version, cbargs->addrsize, cbargs->offset_size,
b1a53d2
+			 attrp->cu, block.length, block.data);
b1a53d2
+	    }
b1a53d2
+	  else
b1a53d2
+	    print_block (block.length, block.data);
b1a53d2
 	  break;
b1a53d2
 	}
b1a53d2
       break;
b1a53d2
@@ -7907,7 +7912,7 @@ print_form_data (Dwarf *dbg, int form, const unsigned char *readp,
b1a53d2
       break;
b1a53d2
 
b1a53d2
     case DW_FORM_block4:
b1a53d2
-      if (readendp - readp < 2)
b1a53d2
+      if (readendp - readp < 4)
b1a53d2
 	goto invalid_data;
b1a53d2
       val = read_4ubyte_unaligned_inc (dbg, readp);
b1a53d2
       if ((size_t) (readendp - readp) < val)
b1a53d2
@@ -7994,9 +7999,9 @@ print_form_data (Dwarf *dbg, int form, const unsigned char *readp,
b1a53d2
 	    {
b1a53d2
 	      Dwarf_Off idx;
b1a53d2
 	      if (offset_len == 8)
b1a53d2
-		idx = read_8ubyte_unaligned_inc (dbg, strreadp);
b1a53d2
+		idx = read_8ubyte_unaligned (dbg, strreadp);
b1a53d2
 	      else
b1a53d2
-		idx = read_4ubyte_unaligned_inc (dbg, strreadp);
b1a53d2
+		idx = read_4ubyte_unaligned (dbg, strreadp);
b1a53d2
 
b1a53d2
 	      data = dbg->sectiondata[IDX_debug_str];
b1a53d2
 	      if (data == NULL || idx >= data->d_size
b1a53d2
@@ -8013,25 +8018,25 @@ print_form_data (Dwarf *dbg, int form, const unsigned char *readp,
b1a53d2
     case DW_FORM_strx1:
b1a53d2
       if (readendp - readp < 1)
b1a53d2
 	goto invalid_data;
b1a53d2
-      val = *readp;
b1a53d2
+      val = *readp++;
b1a53d2
       goto strx_val;
b1a53d2
 
b1a53d2
     case DW_FORM_strx2:
b1a53d2
       if (readendp - readp < 2)
b1a53d2
 	goto invalid_data;
b1a53d2
-      val = read_2ubyte_unaligned (dbg, readp);
b1a53d2
+      val = read_2ubyte_unaligned_inc (dbg, readp);
b1a53d2
       goto strx_val;
b1a53d2
 
b1a53d2
     case DW_FORM_strx3:
b1a53d2
       if (readendp - readp < 3)
b1a53d2
 	goto invalid_data;
b1a53d2
-      val = read_3ubyte_unaligned (dbg, readp);
b1a53d2
+      val = read_3ubyte_unaligned_inc (dbg, readp);
b1a53d2
       goto strx_val;
b1a53d2
 
b1a53d2
     case DW_FORM_strx4:
b1a53d2
       if (readendp - readp < 4)
b1a53d2
 	goto invalid_data;
b1a53d2
-      val = read_4ubyte_unaligned (dbg, readp);
b1a53d2
+      val = read_4ubyte_unaligned_inc (dbg, readp);
b1a53d2
       goto strx_val;
b1a53d2
 
b1a53d2
     default:
b1a53d2
@@ -9230,7 +9235,9 @@ print_debug_loc_section (Dwfl_Module *dwflmod,
b1a53d2
 						    listptr_idx);
b1a53d2
 	  const unsigned char *locp = readp;
b1a53d2
 	  const unsigned char *locendp;
b1a53d2
-	  if (next_off == 0)
b1a53d2
+	  if (next_off == 0
b1a53d2
+	      || next_off > (size_t) (endp
b1a53d2
+				      - (const unsigned char *) data->d_buf))
b1a53d2
 	    locendp = endp;
b1a53d2
 	  else
b1a53d2
 	    locendp = (const unsigned char *) data->d_buf + next_off;