Blame src/arbits/pe_archive_meta.c

906422
/***************************************************************/
906422
/*  perk: PE Resource Kit                                      */
906422
/*  Copyright (C) 2015--2025  SysDeer Technologies, LLC        */
906422
/*  Released under GPLv2 and GPLv3; see COPYING.PERK.          */
906422
/***************************************************************/
906422
906422
#include <stdint.h>
906422
#include <stddef.h>
906422
#include <stdlib.h>
906422
#include <string.h>
906422
#include <sys/mman.h>
906422
906422
#include <perk/perk.h>
906422
#include <perk/perk_arbits.h>
906422
#include "perk_ar_impl.h"
906422
#include "perk_driver_impl.h"
906422
#include "perk_errinfo_impl.h"
906422
#include "perk_visibility_impl.h"
906422
906422
/* transient header info vector */
906422
struct ar_header_info {
906422
	struct ar_raw_file_header * phdr;
906422
	uint32_t                    attr;
906422
};
906422
906422
static const char ar_signature[] = AR_SIGNATURE;
906422
906422
static int pe_ar_free_archive_meta_impl(struct pe_archive_meta_impl * meta, int ret)
906422
{
906422
	if (meta) {
906422
		if (meta->armaps.armap_symrefs_32)
906422
			free(meta->armaps.armap_symrefs_32);
906422
906422
		if (meta->armaps.armap_symrefs_64)
906422
			free(meta->armaps.armap_symrefs_64);
906422
906422
		if (meta->hdrinfov)
906422
			free(meta->hdrinfov);
906422
906422
		if (meta->namestrs)
906422
			free(meta->namestrs);
906422
906422
		if (meta->syminfo)
906422
			free(meta->syminfo);
906422
906422
		if (meta->syminfv)
906422
			free(meta->syminfv);
906422
906422
		if (meta->memberv)
906422
			free(meta->memberv);
906422
906422
		if (meta->offsetv)
906422
			free(meta->offsetv);
906422
906422
		if (meta->members)
906422
			free(meta->members);
906422
906422
		if (meta->symstrv)
906422
			free(meta->symstrv);
906422
906422
		if (meta->mapstrv)
906422
			free(meta->mapstrv);
906422
906422
		if (meta->nminfo)
906422
			(void)0;
906422
906422
		free(meta);
906422
	}
906422
906422
	return ret;
906422
}
906422
906422
906422
static int pe_ar_read_octal(const char * mark, int len, uint32_t * dec)
906422
{
906422
	int       i;
906422
	uint64_t  res;
906422
906422
	for (; len && (mark[len-1]==AR_DEC_PADDING); )
906422
		len--;
906422
906422
	for (i=0,res=0; i
906422
		if ((mark[i] >= '0') && (mark[i] <= '7')) {
906422
			res *= 8;
906422
			res += (mark[i] - '0');
906422
		} else {
906422
			return -1;
906422
		}
906422
	}
906422
906422
	*dec = res;
906422
906422
	return 0;
906422
}
906422
906422
static int pe_ar_read_decimal_64(const char * mark, int len, uint64_t * dec)
906422
{
906422
	int       i;
906422
	uint64_t  res;
906422
906422
	for (; len && (mark[len-1]==AR_DEC_PADDING); )
906422
		len--;
906422
906422
	for (i=0,res=0; i
906422
		if ((mark[i] >= '0') && (mark[i] <= '9')) {
906422
			res *= 10;
906422
			res += (mark[i] - '0');
906422
		} else {
906422
			return -1;
906422
		}
906422
	}
906422
906422
	*dec = res;
906422
906422
	return 0;
906422
}
906422
906422
static int pe_ar_read_decimal_32(const char * mark, int len, uint32_t * dec)
906422
{
906422
	uint64_t res;
906422
906422
	if (pe_ar_read_decimal_64(mark,len,&res) < 0)
906422
		return -1;
906422
906422
	*dec = res;
906422
906422
	return 0;
906422
}
906422
906422
static uint32_t pe_ar_get_member_attr(struct ar_meta_member_info * m)
906422
{
906422
	const char *            hdrname;
906422
	uint32_t                hdrattr;
906422
	const char *            data;
906422
	const char *            data_cap;
906422
	const unsigned char *   udata;
906422
	unsigned char           uch;
906422
	const size_t            siglen = sizeof(struct ar_raw_signature);
906422
906422
	hdrname  = m->ar_file_header.ar_member_name;
906422
	hdrattr  = m->ar_file_header.ar_header_attr;
906422
906422
	data     = m->ar_object_data;
906422
	data_cap = &data[m->ar_file_header.ar_file_size];
906422
906422
	if (hdrattr & AR_HEADER_ATTR_SYSV) {
906422
		/* long names member? */
906422
		if ((hdrname[0] == '/') && (hdrname[1] == '/'))
906422
			return AR_MEMBER_ATTR_NAMESTRS;
906422
906422
		/* mips 64-bit armap member? */
906422
		else if (!strncmp(hdrname,"/SYM64/",7))
906422
			return AR_MEMBER_ATTR_ARMAP;
906422
906422
		/* armap member? */
906422
		else if (hdrname[0] == '/' && (hdrname[1] == '\0'))
906422
			return AR_MEMBER_ATTR_ARMAP;
906422
906422
		/* nested archive? */
906422
		else if (m->ar_file_header.ar_file_size >= siglen)
906422
			if (!strncmp(data,ar_signature,siglen))
906422
				return AR_MEMBER_ATTR_ARCHIVE;
906422
906422
	} else if (hdrattr & AR_HEADER_ATTR_BSD) {
906422
		if (!strcmp(hdrname,"__.SYMDEF"))
906422
			return AR_MEMBER_ATTR_ARMAP;
906422
906422
		else if (!strcmp(hdrname,"__.SYMDEF SORTED"))
906422
			return AR_MEMBER_ATTR_ARMAP;
906422
906422
		else if (!strcmp(hdrname,"__.SYMDEF_64"))
906422
			return AR_MEMBER_ATTR_ARMAP;
906422
906422
		else if (!strcmp(hdrname,"__.SYMDEF_64 SORTED"))
906422
			return AR_MEMBER_ATTR_ARMAP;
906422
	}
906422
906422
	/* ascii only data? */
906422
	for (; data
906422
		if ((uch = *data) >= 0x80)
906422
			break;
906422
906422
		data++;
906422
	}
906422
906422
	if (data == data_cap)
906422
		return AR_MEMBER_ATTR_ASCII;
906422
906422
	data  = m->ar_object_data;
906422
	udata = (unsigned char *)data;
906422
906422
	/* elf object? [quick and dirty] */
906422
	if (m->ar_file_header.ar_file_size >= 5)
906422
		if ((udata[0] == 0x7f)
906422
				&& (udata[1] == 'E')
906422
				&& (udata[2] == 'L')
906422
				&& (udata[3] == 'F'))
906422
			if ((m->ar_object_attr = AR_OBJECT_ATTR_ELF))
906422
				return AR_MEMBER_ATTR_OBJECT;
906422
906422
	/* coff i386 object? [quick and dirty] */
906422
	if (m->ar_file_header.ar_file_size >= 2)
906422
		if ((udata[0] == 0x4c) && (udata[1] == 0x01))
906422
			if ((m->ar_object_attr = AR_OBJECT_ATTR_COFF))
906422
				return AR_MEMBER_ATTR_OBJECT;
906422
906422
	/* coff x86_64 object? [quick and dirty] */
906422
	if (m->ar_file_header.ar_file_size >= 2)
906422
		if ((udata[0] == 0x64) && (udata[1] == 0x86))
906422
			if ((m->ar_object_attr = AR_OBJECT_ATTR_COFF))
906422
				return AR_MEMBER_ATTR_OBJECT;
906422
906422
	/* big endian 32-bit macho object? [quick and dirty] */
906422
	if (m->ar_file_header.ar_file_size >= 4)
906422
		if ((udata[0] == 0xfe) && (udata[1] == 0xed))
906422
			if ((udata[2] == 0xfa) && (udata[3] == 0xce))
906422
				if ((m->ar_object_attr = AR_OBJECT_ATTR_MACHO))
906422
					return AR_MEMBER_ATTR_OBJECT;
906422
906422
	/* big endian 64-bit macho object? [quick and dirty] */
906422
	if (m->ar_file_header.ar_file_size >= 4)
906422
		if ((udata[0] == 0xfe) && (udata[1] == 0xed))
906422
			if ((udata[2] == 0xfa) && (udata[3] == 0xcf))
906422
				if ((m->ar_object_attr = AR_OBJECT_ATTR_MACHO))
906422
					return AR_MEMBER_ATTR_OBJECT;
906422
906422
	/* little endian 32-bit macho object? [quick and dirty] */
906422
	if (m->ar_file_header.ar_file_size >= 4)
906422
		if ((udata[3] == 0xfe) && (udata[2] == 0xed))
906422
			if ((udata[1] == 0xfa) && (udata[0] == 0xce))
906422
				if ((m->ar_object_attr = AR_OBJECT_ATTR_MACHO))
906422
					return AR_MEMBER_ATTR_OBJECT;
906422
906422
	/* little endian 64-bit macho object? [quick and dirty] */
906422
	if (m->ar_file_header.ar_file_size >= 4)
906422
		if ((udata[3] == 0xfe) && (udata[2] == 0xed))
906422
			if ((udata[1] == 0xfa) && (udata[0] == 0xcf))
906422
				if ((m->ar_object_attr = AR_OBJECT_ATTR_MACHO))
906422
					return AR_MEMBER_ATTR_OBJECT;
906422
906422
	/* all other */
906422
	return AR_MEMBER_ATTR_DEFAULT;
906422
}
906422
906422
static int pe_ar_parse_primary_armap(
906422
	const struct pe_driver_ctx *    dctx,
906422
	struct pe_archive_meta_impl *   m)
