/* plt-findfre-2.c -- Test for sframe_find_fre for SFrame FDE of type
PCMASK with with one SFrame FRE only.
Copyright (C) 2025 Free Software Foundation, Inc.
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation; either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see . */
#include "config.h"
#include
#include
#include
#include "sframe-api.h"
/* DejaGnu should not use gnulib's vsnprintf replacement here. */
#undef vsnprintf
#include
/* s390x-specific size in bytes of plt0 and pltN. */
#define PLT_SIZE 32
/* Magic values added to CFA offsets to make them distingishable. Must
be multiple of 8 due to s390x-specific CFA alignment factor. */
#define PLT0_CFA_OFFSET_MAGIC 0
#define PLTN_CFA_OFFSET_MAGIC 8
static int
add_plt0_fde (sframe_encoder_ctx *ectx, uint32_t plt_vaddr,
uint32_t sframe_vaddr, int idx)
{
/* 1 single FRE. */
sframe_frame_row_entry fre
= { 0x0,
{ SFRAME_V2_S390X_CFA_OFFSET_ENCODE (160 + PLT0_CFA_OFFSET_MAGIC) },
SFRAME_V1_FRE_INFO (SFRAME_BASE_REG_SP, 1, SFRAME_FRE_OFFSET_1B) };
unsigned char finfo = sframe_fde_create_func_info (SFRAME_FRE_TYPE_ADDR1,
SFRAME_FDE_TYPE_PCINC);
uint32_t offsetof_fde_in_sec
= sframe_encoder_get_offsetof_fde_start_addr (ectx, idx, NULL);
int32_t func_start_addr = (plt_vaddr
- (sframe_vaddr + offsetof_fde_in_sec));
/* 1 PCINC-type FDE for 1 plt0 entry of 32 bytes. */
int err = sframe_encoder_add_funcdesc_v2 (ectx, func_start_addr,
PLT_SIZE /* func size. */,
finfo,
0 /* rep block size. */,
1 /* num FREs. */);
if (err == -1)
return err;
if (sframe_encoder_add_fre (ectx, idx, &fre) == SFRAME_ERR)
return -1;
return 0;
}
static int
add_pltn_fde (sframe_encoder_ctx *ectx, uint32_t plt_vaddr,
uint32_t sframe_vaddr, int idx)
{
/* 1 single FRE. */
sframe_frame_row_entry fre
= { 0x0,
{ SFRAME_V2_S390X_CFA_OFFSET_ENCODE (160 + PLTN_CFA_OFFSET_MAGIC) },
SFRAME_V1_FRE_INFO (SFRAME_BASE_REG_SP, 1, SFRAME_FRE_OFFSET_1B) };
unsigned char finfo = sframe_fde_create_func_info (SFRAME_FRE_TYPE_ADDR1,
SFRAME_FDE_TYPE_PCMASK);
uint32_t offsetof_fde_in_sec
= sframe_encoder_get_offsetof_fde_start_addr (ectx, idx, NULL);
int32_t func_start_addr = (plt_vaddr
- (sframe_vaddr + offsetof_fde_in_sec));
/* 1 PCMASK-type FDE for 5 pltN entries of 32 bytes each. */
int err = sframe_encoder_add_funcdesc_v2 (ectx, func_start_addr,
5 * PLT_SIZE /* func size. */,
finfo,
PLT_SIZE /* rep block size. */,
1 /* num FREs. */);
if (err == -1)
return err;
if (sframe_encoder_add_fre (ectx, idx, &fre) == SFRAME_ERR)
return -1;
return 0;
}
static
void test_plt_findfre (const char suffix, const uint32_t plt_vaddr,
const uint32_t sframe_vaddr)
{
sframe_encoder_ctx *ectx;
sframe_decoder_ctx *dctx;
sframe_frame_row_entry frep;
char *sframe_buf;
size_t sf_size;
int err = 0;
unsigned int fde_cnt = 0;
int i;
#define TEST(cond, ...) \
do \
{ \
if (cond) \
pass (__VA_ARGS__); \
else \
fail (__VA_ARGS__); \
} \
while (0)
ectx = sframe_encode (SFRAME_VERSION, SFRAME_F_FDE_FUNC_START_PCREL,
SFRAME_ABI_S390X_ENDIAN_BIG,
SFRAME_CFA_FIXED_FP_INVALID,
SFRAME_CFA_FIXED_RA_INVALID,
&err);
TEST (ectx != NULL && err == 0, "plt-findfre-2%c: Creating SFrame encoder", suffix);
err = add_plt0_fde (ectx, plt_vaddr, sframe_vaddr, 0);
TEST (err == 0, "plt-findfre-2%c: Adding FDE for plt0", suffix);
fde_cnt = sframe_encoder_get_num_fidx (ectx);
TEST (fde_cnt == 1, "plt-findfre-2%c: Test FDE count after adding FDE for plt0", suffix);
err = add_pltn_fde (ectx, plt_vaddr + PLT_SIZE, sframe_vaddr, 1);
TEST (err == 0, "plt-findfre-2%c: Adding FDE for pltN", suffix);
fde_cnt = sframe_encoder_get_num_fidx (ectx);
TEST (fde_cnt == 2, "plt-findfre-2%c: Test FDE count after adding FDE for pltN", suffix);
sframe_buf = sframe_encoder_write (ectx, &sf_size, &err);
TEST (err == 0, "plt-findfre-2%c: Encoder write", suffix);
dctx = sframe_decode (sframe_buf, sf_size, &err);
TEST (dctx != NULL, "plt-findfre-2%c: Decoder setup", suffix);
/* Find the only FRE in PLT0 at offset 0. */
err = sframe_find_fre (dctx, (plt_vaddr + 0 - sframe_vaddr), &frep);
TEST (err == 0 && sframe_fre_get_cfa_offset (dctx, &frep, &err) == 160 + PLT0_CFA_OFFSET_MAGIC,
"plt-findfre-2%c: Find only FRE in PLT0 at offset 0", suffix);
/* Find the only FRE in PLT0 at offset PLT_SIZE-1. */
err = sframe_find_fre (dctx, (plt_vaddr + (PLT_SIZE-1) - sframe_vaddr), &frep);
TEST (err == 0 && sframe_fre_get_cfa_offset (dctx, &frep, &err) == 160 + PLT0_CFA_OFFSET_MAGIC,
"plt-findfre-2%c: Find only FRE in PLT0 at offset PLT_SIZE-1", suffix);
/* Find the only FRE in PLT1-5 at offset 0 and PLT_SIZE-1. */
for (i = 1; i < 5; i++)
{
/* Find the only FRE in PLTN at offset 0. */
err = sframe_find_fre (dctx, (plt_vaddr + i * PLT_SIZE + 0 - sframe_vaddr), &frep);
TEST (err == 0 && sframe_fre_get_cfa_offset (dctx, &frep, &err) == 160 + PLTN_CFA_OFFSET_MAGIC,
"plt-findfre-2%c: Find only FRE in PLT%d at offset 0", suffix, i);
/* Find the only FRE in PLTN at offset 31. */
err = sframe_find_fre (dctx, (plt_vaddr + i * PLT_SIZE + (PLT_SIZE-1) - sframe_vaddr), &frep);
TEST (err == 0 && sframe_fre_get_cfa_offset (dctx, &frep, &err) == 160 + PLTN_CFA_OFFSET_MAGIC,
"plt-findfre-2%c: Find only FRE in PLT%d at offset PLT_SIZE-1", suffix, i);
}
/* Find no FRE in non-existing PLT6 at offset 0. */
err = sframe_find_fre (dctx, (plt_vaddr + 6 * PLT_SIZE + 0 - sframe_vaddr), &frep);
TEST (err != 0, "plt-findfre-2%c: Find no FRE in out of range PLT6 at offset 0", suffix);
sframe_encoder_free (&ectx);
sframe_decoder_free (&dctx);
}
int
main (void)
{
uint32_t sframe_vaddr = 0x402220;
uint32_t plt_vaddr = 0x401020;
printf ("plt-findfre-2a: Testing with plt_vaddr = %#x; sframe_vaddr = %#x\n",
plt_vaddr, sframe_vaddr);
test_plt_findfre ('a', plt_vaddr, sframe_vaddr);
sframe_vaddr = 0x401020;
plt_vaddr = 0x402220;
printf ("plt-findfre-2b: Testing with plt_vaddr = %#x; sframe_vaddr = %#x\n",
plt_vaddr, sframe_vaddr);
test_plt_findfre ('b', plt_vaddr, sframe_vaddr);
return 0;
}