/***************************************************************/
/* perk: PE Resource Kit */
/* Copyright (C) 2015--2025 SysDeer Technologies, LLC */
/* Released under GPLv2 and GPLv3; see COPYING.PERK. */
/***************************************************************/
#include <perk/perk.h>
#include "perk_driver_impl.h"
#include "perk_errinfo_impl.h"
#include "perk_ar_impl.h"
int pe_ar_list_members(const struct pe_archive_meta *, const char **);
static int pe_cmd_ar_perform_unit_actions(
const struct pe_driver_ctx * dctx,
const char * arname,
const char ** members,
uint64_t action)
{
struct pe_unit_ctx * arctx = 0;
int (*pe_ar_fn)(const struct pe_archive_meta *,const char **);
if (pe_lib_get_unit_ctx(dctx,arname,&arctx) < 0)
return PERK_NESTED_ERROR(dctx);
if (arctx->armeta == 0)
return PERK_CUSTOM_ERROR(dctx,
PERK_ERR_AR_NON_ARCHIVE_IMAGE);
if (action == PERK_DRIVER_AR_LIST_MEMBERS) {
pe_ar_fn = pe_ar_list_members;
}
if (pe_ar_fn(arctx->armeta,members) < 0) {
pe_lib_free_unit_ctx(arctx);
return PERK_NESTED_ERROR(dctx);
}
pe_lib_free_unit_ctx(arctx);
return 0;
}
static int pe_cmd_ar_verify_cmdline(
const struct pe_driver_ctx * dctx,
uint64_t flags,
const char * posname,
const char * arname,
const char ** members)
{
uint64_t action;
uint64_t poscmd;
uint64_t vercmd;
action = (flags & AR_ACTION_MASK);
poscmd = (flags & AR_POSNAME_MASK);
vercmd = (flags & PERK_DRIVER_VERSION);
if (vercmd && !posname && !arname && !members)
return 0;
switch (action) {
case 0:
return PERK_CUSTOM_ERROR(dctx,
PERK_ERR_AR_MISSING_ACTION);
case PERK_DRIVER_AR_LIST_MEMBERS:
case PERK_DRIVER_AR_DELETE_MEMBERS:
case PERK_DRIVER_AR_APPEND_MEMBERS:
case PERK_DRIVER_AR_EXTRACT_MEMBERS:
case PERK_DRIVER_AR_PRINT_ARCHIVE:
if (poscmd || posname)
return PERK_CUSTOM_ERROR(dctx,
PERK_ERR_AR_INVALID_ANCHORS);
break;
case AR_UPDATE_MASK:
case PERK_DRIVER_AR_REPLACE_MEMBERS:
case PERK_DRIVER_AR_MOVE_MEMBERS:
switch (poscmd) {
case 0:
if (posname)
return PERK_CUSTOM_ERROR(dctx,
PERK_ERR_AR_MISSING_ANCHOR);
break;
case PERK_DRIVER_AR_POSITION_AFTER:
case PERK_DRIVER_AR_POSITION_BEFORE:
if (!posname)
return PERK_CUSTOM_ERROR(dctx,
PERK_ERR_AR_NULL_POSNAME);
break;
default:
return PERK_CUSTOM_ERROR(dctx,
PERK_ERR_AR_MULTIPLE_ANCHORS);
}
default:
return PERK_CUSTOM_ERROR(dctx,
PERK_ERR_AR_MULTIPLE_ACTIONS);
}
if (!arname)
return PERK_CUSTOM_ERROR(dctx,
PERK_ERR_AR_NULL_ARNAME);
return 0;
}
int pe_cmd_ar(
const struct pe_driver_ctx * dctx,
uint64_t flags,
const char * posname,
const char * arname,
const char ** members)
{
uint64_t action = (flags & AR_ACTION_MASK);
if (pe_cmd_ar_verify_cmdline(dctx,flags,posname,arname,members) < 0)
return PERK_NESTED_ERROR(dctx);
if (pe_cmd_ar_perform_unit_actions(dctx,arname,members,action) < 0)
return PERK_NESTED_ERROR(dctx);
return 0;
}