906422
906422
{
906422
	struct ar_meta_member_info *    memberp;
906422
	const char *                    hdrname;
906422
	uint32_t                        hdrattr;
906422
906422
	memberp = m->memberv[0];
906422
	hdrname = memberp->ar_file_header.ar_member_name;
906422
	hdrattr = memberp->ar_file_header.ar_header_attr;
906422
906422
	if (!(memberp->ar_member_attr & AR_MEMBER_ATTR_ARMAP))
906422
		return 0;
906422
906422
	if (hdrattr & AR_HEADER_ATTR_SYSV) {
906422
		/* mips 64-bit armap member? */
906422
		if (!strncmp(hdrname,"/SYM64/",7))
906422
			return pe_ar_parse_primary_armap_sysv_64(
906422
				dctx,m);
906422
906422
		/* sysv 32-bit armap member */
906422
		return pe_ar_parse_primary_armap_sysv_32(
906422
			dctx,m);
906422
906422
	} else if (hdrattr & AR_HEADER_ATTR_BSD) {
906422
		if (!strcmp(hdrname,"__.SYMDEF"))
906422
			return pe_ar_parse_primary_armap_bsd_32(
906422
				dctx,m);
906422
906422
		else if (!strcmp(hdrname,"__.SYMDEF SORTED"))
906422
			return pe_ar_parse_primary_armap_bsd_32(
906422
				dctx,m);
906422
906422
		else if (!strcmp(hdrname,"__.SYMDEF_64"))
906422
			return pe_ar_parse_primary_armap_bsd_64(
906422
				dctx,m);
906422
906422
		else if (!strcmp(hdrname,"__.SYMDEF_64 SORTED"))
906422
			return pe_ar_parse_primary_armap_bsd_64(
906422
				dctx,m);
906422
	}
906422
906422
	return PERK_CUSTOM_ERROR(dctx,PERK_ERR_FLOW_ERROR);
906422
}
906422
906422
perk_hidden struct ar_meta_member_info * pe_archive_member_from_offset(
906422
	struct pe_archive_meta_impl *   meta,
906422
	off_t                           offset)
