#include <elfedit.h>
#include <strings.h>
#include <conv.h>
#include <debug.h>
#include <phdr_msg.h>
typedef enum {
PHDR_CMD_T_DUMP = 0,
PHDR_CMD_T_P_TYPE = 1,
PHDR_CMD_T_P_OFFSET = 2,
PHDR_CMD_T_P_VADDR = 3,
PHDR_CMD_T_P_PADDR = 4,
PHDR_CMD_T_P_FILESZ = 5,
PHDR_CMD_T_P_MEMSZ = 6,
PHDR_CMD_T_P_FLAGS = 7,
PHDR_CMD_T_P_ALIGN = 8,
PHDR_CMD_T_INTERP = 9,
PHDR_CMD_T_DELETE = 10,
PHDR_CMD_T_MOVE = 11
} PHDR_CMD_T;
typedef struct {
Word phndx;
Phdr *phdr;
elfedit_section_t *sec;
Word stroff;
const char *str;
} INTERP_STATE;
#ifndef _ELF64
const char *
_phdr_msg(Msg mid)
{
return (gettext(MSG_ORIG(mid)));
}
#endif
static const char *
mod_i18nhdl_to_str(elfedit_i18nhdl_t hdl)
{
Msg msg = (Msg)hdl;
return (MSG_INTL(msg));
}
typedef enum {
PHDR_OPT_F_AND = 1,
PHDR_OPT_F_CMP = 2,
PHDR_OPT_F_PHNDX = 4,
PHDR_OPT_F_OR = 8
} phdr_opt_t;
typedef struct {
elfedit_obj_state_t *obj_state;
phdr_opt_t optmask;
int argc;
const char **argv;
int ndx_set;
Word ndx;
int print_req;
} ARGSTATE;
static void
process_args(elfedit_obj_state_t *obj_state, int argc, const char *argv[],
PHDR_CMD_T cmd, ARGSTATE *argstate)
{
elfedit_getopt_state_t getopt_state;
elfedit_getopt_ret_t *getopt_ret;
bzero(argstate, sizeof (*argstate));
argstate->obj_state = obj_state;
elfedit_getopt_init(&getopt_state, &argc, &argv);
while ((getopt_ret = elfedit_getopt(&getopt_state)) != NULL)
argstate->optmask |= getopt_ret->gor_idmask;
switch (cmd) {
case PHDR_CMD_T_DUMP:
if (argc > 1)
elfedit_command_usage();
argstate->print_req = 1;
break;
case PHDR_CMD_T_P_FLAGS:
argstate->print_req = (argc < 2);
break;
case PHDR_CMD_T_INTERP:
if (argc > 1)
elfedit_command_usage();
argstate->print_req = (argc == 0);
break;
case PHDR_CMD_T_DELETE:
if ((argc < 1) || (argc > 2))
elfedit_command_usage();
argstate->print_req = 0;
break;
case PHDR_CMD_T_MOVE:
if ((argc < 2) || (argc > 3))
elfedit_command_usage();
argstate->print_req = 0;
break;
default:
if (argc > 2)
elfedit_command_usage();
argstate->print_req = (argc < 2);
break;
}
argstate->argc = argc;
argstate->argv = argv;
argstate->ndx_set = 0;
if ((argc > 0) && (cmd != PHDR_CMD_T_INTERP)) {
if (argstate->optmask & PHDR_OPT_F_PHNDX) {
argstate->ndx = (Word) elfedit_atoui_range(
argstate->argv[0], MSG_ORIG(MSG_STR_ELEMENT), 0,
argstate->obj_state->os_phnum - 1, NULL);
argstate->ndx_set = 1;
} else {
Conv_inv_buf_t inv_buf;
Ehdr *ehdr = obj_state->os_ehdr;
Half mach = ehdr->e_machine;
uchar_t osabi = ehdr->e_ident[EI_OSABI];
Word i;
Phdr *phdr;
argstate->ndx = (Word) elfedit_atoconst(
argstate->argv[0], ELFEDIT_CONST_PT);
phdr = obj_state->os_phdr;
for (i = 0; i < obj_state->os_phnum; i++, phdr++) {
if (phdr->p_type == argstate->ndx) {
argstate->ndx = i;
argstate->ndx_set = 1;
elfedit_msg(ELFEDIT_MSG_DEBUG,
MSG_INTL(MSG_DEBUG_PHDR),
EC_WORD(i), conv_phdr_type(osabi,
mach, phdr->p_type, 0, &inv_buf));
break;
}
}
if (i == argstate->obj_state->os_phnum)
elfedit_msg(ELFEDIT_MSG_ERR,
MSG_INTL(MSG_ERR_NOPHDR), conv_phdr_type(
osabi, mach, argstate->ndx, 0, &inv_buf));
}
}
if (argc == 0)
elfedit_pager_init();
}
static const char *
locate_interp(elfedit_obj_state_t *obj_state, INTERP_STATE *interp)
{
INTERP_STATE local_interp;
elfedit_section_t *strsec;
size_t phnum;
int phndx;
Phdr *phdr;
Word i;
if (interp == NULL)
interp = &local_interp;
phnum = obj_state->os_phnum;
phdr = obj_state->os_phdr;
for (phndx = 0; phndx < phnum; phndx++) {
if (phdr[phndx].p_type == PT_INTERP) {
interp->phndx = phndx;
interp->phdr = phdr + phndx;
break;
}
}
if (phndx == phnum)
elfedit_elferr(obj_state->os_file,
MSG_INTL(MSG_ERR_NOINTERPPHDR));
for (i = 1; i < obj_state->os_shnum; i++) {
strsec = &obj_state->os_secarr[i];
if ((strsec->sec_shdr->sh_type != SHT_NOBITS) &&
(interp->phdr->p_offset >= strsec->sec_shdr->sh_offset) &&
((interp->phdr->p_offset + interp->phdr->p_filesz) <=
(strsec->sec_shdr->sh_offset +
strsec->sec_shdr->sh_size))) {
interp->sec = strsec;
interp->stroff = interp->phdr->p_offset -
strsec->sec_shdr->sh_offset;
interp->str = ((char *)strsec->sec_data->d_buf) +
interp->stroff;
return (interp->str);
}
}
elfedit_msg(ELFEDIT_MSG_ERR, MSG_INTL(MSG_ERR_NOINTERPSEC));
return (NULL);
}
static void
print_phdr(PHDR_CMD_T cmd, int autoprint, ARGSTATE *argstate)
{
elfedit_outstyle_t outstyle;
Ehdr *ehdr = argstate->obj_state->os_ehdr;
uchar_t osabi = ehdr->e_ident[EI_OSABI];
Half mach = ehdr->e_machine;
Word ndx, cnt, by_type, type;
Phdr *phdr;
if (autoprint && ((elfedit_flags() & ELFEDIT_F_AUTOPRINT) == 0))
return;
by_type = 0;
if (argstate->ndx_set) {
ndx = argstate->ndx;
if (argstate->print_req &&
((argstate->optmask & PHDR_OPT_F_PHNDX) == 0)) {
by_type = 1;
type = argstate->obj_state->os_phdr[ndx].p_type;
cnt = argstate->obj_state->os_phnum - ndx;
} else {
cnt = 1;
}
} else {
ndx = 0;
cnt = argstate->obj_state->os_phnum;
}
phdr = argstate->obj_state->os_phdr + ndx;
outstyle = (cmd == PHDR_CMD_T_DUMP) ?
ELFEDIT_OUTSTYLE_DEFAULT : elfedit_outstyle();
if ((outstyle == ELFEDIT_OUTSTYLE_DEFAULT) &&
(cmd != PHDR_CMD_T_INTERP)) {
for (; cnt--; ndx++, phdr++) {
if (by_type && (type != phdr->p_type))
continue;
elfedit_printf(MSG_ORIG(MSG_STR_NL));
elfedit_printf(MSG_INTL(MSG_ELF_PHDR), EC_WORD(ndx));
Elf_phdr(0, osabi, mach, phdr);
}
return;
}
if (cmd == PHDR_CMD_T_INTERP) {
INTERP_STATE interp;
(void) locate_interp(argstate->obj_state, &interp);
switch (outstyle) {
case ELFEDIT_OUTSTYLE_DEFAULT:
elfedit_printf(MSG_INTL(MSG_FMT_ELF_INTERP),
interp.sec->sec_name, interp.str);
break;
case ELFEDIT_OUTSTYLE_SIMPLE:
elfedit_printf(MSG_ORIG(MSG_FMT_STRNL), interp.str);
break;
case ELFEDIT_OUTSTYLE_NUM:
elfedit_printf(MSG_ORIG(MSG_FMT_U_NL),
EC_WORD(interp.stroff));
break;
}
return;
}
for (; cnt--; ndx++, phdr++) {
if (by_type && (type != phdr->p_type))
continue;
switch (cmd) {
case PHDR_CMD_T_P_TYPE:
if (outstyle == ELFEDIT_OUTSTYLE_SIMPLE) {
Conv_inv_buf_t inv_buf;
elfedit_printf(MSG_ORIG(MSG_FMT_STRNL),
conv_phdr_type(osabi,
argstate->obj_state->os_ehdr->e_machine,
phdr->p_type, 0, &inv_buf));
} else {
elfedit_printf(MSG_ORIG(MSG_FMT_X_NL),
EC_WORD(phdr->p_type));
}
break;
case PHDR_CMD_T_P_OFFSET:
elfedit_printf(MSG_ORIG(MSG_FMT_LLX_NL),
EC_OFF(phdr->p_offset));
break;
case PHDR_CMD_T_P_VADDR:
elfedit_printf(MSG_ORIG(MSG_FMT_LLX_NL),
EC_ADDR(phdr->p_vaddr));
break;
case PHDR_CMD_T_P_PADDR:
elfedit_printf(MSG_ORIG(MSG_FMT_LLX_NL),
EC_ADDR(phdr->p_paddr));
break;
case PHDR_CMD_T_P_FILESZ:
elfedit_printf(MSG_ORIG(MSG_FMT_LLX_NL),
EC_XWORD(phdr->p_filesz));
break;
case PHDR_CMD_T_P_MEMSZ:
elfedit_printf(MSG_ORIG(MSG_FMT_LLX_NL),
EC_XWORD(phdr->p_memsz));
break;
case PHDR_CMD_T_P_FLAGS:
if (outstyle == ELFEDIT_OUTSTYLE_SIMPLE) {
Conv_phdr_flags_buf_t phdr_flags_buf;
elfedit_printf(MSG_ORIG(MSG_FMT_STRNL),
conv_phdr_flags(osabi, phdr->p_flags,
CONV_FMT_NOBKT, &phdr_flags_buf));
} else {
elfedit_printf(MSG_ORIG(MSG_FMT_X_NL),
EC_WORD(phdr->p_flags));
}
break;
case PHDR_CMD_T_P_ALIGN:
elfedit_printf(MSG_ORIG(MSG_FMT_LLX_NL),
EC_XWORD(phdr->p_align));
break;
}
}
}
static elfedit_cmdret_t
cmd_body_set_interp(ARGSTATE *argstate)
{
elfedit_obj_state_t *obj_state = argstate->obj_state;
elfedit_section_t *strsec;
INTERP_STATE interp;
Word numdyn;
size_t phnum;
Phdr *phdr;
Word i, j;
Word str_offset;
int str_found = 0;
Word str_size;
phnum = obj_state->os_phnum;
phdr = obj_state->os_phdr;
(void) locate_interp(obj_state, &interp);
strsec = interp.sec;
str_offset = interp.stroff;
if (strcmp(interp.str, argstate->argv[0]) == 0) {
elfedit_msg(ELFEDIT_MSG_DEBUG, MSG_INTL(MSG_DEBUG_OLDINTERPOK),
EC_WORD(strsec->sec_shndx), strsec->sec_name,
EC_WORD(str_offset), interp.str);
return (ELFEDIT_CMDRET_NONE);
}
str_size = strlen(argstate->argv[0]) + 1;
for (i = 1; i < obj_state->os_shnum; i++) {
strsec = &obj_state->os_secarr[i];
if ((strcmp(strsec->sec_name, MSG_ORIG(MSG_SEC_INTERP)) == 0) &&
(strsec->sec_shdr->sh_flags & SHF_ALLOC) &&
(strsec->sec_shdr->sh_type & SHT_PROGBITS)) {
for (j = 0; j < phnum; j++) {
Phdr *tphdr = &phdr[j];
if ((strsec->sec_shdr->sh_offset >=
tphdr->p_offset) &&
((strsec->sec_shdr->sh_offset +
strsec->sec_shdr->sh_size) <=
(tphdr->p_offset + tphdr->p_filesz)) &&
(tphdr->p_flags & PF_W)) {
break;
}
}
if ((j == phnum) &&
(str_size <= strsec->sec_shdr->sh_size)) {
str_found = 1;
str_offset = 0;
elfedit_msg(ELFEDIT_MSG_DEBUG,
MSG_INTL(MSG_DEBUG_NEWISTR), EC_WORD(j),
strsec->sec_name, EC_WORD(str_offset),
argstate->argv[0]);
(void) strncpy((char *)strsec->sec_data->d_buf,
argstate->argv[0],
strsec->sec_shdr->sh_size);
elfedit_modified_data(strsec);
break;
} else {
elfedit_msg(ELFEDIT_MSG_DEBUG,
MSG_INTL(MSG_DEBUG_LNGISTR), EC_WORD(j),
strsec->sec_name, EC_WORD(str_offset),
EC_WORD(str_size),
EC_WORD(strsec->sec_shdr->sh_size),
argstate->argv[0]);
}
}
}
if (!str_found) {
elfedit_section_t *dynsec;
Dyn *dyn;
dynsec = elfedit_sec_getdyn(obj_state, &dyn, &numdyn);
strsec = elfedit_sec_getstr(obj_state,
dynsec->sec_shdr->sh_link, 0);
str_offset = elfedit_strtab_insert(obj_state, strsec,
dynsec, argstate->argv[0]);
}
interp.phdr->p_offset = strsec->sec_shdr->sh_offset + str_offset;
interp.phdr->p_filesz = str_size;
elfedit_modified_phdr(obj_state);
elfedit_msg(ELFEDIT_MSG_DEBUG, MSG_INTL(MSG_DEBUG_SETPHINTERP),
EC_WORD(interp.phndx), EC_XWORD(interp.phdr->p_offset),
EC_XWORD(interp.phdr->p_filesz));
return (ELFEDIT_CMDRET_MOD);
}
static elfedit_cmdret_t
cmd_body(PHDR_CMD_T cmd, elfedit_obj_state_t *obj_state,
int argc, const char *argv[])
{
ARGSTATE argstate;
Phdr *phdr;
elfedit_cmdret_t ret = ELFEDIT_CMDRET_NONE;
int do_autoprint = 1;
process_args(obj_state, argc, argv, cmd, &argstate);
if (argstate.print_req) {
print_phdr(cmd, 0, &argstate);
return (ELFEDIT_CMDRET_NONE);
}
if (argstate.ndx_set)
phdr = &argstate.obj_state->os_phdr[argstate.ndx];
switch (cmd) {
case PHDR_CMD_T_P_TYPE:
{
Ehdr *ehdr = obj_state->os_ehdr;
uchar_t osabi = ehdr->e_ident[EI_OSABI];
Half mach = ehdr->e_machine;
Word p_type = elfedit_atoconst(argstate.argv[1],
ELFEDIT_CONST_PT);
Conv_inv_buf_t inv_buf1, inv_buf2;
if (phdr->p_type == p_type) {
elfedit_msg(ELFEDIT_MSG_DEBUG,
MSG_INTL(MSG_DEBUG_S_OK),
argstate.ndx, MSG_ORIG(MSG_CMD_P_TYPE),
conv_phdr_type(osabi, mach, phdr->p_type,
0, &inv_buf1));
} else {
elfedit_msg(ELFEDIT_MSG_DEBUG,
MSG_INTL(MSG_DEBUG_S_CHG),
argstate.ndx, MSG_ORIG(MSG_CMD_P_TYPE),
conv_phdr_type(osabi, mach,
phdr->p_type, 0, &inv_buf1),
conv_phdr_type(osabi, mach,
p_type, 0, &inv_buf2));
ret = ELFEDIT_CMDRET_MOD;
phdr->p_type = p_type;
}
}
break;
case PHDR_CMD_T_P_OFFSET:
{
Off p_offset;
p_offset = elfedit_atoui(argstate.argv[1], NULL);
if (phdr->p_offset == p_offset) {
elfedit_msg(ELFEDIT_MSG_DEBUG,
MSG_INTL(MSG_DEBUG_LLX_OK),
argstate.ndx, MSG_ORIG(MSG_CMD_P_OFFSET),
EC_XWORD(phdr->p_offset));
} else {
elfedit_msg(ELFEDIT_MSG_DEBUG,
MSG_INTL(MSG_DEBUG_LLX_CHG),
argstate.ndx, MSG_ORIG(MSG_CMD_P_OFFSET),
EC_XWORD(phdr->p_offset),
EC_XWORD(p_offset));
ret = ELFEDIT_CMDRET_MOD;
phdr->p_offset = p_offset;
}
}
break;
case PHDR_CMD_T_P_VADDR:
{
Addr p_vaddr = elfedit_atoui(argstate.argv[1], NULL);
if (phdr->p_vaddr == p_vaddr) {
elfedit_msg(ELFEDIT_MSG_DEBUG,
MSG_INTL(MSG_DEBUG_LLX_OK),
argstate.ndx, MSG_ORIG(MSG_CMD_P_VADDR),
EC_ADDR(phdr->p_vaddr));
} else {
elfedit_msg(ELFEDIT_MSG_DEBUG,
MSG_INTL(MSG_DEBUG_LLX_CHG),
argstate.ndx, MSG_ORIG(MSG_CMD_P_VADDR),
EC_ADDR(phdr->p_vaddr), EC_ADDR(p_vaddr));
ret = ELFEDIT_CMDRET_MOD;
phdr->p_vaddr = p_vaddr;
}
}
break;
case PHDR_CMD_T_P_PADDR:
{
Addr p_paddr = elfedit_atoui(argstate.argv[1], NULL);
if (phdr->p_paddr == p_paddr) {
elfedit_msg(ELFEDIT_MSG_DEBUG,
MSG_INTL(MSG_DEBUG_LLX_OK),
argstate.ndx, MSG_ORIG(MSG_CMD_P_PADDR),
EC_ADDR(phdr->p_paddr));
} else {
elfedit_msg(ELFEDIT_MSG_DEBUG,
MSG_INTL(MSG_DEBUG_LLX_CHG),
argstate.ndx, MSG_ORIG(MSG_CMD_P_PADDR),
EC_ADDR(phdr->p_paddr), EC_ADDR(p_paddr));
ret = ELFEDIT_CMDRET_MOD;
phdr->p_paddr = p_paddr;
}
}
break;
case PHDR_CMD_T_P_FILESZ:
{
Xword p_filesz = elfedit_atoui(argstate.argv[1], NULL);
if (phdr->p_filesz == p_filesz) {
elfedit_msg(ELFEDIT_MSG_DEBUG,
MSG_INTL(MSG_DEBUG_LLX_OK),
argstate.ndx, MSG_ORIG(MSG_CMD_P_FILESZ),
EC_XWORD(phdr->p_filesz));
} else {
elfedit_msg(ELFEDIT_MSG_DEBUG,
MSG_INTL(MSG_DEBUG_LLX_CHG),
argstate.ndx, MSG_ORIG(MSG_CMD_P_FILESZ),
EC_XWORD(phdr->p_filesz),
EC_XWORD(p_filesz));
ret = ELFEDIT_CMDRET_MOD;
phdr->p_filesz = p_filesz;
}
}
break;
case PHDR_CMD_T_P_MEMSZ:
{
Xword p_memsz = elfedit_atoui(argstate.argv[1], NULL);
if (phdr->p_memsz == p_memsz) {
elfedit_msg(ELFEDIT_MSG_DEBUG,
MSG_INTL(MSG_DEBUG_LLX_OK),
argstate.ndx, MSG_ORIG(MSG_CMD_P_MEMSZ),
EC_XWORD(phdr->p_memsz));
} else {
elfedit_msg(ELFEDIT_MSG_DEBUG,
MSG_INTL(MSG_DEBUG_LLX_CHG),
argstate.ndx, MSG_ORIG(MSG_CMD_P_MEMSZ),
EC_XWORD(phdr->p_memsz),
EC_XWORD(p_memsz));
ret = ELFEDIT_CMDRET_MOD;
phdr->p_memsz = p_memsz;
}
}
break;
case PHDR_CMD_T_P_FLAGS:
{
Ehdr *ehdr = obj_state->os_ehdr;
uchar_t osabi = ehdr->e_ident[EI_OSABI];
Conv_phdr_flags_buf_t buf1, buf2;
Word p_flags = 0;
int i;
for (i = 1; i < argstate.argc; i++)
p_flags |=
(Word) elfedit_atoconst(argstate.argv[i],
ELFEDIT_CONST_PF);
if (argstate.optmask & PHDR_OPT_F_CMP)
p_flags = ~p_flags;
if (argstate.optmask & PHDR_OPT_F_AND)
p_flags &= phdr->p_flags;
else if (argstate.optmask & PHDR_OPT_F_OR)
p_flags |= phdr->p_flags;
if (phdr->p_flags == p_flags) {
elfedit_msg(ELFEDIT_MSG_DEBUG,
MSG_INTL(MSG_DEBUG_S_OK),
argstate.ndx, MSG_ORIG(MSG_CMD_P_FLAGS),
conv_phdr_flags(osabi, phdr->p_flags,
0, &buf1));
} else {
elfedit_msg(ELFEDIT_MSG_DEBUG,
MSG_INTL(MSG_DEBUG_S_CHG),
argstate.ndx, MSG_ORIG(MSG_CMD_P_FLAGS),
conv_phdr_flags(osabi, phdr->p_flags,
0, &buf1),
conv_phdr_flags(osabi, p_flags, 0, &buf2));
ret = ELFEDIT_CMDRET_MOD;
phdr->p_flags = p_flags;
}
}
break;
case PHDR_CMD_T_P_ALIGN:
{
Xword p_align = elfedit_atoui(argstate.argv[1], NULL);
if (phdr->p_align == p_align) {
elfedit_msg(ELFEDIT_MSG_DEBUG,
MSG_INTL(MSG_DEBUG_LLX_OK),
argstate.ndx, MSG_ORIG(MSG_CMD_P_ALIGN),
EC_XWORD(phdr->p_align));
} else {
elfedit_msg(ELFEDIT_MSG_DEBUG,
MSG_INTL(MSG_DEBUG_LLX_CHG),
argstate.ndx, MSG_ORIG(MSG_CMD_P_ALIGN),
EC_XWORD(phdr->p_align),
EC_XWORD(p_align));
ret = ELFEDIT_CMDRET_MOD;
phdr->p_align = p_align;
}
}
break;
case PHDR_CMD_T_INTERP:
ret = cmd_body_set_interp(&argstate);
break;
case PHDR_CMD_T_DELETE:
{
Word cnt = (argstate.argc == 1) ? 1 :
(Word) elfedit_atoui_range(argstate.argv[1],
MSG_ORIG(MSG_STR_COUNT), 1,
obj_state->os_phnum - argstate.ndx, NULL);
elfedit_array_elts_delete(MSG_ORIG(MSG_MOD_NAME),
obj_state->os_phdr, sizeof (Phdr),
obj_state->os_phnum, argstate.ndx, cnt);
do_autoprint = 0;
ret = ELFEDIT_CMDRET_MOD;
}
break;
case PHDR_CMD_T_MOVE:
{
Phdr save;
Word cnt;
Word dstndx;
do_autoprint = 0;
dstndx = (Word)
elfedit_atoui_range(argstate.argv[1],
MSG_ORIG(MSG_STR_DST_INDEX), 0,
obj_state->os_phnum - 1, NULL);
if (argstate.argc == 2) {
cnt = 1;
} else {
cnt = (Word) elfedit_atoui_range(
argstate.argv[2], MSG_ORIG(MSG_STR_COUNT),
1, obj_state->os_phnum, NULL);
}
elfedit_array_elts_move(MSG_ORIG(MSG_MOD_NAME),
obj_state->os_phdr, sizeof (save),
obj_state->os_phnum, argstate.ndx, dstndx,
cnt, &save);
ret = ELFEDIT_CMDRET_MOD;
}
break;
}
if (ret == ELFEDIT_CMDRET_MOD)
elfedit_modified_phdr(obj_state);
if (do_autoprint)
print_phdr(cmd, 1, &argstate);
return (ret);
}
static void
cpl_1starg_pt(elfedit_obj_state_t *obj_state, void *cpldata, int argc,
const char *argv[], int num_opt)
{
int i;
for (i = 0; i < num_opt; i++)
if (strcmp(MSG_ORIG(MSG_STR_MINUS_PHNDX), argv[i]) == 0)
return;
if (argc == (num_opt + 1))
elfedit_cpl_atoconst(cpldata, ELFEDIT_CONST_PT);
}
static void
cpl_p_type(elfedit_obj_state_t *obj_state, void *cpldata, int argc,
const char *argv[], int num_opt)
{
cpl_1starg_pt(obj_state, cpldata, argc, argv, num_opt);
if (argc == (num_opt + 2))
elfedit_cpl_atoconst(cpldata, ELFEDIT_CONST_PT);
}
static void
cpl_p_flags(elfedit_obj_state_t *obj_state, void *cpldata, int argc,
const char *argv[], int num_opt)
{
cpl_1starg_pt(obj_state, cpldata, argc, argv, num_opt);
if (argc >= (num_opt + 2))
elfedit_cpl_atoconst(cpldata, ELFEDIT_CONST_PF);
}
static elfedit_cmdret_t
cmd_dump(elfedit_obj_state_t *obj_state, int argc, const char *argv[])
{
return (cmd_body(PHDR_CMD_T_DUMP, obj_state, argc, argv));
}
static elfedit_cmdret_t
cmd_p_type(elfedit_obj_state_t *obj_state, int argc, const char *argv[])
{
return (cmd_body(PHDR_CMD_T_P_TYPE, obj_state, argc, argv));
}
static elfedit_cmdret_t
cmd_p_offset(elfedit_obj_state_t *obj_state, int argc, const char *argv[])
{
return (cmd_body(PHDR_CMD_T_P_OFFSET, obj_state, argc, argv));
}
static elfedit_cmdret_t
cmd_p_vaddr(elfedit_obj_state_t *obj_state, int argc, const char *argv[])
{
return (cmd_body(PHDR_CMD_T_P_VADDR, obj_state, argc, argv));
}
static elfedit_cmdret_t
cmd_p_paddr(elfedit_obj_state_t *obj_state, int argc, const char *argv[])
{
return (cmd_body(PHDR_CMD_T_P_PADDR, obj_state, argc, argv));
}
static elfedit_cmdret_t
cmd_p_filesz(elfedit_obj_state_t *obj_state, int argc, const char *argv[])
{
return (cmd_body(PHDR_CMD_T_P_FILESZ, obj_state, argc, argv));
}
static elfedit_cmdret_t
cmd_p_memsz(elfedit_obj_state_t *obj_state, int argc, const char *argv[])
{
return (cmd_body(PHDR_CMD_T_P_MEMSZ, obj_state, argc, argv));
}
static elfedit_cmdret_t
cmd_p_flags(elfedit_obj_state_t *obj_state, int argc, const char *argv[])
{
return (cmd_body(PHDR_CMD_T_P_FLAGS, obj_state, argc, argv));
}
static elfedit_cmdret_t
cmd_p_align(elfedit_obj_state_t *obj_state, int argc, const char *argv[])
{
return (cmd_body(PHDR_CMD_T_P_ALIGN, obj_state, argc, argv));
}
static elfedit_cmdret_t
cmd_interp(elfedit_obj_state_t *obj_state, int argc, const char *argv[])
{
return (cmd_body(PHDR_CMD_T_INTERP, obj_state, argc, argv));
}
static elfedit_cmdret_t
cmd_delete(elfedit_obj_state_t *obj_state, int argc, const char *argv[])
{
return (cmd_body(PHDR_CMD_T_DELETE, obj_state, argc, argv));
}
static elfedit_cmdret_t
cmd_move(elfedit_obj_state_t *obj_state, int argc, const char *argv[])
{
return (cmd_body(PHDR_CMD_T_MOVE, obj_state, argc, argv));
}
elfedit_module_t *
elfedit_init(elfedit_module_version_t version)
{
static elfedit_cmd_optarg_t opt_std[] = {
{ ELFEDIT_STDOA_OPT_O, 0,
ELFEDIT_CMDOA_F_INHERIT, 0, 0 },
{ MSG_ORIG(MSG_STR_MINUS_PHNDX),
ELFEDIT_I18NHDL(MSG_OPTDESC_PHNDX), 0,
PHDR_OPT_F_PHNDX, 0 },
{ NULL }
};
static elfedit_cmd_optarg_t opt_minus_phndx[] = {
{ MSG_ORIG(MSG_STR_MINUS_PHNDX),
ELFEDIT_I18NHDL(MSG_OPTDESC_PHNDX), 0,
PHDR_OPT_F_PHNDX, 0 },
{ NULL }
};
static const char *name_dump[] = {
MSG_ORIG(MSG_CMD_DUMP),
MSG_ORIG(MSG_STR_EMPTY),
NULL
};
static elfedit_cmd_optarg_t arg_dump[] = {
{ MSG_ORIG(MSG_STR_ELEMENT),
ELFEDIT_I18NHDL(MSG_A1_ELEMENT),
ELFEDIT_CMDOA_F_OPT },
{ NULL }
};
static const char *name_p_type[] = { MSG_ORIG(MSG_CMD_P_TYPE), NULL };
static elfedit_cmd_optarg_t arg_p_type[] = {
{ MSG_ORIG(MSG_STR_ELEMENT),
ELFEDIT_I18NHDL(MSG_A1_ELEMENT),
ELFEDIT_CMDOA_F_OPT },
{ MSG_ORIG(MSG_STR_TYPE),
ELFEDIT_I18NHDL(MSG_A2_P_TYPE_TYPE),
ELFEDIT_CMDOA_F_OPT },
{ NULL }
};
static const char *name_p_offset[] = { MSG_ORIG(MSG_CMD_P_OFFSET),
NULL };
static elfedit_cmd_optarg_t arg_p_offset[] = {
{ MSG_ORIG(MSG_STR_ELEMENT),
ELFEDIT_I18NHDL(MSG_A1_ELEMENT),
ELFEDIT_CMDOA_F_OPT },
{ MSG_ORIG(MSG_STR_VALUE),
ELFEDIT_I18NHDL(MSG_A2_P_OFFSET_VALUE),
ELFEDIT_CMDOA_F_OPT },
{ NULL }
};
static const char *name_p_vaddr[] = { MSG_ORIG(MSG_CMD_P_VADDR),
NULL };
static elfedit_cmd_optarg_t arg_p_vaddr[] = {
{ MSG_ORIG(MSG_STR_ELEMENT),
ELFEDIT_I18NHDL(MSG_A1_ELEMENT),
ELFEDIT_CMDOA_F_OPT },
{ MSG_ORIG(MSG_STR_ADDR),
ELFEDIT_I18NHDL(MSG_A2_P_VADDR_ADDR),
ELFEDIT_CMDOA_F_OPT },
{ NULL }
};
static const char *name_p_paddr[] = { MSG_ORIG(MSG_CMD_P_PADDR),
NULL };
static elfedit_cmd_optarg_t arg_p_paddr[] = {
{ MSG_ORIG(MSG_STR_ELEMENT),
ELFEDIT_I18NHDL(MSG_A1_ELEMENT),
ELFEDIT_CMDOA_F_OPT },
{ MSG_ORIG(MSG_STR_ADDR),
ELFEDIT_I18NHDL(MSG_A2_P_PADDR_ADDR),
ELFEDIT_CMDOA_F_OPT },
{ NULL }
};
static const char *name_p_filesz[] = { MSG_ORIG(MSG_CMD_P_FILESZ),
NULL };
static elfedit_cmd_optarg_t arg_p_filesz[] = {
{ MSG_ORIG(MSG_STR_ELEMENT), ELFEDIT_I18NHDL(MSG_A1_ELEMENT),
ELFEDIT_CMDOA_F_OPT },
{ MSG_ORIG(MSG_STR_SIZE),
ELFEDIT_I18NHDL(MSG_A2_P_FILESZ_SIZE),
ELFEDIT_CMDOA_F_OPT },
{ NULL }
};
static const char *name_p_memsz[] = { MSG_ORIG(MSG_CMD_P_MEMSZ),
NULL };
static elfedit_cmd_optarg_t arg_p_memsz[] = {
{ MSG_ORIG(MSG_STR_ELEMENT),
ELFEDIT_I18NHDL(MSG_A1_ELEMENT),
ELFEDIT_CMDOA_F_OPT },
{ MSG_ORIG(MSG_STR_SIZE),
ELFEDIT_I18NHDL(MSG_A2_P_MEMSZ_SIZE),
ELFEDIT_CMDOA_F_OPT },
{ NULL }
};
static const char *name_p_flags[] = {
MSG_ORIG(MSG_CMD_P_FLAGS), NULL };
static elfedit_cmd_optarg_t opt_p_flags[] = {
{ ELFEDIT_STDOA_OPT_AND, 0,
ELFEDIT_CMDOA_F_INHERIT, PHDR_OPT_F_AND, PHDR_OPT_F_OR },
{ ELFEDIT_STDOA_OPT_CMP, 0,
ELFEDIT_CMDOA_F_INHERIT, PHDR_OPT_F_CMP, 0 },
{ MSG_ORIG(MSG_STR_MINUS_PHNDX),
ELFEDIT_I18NHDL(MSG_OPTDESC_PHNDX), 0,
PHDR_OPT_F_PHNDX, 0 },
{ ELFEDIT_STDOA_OPT_O, 0,
ELFEDIT_CMDOA_F_INHERIT, 0, 0 },
{ ELFEDIT_STDOA_OPT_OR, 0,
ELFEDIT_CMDOA_F_INHERIT, PHDR_OPT_F_OR, PHDR_OPT_F_AND },
{ NULL }
};
static elfedit_cmd_optarg_t arg_p_flags[] = {
{ MSG_ORIG(MSG_STR_ELEMENT),
ELFEDIT_I18NHDL(MSG_A1_ELEMENT),
ELFEDIT_CMDOA_F_OPT },
{ MSG_ORIG(MSG_STR_VALUE),
ELFEDIT_I18NHDL(MSG_A2_P_FLAGS_VALUE),
ELFEDIT_CMDOA_F_OPT | ELFEDIT_CMDOA_F_MULT },
{ NULL }
};
static const char *name_p_align[] = { MSG_ORIG(MSG_CMD_P_ALIGN),
NULL };
static elfedit_cmd_optarg_t arg_p_align[] = {
{ MSG_ORIG(MSG_STR_ELEMENT),
ELFEDIT_I18NHDL(MSG_A1_ELEMENT),
ELFEDIT_CMDOA_F_OPT },
{ MSG_ORIG(MSG_STR_ALIGN),
ELFEDIT_I18NHDL(MSG_A2_P_ALIGN_ALIGN),
ELFEDIT_CMDOA_F_OPT },
{ NULL }
};
static const char *name_interp[] = { MSG_ORIG(MSG_CMD_INTERP), NULL };
static elfedit_cmd_optarg_t opt_interp[] = {
{ ELFEDIT_STDOA_OPT_O, 0,
ELFEDIT_CMDOA_F_INHERIT, 0, 0 },
{ NULL }
};
static elfedit_cmd_optarg_t arg_interp[] = {
{ MSG_ORIG(MSG_STR_NEWPATH),
ELFEDIT_I18NHDL(MSG_A1_INTERP_NEWPATH),
ELFEDIT_CMDOA_F_OPT },
{ NULL }
};
static const char *name_delete[] = { MSG_ORIG(MSG_CMD_DELETE), NULL };
static elfedit_cmd_optarg_t arg_delete[] = {
{ MSG_ORIG(MSG_STR_ELEMENT),
ELFEDIT_I18NHDL(MSG_A1_ELEMENT),
0 },
{ MSG_ORIG(MSG_STR_COUNT),
ELFEDIT_I18NHDL(MSG_A2_DELETE_COUNT),
ELFEDIT_CMDOA_F_OPT },
{ NULL }
};
static const char *name_move[] = { MSG_ORIG(MSG_CMD_MOVE), NULL };
static elfedit_cmd_optarg_t arg_move[] = {
{ MSG_ORIG(MSG_STR_ELEMENT),
ELFEDIT_I18NHDL(MSG_A1_ELEMENT),
ELFEDIT_CMDOA_F_OPT },
{ MSG_ORIG(MSG_STR_DST_INDEX),
ELFEDIT_I18NHDL(MSG_A2_MOVE_DST_INDEX),
0 },
{ MSG_ORIG(MSG_STR_COUNT),
ELFEDIT_I18NHDL(MSG_A3_MOVE_COUNT),
ELFEDIT_CMDOA_F_OPT },
{ NULL }
};
static elfedit_cmd_t cmds[] = {
{ cmd_dump, cpl_1starg_pt, name_dump,
ELFEDIT_I18NHDL(MSG_DESC_DUMP),
ELFEDIT_I18NHDL(MSG_HELP_DUMP),
opt_minus_phndx, arg_dump },
{ cmd_p_type, cpl_p_type, name_p_type,
ELFEDIT_I18NHDL(MSG_DESC_P_TYPE),
ELFEDIT_I18NHDL(MSG_HELP_P_TYPE),
opt_std, arg_p_type },
{ cmd_p_offset, cpl_1starg_pt, name_p_offset,
ELFEDIT_I18NHDL(MSG_DESC_P_OFFSET),
ELFEDIT_I18NHDL(MSG_HELP_P_OFFSET),
opt_std, arg_p_offset },
{ cmd_p_vaddr, cpl_1starg_pt, name_p_vaddr,
ELFEDIT_I18NHDL(MSG_DESC_P_VADDR),
ELFEDIT_I18NHDL(MSG_HELP_P_VADDR),
opt_std, arg_p_vaddr },
{ cmd_p_paddr, cpl_1starg_pt, name_p_paddr,
ELFEDIT_I18NHDL(MSG_DESC_P_PADDR),
ELFEDIT_I18NHDL(MSG_HELP_P_PADDR),
opt_std, arg_p_paddr },
{ cmd_p_filesz, cpl_1starg_pt, name_p_filesz,
ELFEDIT_I18NHDL(MSG_DESC_P_FILESZ),
ELFEDIT_I18NHDL(MSG_HELP_P_FILESZ),
opt_std, arg_p_filesz },
{ cmd_p_memsz, cpl_1starg_pt, name_p_memsz,
ELFEDIT_I18NHDL(MSG_DESC_P_MEMSZ),
ELFEDIT_I18NHDL(MSG_HELP_P_MEMSZ),
opt_std, arg_p_memsz },
{ cmd_p_flags, cpl_p_flags, name_p_flags,
ELFEDIT_I18NHDL(MSG_DESC_P_FLAGS),
ELFEDIT_I18NHDL(MSG_HELP_P_FLAGS),
opt_p_flags, arg_p_flags },
{ cmd_p_align, cpl_1starg_pt, name_p_align,
ELFEDIT_I18NHDL(MSG_DESC_P_ALIGN),
ELFEDIT_I18NHDL(MSG_HELP_P_ALIGN),
opt_std, arg_p_align },
{ cmd_interp, NULL, name_interp,
ELFEDIT_I18NHDL(MSG_DESC_INTERP),
ELFEDIT_I18NHDL(MSG_HELP_INTERP),
opt_interp, arg_interp },
{ cmd_delete, cpl_1starg_pt, name_delete,
ELFEDIT_I18NHDL(MSG_DESC_DELETE),
ELFEDIT_I18NHDL(MSG_HELP_DELETE),
opt_minus_phndx, arg_delete },
{ cmd_move, cpl_1starg_pt, name_move,
ELFEDIT_I18NHDL(MSG_DESC_MOVE),
ELFEDIT_I18NHDL(MSG_HELP_MOVE),
opt_minus_phndx, arg_move },
{ NULL }
};
static elfedit_module_t module = {
ELFEDIT_VER_CURRENT, MSG_ORIG(MSG_MOD_NAME),
ELFEDIT_I18NHDL(MSG_MOD_DESC),
cmds, mod_i18nhdl_to_str };
return (&module);
}