906422
{
906422
	intptr_t l,r,m;
906422
	off_t *  offsetv;
906422
906422
	l = 0;
906422
	r = meta->nentries - 1;
906422
906422
	offsetv = meta->offsetv;
906422
906422
	while (l != r) {
906422
		m  = (l + r) / 2;
906422
		m += (l + r) % 2;
906422
906422
		if (offsetv[m] > offset) {
906422
			r = --m;
906422
		} else {
906422
			l = m;
906422
		}
906422
	}
906422
906422
	return (offsetv[l] == offset) ? meta->memberv[l] : 0;
906422
}
906422
906422
int pe_ar_get_archive_meta(
906422
	const struct pe_driver_ctx *    dctx,
906422
	const struct pe_raw_archive *   archive,
906422
	struct pe_archive_meta **       meta)
906422
{
906422
	const char *                    mark;
906422
	const char *                    cap;
906422
	struct pe_archive_meta_impl * m;
906422
	const char *                    slash;
906422
	const char *                    ch;
906422
	const char *                    fldcap;
906422
	size_t				nelements;
906422
	uint64_t                        nentries;
906422
	uint64_t                        nmembers;
906422
	uint64_t                        stblsize;
906422
	uint64_t                        filesize;
906422
	uint64_t                        namelen;
906422
	uint64_t                        nameoff;
906422
	uint32_t			attr;
906422
	void *                          s_addr;
906422
	void *                          m_addr;
906422
	const char *                    s_ptr;
906422
	const char *                    m_ptr;
906422
	struct ar_raw_file_header *	arhdr;
906422
	struct ar_raw_file_header *	arlongnames;
906422
	struct ar_meta_member_info *    memberp;
906422
	char *				longnamep;
906422
	size_t				idx;
906422
	struct ar_meta_armap_ref_32 *   symrefs_32;
906422
	struct ar_meta_armap_ref_64 *   symrefs_64;
906422
	struct ar_header_info *		hdrinfov;
906422
	struct ar_header_info *		hdrinfov_cap;
906422
	struct ar_header_info *		hdrinfov_next;
906422
	struct ar_header_info		hdrinfobuf[AR_STACK_VECTOR_ELEMENTS];
906422
906422
	/* init */
906422
	hdrinfov     = hdrinfobuf;
906422
	hdrinfov_cap = &hdrinfobuf[AR_STACK_VECTOR_ELEMENTS];
906422
	nelements    = AR_STACK_VECTOR_ELEMENTS;
906422
906422
	memset(hdrinfobuf,0,sizeof(hdrinfobuf));
906422
906422
	mark = archive->map_addr;
906422
	cap  = &mark[archive->map_size];
906422
906422
	/* preliminary validation */
906422
	if (archive->map_size < sizeof(struct ar_raw_signature))
906422
		return PERK_CUSTOM_ERROR(
906422
			dctx,
906422
			PERK_ERR_AR_INVALID_SIGNATURE);
906422
906422
	else if (strncmp(mark,ar_signature,sizeof(struct ar_raw_signature)))
906422
		return PERK_CUSTOM_ERROR(
906422
			dctx,
906422
			PERK_ERR_AR_INVALID_SIGNATURE);
906422
906422
	/* alloc */
906422
	if (!(m = calloc(1,sizeof(*m))))
906422
		return PERK_SYSTEM_ERROR(dctx);
906422
906422
	/* associated driver context */
906422
	m->dctx = dctx;
906422
906422
	/* archive map info */
906422
	m->armeta.r_archive.map_addr = archive->map_addr;
906422
	m->armeta.r_archive.map_size = archive->map_size;
906422
906422
	/* archive signature */
906422
	m->armeta.r_signature = (struct ar_raw_signature *)mark;
906422
	m->armeta.m_signature = (struct ar_meta_signature *)ar_signature;
906422
906422
	/* signature only? */
906422
	if (archive->map_size == sizeof(struct ar_raw_signature)) {
906422
		*meta = &m->armeta;
906422
		return 0;
906422
	}
906422
906422
	mark += sizeof(struct ar_raw_signature);
906422
906422
	/* only trailing null characters past the signature? */
906422
	if (cap < &mark[sizeof(*arhdr)])
906422
		for (ch=mark; ch
906422
			if (*ch)
906422
				return pe_ar_free_archive_meta_impl(
906422
					m,PERK_CUSTOM_ERROR(
906422
						dctx,
906422
						PERK_ERR_AR_INVALID_HEADER));
906422
906422
	/* count entries, calculate string table size */
906422
	for (nentries=0,stblsize=0,arlongnames=0; mark
906422
		arhdr = (struct ar_raw_file_header *)mark;
906422
906422
		/* file size */
906422
		if ((pe_ar_read_decimal_64(
906422
				arhdr->ar_file_size,
906422
				sizeof(arhdr->ar_file_size),
906422
				&filesize)) < 0)
906422
			return pe_ar_free_archive_meta_impl(
906422
				m,PERK_CUSTOM_ERROR(
906422
					dctx,
906422
					PERK_ERR_AR_INVALID_HEADER));
906422
906422
		mark += sizeof(struct ar_raw_file_header);
906422
906422
		/* stblsize, member name type */
906422
		fldcap = &arhdr->ar_file_id[sizeof(arhdr->ar_file_id)];
906422
906422
		/* sysv long names table? */
906422
		if ((arhdr->ar_file_id[0] == '/') && (arhdr->ar_file_id[1] == '/')) {
906422
			for (ch=&arhdr->ar_file_id[2]; ch
906422
				if (*ch != AR_DEC_PADDING)
906422
					return pe_ar_free_archive_meta_impl(
906422
						m,PERK_CUSTOM_ERROR(
906422
							dctx,
906422
							PERK_ERR_AR_INVALID_HEADER));
906422
906422
			if (pe_ar_read_decimal_64(
906422
					arhdr->ar_file_size,
906422
					sizeof(arhdr->ar_file_size),
906422
					&namelen) < 0)
906422
				return pe_ar_free_archive_meta_impl(
906422
					m,PERK_CUSTOM_ERROR(
906422
						dctx,
906422
						PERK_ERR_AR_INVALID_HEADER));
906422
906422
906422
			/* duplicate long names member? */
906422
			if (arlongnames)
906422
				return pe_ar_free_archive_meta_impl(
906422
					m,PERK_CUSTOM_ERROR(
906422
						dctx,
906422
						PERK_ERR_AR_DUPLICATE_LONG_NAMES));
906422
906422
			attr = AR_HEADER_ATTR_FILE_ID | AR_HEADER_ATTR_SYSV;
906422
906422
			stblsize++;
906422
			stblsize++;
906422
			stblsize++;
906422
906422
			stblsize += namelen;
906422
906422
			arlongnames = arhdr;
906422
906422
		/* the /SYM64/ string must be special cased, also below when it gets copied */
906422
		} else if (!strncmp(arhdr->ar_file_id,"/SYM64/",7)) {
906422
			for (ch=&arhdr->ar_file_id[7]; ch
906422
				if (*ch != AR_DEC_PADDING)
906422
					return pe_ar_free_archive_meta_impl(
906422
						m,PERK_CUSTOM_ERROR(
906422
							dctx,
906422
							PERK_ERR_AR_INVALID_HEADER));
906422
906422
			attr      = AR_HEADER_ATTR_FILE_ID | AR_HEADER_ATTR_SYSV;
906422
			stblsize += 8;
906422
906422
		/* sysv armap member or sysv long name reference? */
906422
		} else if (arhdr->ar_file_id[0] == '/') {
906422
			if (pe_ar_read_decimal_64(
906422
					&arhdr->ar_file_id[1],
906422
					sizeof(arhdr->ar_file_id)-1,
906422
					&nameoff) < 0)
906422
				return pe_ar_free_archive_meta_impl(
906422
					m,PERK_CUSTOM_ERROR(
906422
						dctx,
906422
						PERK_ERR_AR_INVALID_HEADER));
906422
906422
			if (arhdr->ar_file_id[1] == AR_DEC_PADDING) {
906422
				attr = AR_HEADER_ATTR_FILE_ID | AR_HEADER_ATTR_SYSV;
906422
				stblsize++;
906422
				stblsize++;
906422
			} else {
906422
				attr = AR_HEADER_ATTR_NAME_REF | AR_HEADER_ATTR_SYSV;
906422
			}
906422
906422
		/* bsd long name reference? */
906422
		} else if ((arhdr->ar_file_id[0] == '#')
906422
				&& (arhdr->ar_file_id[1] == '1')
906422
				&& (arhdr->ar_file_id[2] == '/')) {
906422
			if (pe_ar_read_decimal_64(
906422
					&arhdr->ar_file_id[3],
906422
					sizeof(arhdr->ar_file_id)-3,
906422
					&namelen) < 0)
906422
				return pe_ar_free_archive_meta_impl(
906422
					m,PERK_CUSTOM_ERROR(
906422
						dctx,
906422
						PERK_ERR_AR_INVALID_HEADER));
906422
906422
			attr = AR_HEADER_ATTR_NAME_REF | AR_HEADER_ATTR_BSD;
906422
906422
			stblsize += namelen + 1;
906422
906422
		/* must be either a sysv short member name, or a (legacy) bsd short name */
906422
		} else {
906422
			for (ch=arhdr->ar_file_id,slash=0; (ch
906422
				if (*ch == '/')
906422
					slash = ch;
906422
906422
			if (slash) {
906422
				attr      = AR_HEADER_ATTR_FILE_ID | AR_HEADER_ATTR_SYSV;
906422
				stblsize += (slash - arhdr->ar_file_id) + 1;
906422
			} else {
906422
				attr      = AR_HEADER_ATTR_FILE_ID | AR_HEADER_ATTR_BSD;
906422
				stblsize += sizeof(arhdr->ar_file_id) + 1;
906422
			}
906422
906422
			for (; ch
906422
				if (*ch++ != AR_DEC_PADDING)
906422
					return pe_ar_free_archive_meta_impl(
906422
						m,PERK_CUSTOM_ERROR(
906422
							dctx,
906422
							PERK_ERR_AR_INVALID_HEADER));
906422
906422
		}
906422
906422
		/* truncated data? */
906422
		if (cap < &mark[filesize])
906422
			return pe_ar_free_archive_meta_impl(
906422
				m,PERK_CUSTOM_ERROR(
906422
					dctx,
906422
					PERK_ERR_AR_TRUNCATED_DATA));
906422
906422
		/* ar member alignment */
906422
		filesize += 1;
906422
		filesize |= 1;
906422
		filesize ^= 1;
906422
906422
		mark += filesize;
906422
906422
		/* only trailing null characters past the signature? */
906422
		if (cap < &mark[sizeof(*arhdr)])
906422
			for (; mark
906422
				if (*mark++)
906422
					return pe_ar_free_archive_meta_impl(
906422
						m,PERK_CUSTOM_ERROR(
906422
							dctx,
906422
							PERK_ERR_AR_INVALID_HEADER));
906422
906422
		/* transient header info vector */
906422
		if (&hdrinfov[nentries] == hdrinfov_cap) {
906422
			nelements = (nelements == AR_STACK_VECTOR_ELEMENTS)
906422
				? (nelements << 4) : (nelements << 1);
906422
906422
			if (!(hdrinfov_next = calloc(nelements,sizeof(*hdrinfov))))
906422
				return pe_ar_free_archive_meta_impl(
906422
					m,PERK_CUSTOM_ERROR(
906422
						dctx,
906422
						PERK_ERR_AR_TRUNCATED_DATA));
906422
906422
			for (idx=0; idx
906422
				hdrinfov_next[idx].phdr = hdrinfov[idx].phdr;
906422
				hdrinfov_next[idx].attr = hdrinfov[idx].attr;
906422
			};
906422
906422
			if (hdrinfov != hdrinfobuf)
906422
				free(hdrinfov);
906422
906422
			hdrinfov     = hdrinfov_next;
906422
			hdrinfov_cap = &hdrinfov_next[nelements];
906422
			m->hdrinfov  = hdrinfov;
906422
		}
906422
906422
		hdrinfov[nentries].phdr = arhdr;
906422
		hdrinfov[nentries].attr = attr;
906422
	}
906422
906422
	/* allocate name strings, member vector */
906422
	if (!(m->namestrs = calloc(1,stblsize)))
906422
		return pe_ar_free_archive_meta_impl(
906422
			m,PERK_SYSTEM_ERROR(dctx));
906422
906422
	if (!(m->offsetv = calloc(nentries+1,sizeof(*m->offsetv))))
906422
		return pe_ar_free_archive_meta_impl(
906422
			m,PERK_SYSTEM_ERROR(dctx));
906422
906422
	if (!(m->memberv = calloc(nentries+1,sizeof(*m->memberv))))
906422
		return pe_ar_free_archive_meta_impl(
906422
			m,PERK_SYSTEM_ERROR(dctx));
906422
906422
	if (!(m->members = calloc(nentries,sizeof(*m->members))))
906422
		return pe_ar_free_archive_meta_impl(
906422
			m,PERK_SYSTEM_ERROR(dctx));
906422
906422
	/* archive signature reference */
906422
	s_addr = archive->map_addr;
906422
	s_ptr  = s_addr;
906422
906422
	/* iterate, store meta data in library-friendly form */
906422
	for (idx=0,longnamep=m->namestrs; idx
906422
		arhdr           = hdrinfov[idx].phdr;
906422
		attr            = hdrinfov[idx].attr;
906422
906422
		m_addr          = arhdr;
906422
		m_ptr           = m_addr;
906422
906422
		memberp         = &m->members[idx];
906422
		m->offsetv[idx] = m_ptr - s_ptr;
906422
		m->memberv[idx] = memberp;
906422
906422
		memberp->ar_file_header.ar_header_attr = attr;
906422
906422
		pe_ar_read_decimal_64(
906422
			arhdr->ar_time_date_stamp,
906422
			sizeof(arhdr->ar_time_date_stamp),
906422
			&memberp->ar_file_header.ar_time_date_stamp);
906422
906422
		pe_ar_read_decimal_32(
906422
			arhdr->ar_uid,
906422
			sizeof(arhdr->ar_uid),
906422
			&memberp->ar_file_header.ar_uid);
906422
906422
		pe_ar_read_decimal_32(
906422
			arhdr->ar_gid,
906422
			sizeof(arhdr->ar_gid),
906422
			&memberp->ar_file_header.ar_gid);
906422
906422
		pe_ar_read_octal(
906422
			arhdr->ar_file_mode,
906422
			sizeof(arhdr->ar_file_mode),
906422
			&memberp->ar_file_header.ar_file_mode);
906422
906422
		pe_ar_read_decimal_64(
906422
			arhdr->ar_file_size,
906422
			sizeof(arhdr->ar_file_size),
906422
			&memberp->ar_file_header.ar_file_size);
906422
906422
		memberp->ar_file_header.ar_member_name = longnamep;
906422
906422
		if (attr == (AR_HEADER_ATTR_FILE_ID | AR_HEADER_ATTR_SYSV)) {
906422
			if ((arhdr->ar_file_id[0] == '/') && (arhdr->ar_file_id[1] == '/')) {
906422
				*longnamep++ = '/';
906422
				*longnamep++ = '/';
906422
				longnamep++;
906422
906422
			} else if ((arhdr->ar_file_id[0] == '/') && (arhdr->ar_file_id[1] == 'S')) {
906422
				*longnamep++ = '/';
906422
				*longnamep++ = 'S';
906422
				*longnamep++ = 'Y';
906422
				*longnamep++ = 'M';
906422
				*longnamep++ = '6';
906422
				*longnamep++ = '4';
906422
				*longnamep++ = '/';
906422
				longnamep++;
906422
906422
			} else if (arhdr->ar_file_id[0] == '/') {
906422
				*longnamep++ = '/';
906422
				longnamep++;
906422
906422
			} else {
906422
				ch = arhdr->ar_file_id;
906422
906422
				for (; (*ch != '/'); )
906422
					*longnamep++ = *ch++;
906422
906422
				longnamep++;
906422
			}
906422
906422
		} else if (attr == (AR_HEADER_ATTR_FILE_ID | AR_HEADER_ATTR_BSD)) {
906422
			ch     = arhdr->ar_file_id;
906422
			fldcap = &ch[sizeof(arhdr->ar_file_id)];
906422
906422
			for (; (ch
906422
				*longnamep++ = *ch++;
906422
906422
			longnamep++;
906422
906422
		} else if (attr == (AR_HEADER_ATTR_NAME_REF | AR_HEADER_ATTR_SYSV)) {
906422
			pe_ar_read_decimal_64(
906422
				&arhdr->ar_file_id[1],
906422
				sizeof(arhdr->ar_file_id) - 1,
906422
				&nameoff);
906422
906422
			ch  = arlongnames->ar_file_id;
906422
			ch += sizeof(*arlongnames);
906422
			ch += nameoff;
906422
906422
			for (; *ch && (*ch != '/') && (*ch != AR_OBJ_PADDING); )
906422
				*longnamep++ = *ch++;
906422
906422
			longnamep++;
906422
906422
		} else if (attr == (AR_HEADER_ATTR_NAME_REF | AR_HEADER_ATTR_BSD)) {
906422
			pe_ar_read_decimal_64(
906422
				&arhdr->ar_file_id[3],
906422
				sizeof(arhdr->ar_file_id) - 3,
906422
				&namelen);
906422
906422
			mark  = arhdr->ar_file_id;
906422
			mark += sizeof(*arhdr);
906422
906422
			memcpy(longnamep,mark,namelen);
906422
906422
			longnamep += namelen;
906422
			longnamep++;
906422
		}
906422
906422
		/* member raw header, object size, object data */
906422
		mark    = arhdr->ar_file_id;
906422
		mark   += sizeof(*arhdr);
906422
		namelen = 0;
906422
906422
		if (attr == (AR_HEADER_ATTR_NAME_REF | AR_HEADER_ATTR_BSD)) {
906422
			pe_ar_read_decimal_64(
906422
				&arhdr->ar_file_id[3],
906422
				sizeof(arhdr->ar_file_id)-3,
906422
				&namelen);
906422
906422
			namelen += 1;
906422
			namelen |= 1;
906422
			namelen ^= 1;
906422
906422
			mark += namelen;
906422
		};
906422
906422
		memberp->ar_member_data = arhdr;
906422
		memberp->ar_object_data = (void *)mark;
906422
		memberp->ar_object_size = memberp->ar_file_header.ar_file_size - namelen;
906422
906422
		/* member attribute */
906422
		memberp->ar_member_attr = pe_ar_get_member_attr(memberp);
906422
906422
		/* pe/coff second linker member? */
906422
		if ((idx == 1) && (memberp->ar_member_attr == AR_MEMBER_ATTR_ARMAP))
906422
			if (hdrinfov[0].attr & AR_HEADER_ATTR_SYSV)
906422
				if (m->members[0].ar_member_attr == AR_MEMBER_ATTR_ARMAP)
906422
					if (attr & AR_HEADER_ATTR_SYSV)
906422
						memberp->ar_member_attr = AR_MEMBER_ATTR_LINKINFO;
906422
906422
		/* armap member must be the first */
906422
		if ((memberp->ar_member_attr == AR_MEMBER_ATTR_ARMAP) && (idx > 0)) {
906422
			if (m->members[0].ar_member_attr == AR_MEMBER_ATTR_ARMAP)
906422
				return pe_ar_free_archive_meta_impl(
906422
					m,PERK_CUSTOM_ERROR(
906422
						dctx,
906422
						PERK_ERR_AR_DUPLICATE_ARMAP_MEMBER));
906422
906422
			return pe_ar_free_archive_meta_impl(
906422
				m,PERK_CUSTOM_ERROR(
906422
					dctx,
906422
					PERK_ERR_AR_MISPLACED_ARMAP_MEMBER));
906422
		}
906422
	}
906422
906422
	/* number of archive members, including internal ones */
906422
	m->nentries = nentries;
906422
906422
	/* primary armap (first linker member) */
906422
	if (pe_ar_parse_primary_armap(dctx,m) < 0)
906422
		return pe_ar_free_archive_meta_impl(
906422
			m,PERK_NESTED_ERROR(dctx));
906422
906422
	for (idx=0,ch=m->symstrs; idx<m->armaps.armap_nsyms; idx++) {
906422
		m->symstrv[idx] = ch;
906422
		ch += strlen(ch);
906422
		ch++;
906422
	}
906422
906422
	if (m->armaps.armap_common_32.ar_member) {
906422
		symrefs_32 = m->armaps.armap_symrefs_32;
906422
906422
		for (idx=0; idx<m->armaps.armap_nsyms; idx++) {
906422
			if (m->armaps.armap_common_32.ar_armap_attr & AR_ARMAP_ATTR_SYSV)
906422
				symrefs_32[idx].ar_name_offset = m->symstrv[idx] - m->symstrv[0];
906422
906422
			if (!pe_archive_member_from_offset(m,symrefs_32[idx].ar_member_offset))
906422
				return pe_ar_free_archive_meta_impl(
906422
					m,PERK_CUSTOM_ERROR(
906422
						dctx,
906422
						PERK_ERR_AR_INVALID_ARMAP_MEMBER_OFFSET));
906422
906422
			if (symrefs_32[idx].ar_name_offset) {
906422
				ch = &m->symstrs[symrefs_32[idx].ar_name_offset];
906422
906422
				if ((ch > m->symstrv[m->armaps.armap_nsyms - 1]) || *--ch)
906422
					return pe_ar_free_archive_meta_impl(
906422
						m,PERK_CUSTOM_ERROR(
906422
							dctx,
906422
							PERK_ERR_AR_INVALID_ARMAP_NAME_OFFSET));
906422
			}
906422
906422
		}
906422
	}
906422
906422
	if (m->armaps.armap_common_64.ar_member) {
906422
		symrefs_64 = m->armaps.armap_symrefs_64;
906422
906422
		for (idx=0; idx<m->armaps.armap_nsyms; idx++) {
906422
			if (m->armaps.armap_common_64.ar_armap_attr & AR_ARMAP_ATTR_SYSV)
906422
				symrefs_64[idx].ar_name_offset = m->symstrv[idx] - m->symstrv[0];
906422
906422
			if (!pe_archive_member_from_offset(m,symrefs_64[idx].ar_member_offset))
906422
				return pe_ar_free_archive_meta_impl(
906422
					m,PERK_CUSTOM_ERROR(
906422
						dctx,
906422
						PERK_ERR_AR_INVALID_ARMAP_MEMBER_OFFSET));
906422
906422
			if (symrefs_64[idx].ar_name_offset) {
906422
				ch = &m->symstrs[symrefs_64[idx].ar_name_offset];
906422
906422
				if ((ch > m->symstrv[m->armaps.armap_nsyms - 1]) || *--ch)
906422
					return pe_ar_free_archive_meta_impl(
906422
						m,PERK_CUSTOM_ERROR(
906422
							dctx,
906422
							PERK_ERR_AR_INVALID_ARMAP_NAME_OFFSET));
906422
			}
906422
906422
		}
906422
	}
906422
906422
	/* number of public archive members */
906422
	for (idx=0,nmembers=0; idx
906422
		switch (m->memberv[idx]->ar_member_attr) {
906422
			case AR_MEMBER_ATTR_ARMAP:
906422
			case AR_MEMBER_ATTR_LINKINFO:
906422
			case AR_MEMBER_ATTR_NAMESTRS:
906422
				break;
906422
906422
			default:
906422
				nmembers++;
906422
		}
906422
	}
906422
906422
	if (m->armaps.armap_common_32.ar_member)
906422
		m->armaps.armap_common_32.ar_num_of_members = nmembers;
906422
906422
	if (m->armaps.armap_common_64.ar_member)
906422
		m->armaps.armap_common_64.ar_num_of_members = nmembers;
906422
906422
	/* pe/coff armap attributes (second linker member) */
906422
	(void)m->armeta.a_armap_pecoff;
906422
906422
	/* reference to the long names member */
906422
	if (arlongnames)
906422
		for (idx=0; idx<nentries && !m->armeta.a_arref_longnames; idx++)
906422
			if (m->memberv[idx]->ar_member_data == arlongnames)
906422
				m->armeta.a_arref_longnames = m->memberv[idx];
906422
906422
	/* common binary format (information only) */
906422
	for (idx=0,nmembers=0; idx
906422
		if (m->memberv[idx]->ar_member_attr == AR_MEMBER_ATTR_OBJECT) {
906422
			if (m->ofmtattr && (m->ofmtattr != m->memberv[idx]->ar_object_attr)) {
906422
				m->ofmtattr = 0;
906422
				idx = nentries;
906422
			} else if (!m->ofmtattr) {
906422
				m->ofmtattr = m->memberv[idx]->ar_object_attr;
906422
			}
906422
		}
906422
	}
906422
906422
	/* member vector */
906422
	m->armeta.a_memberv = m->memberv;
906422
906422
	/* all done */
906422
	if (m->hdrinfov) {
906422
		free(m->hdrinfov);
906422
		m->hdrinfov = 0;
906422
	}
906422
906422
	*meta = &m->armeta;
906422
906422
	return 0;
906422
}
906422
906422
void pe_ar_free_archive_meta(struct pe_archive_meta * meta)
906422
{
906422
	struct pe_archive_meta_impl * m;
906422
906422
	if (meta) {
906422
		m = pe_archive_meta_ictx(meta);
906422
		pe_ar_free_archive_meta_impl(m,0);
906422
	}
906422
}