blob: f83317ea0503f87a9f312802db5994e2845fe490 [file] [log] [blame]
/*
* Copyright (c) 2013-2017, Intel Corporation
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are met:
*
* * Redistributions of source code must retain the above copyright notice,
* this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above copyright notice,
* this list of conditions and the following disclaimer in the documentation
* and/or other materials provided with the distribution.
* * Neither the name of Intel Corporation nor the names of its contributors
* may be used to endorse or promote products derived from this software
* without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
* IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE
* LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
* POSSIBILITY OF SUCH DAMAGE.
*/
#include "ptunit.h"
#include "pt_image.h"
#include "pt_section.h"
#include "pt_mapped_section.h"
#include "intel-pt.h"
struct image_fixture;
/* A test mapping. */
struct ifix_mapping {
/* The contents. */
uint8_t content[0x10];
/* The size - between 0 and sizeof(content). */
uint64_t size;
/* An artificial error code to be injected into pt_section_read().
*
* If @errcode is non-zero, pt_section_read() fails with @errcode.
*/
int errcode;
};
/* A test file status - turned into a section status. */
struct ifix_status {
/* Delete indication:
* - zero if initialized and not (yet) deleted
* - non-zero if deleted and not (re-)initialized
*/
int deleted;
/* Put with use-count of zero indication. */
int bad_put;
/* The test mapping to be used. */
struct ifix_mapping *mapping;
/* A link back to the test fixture providing this section. */
struct image_fixture *ifix;
};
enum {
ifix_nsecs = 5
};
/* A fake image section cache. */
struct pt_image_section_cache {
/* The cached sections. */
struct pt_section *section[ifix_nsecs];
/* Their load addresses. */
uint64_t laddr[ifix_nsecs];
/* The number of used sections. */
int nsecs;
};
/* A test fixture providing an image, test sections, and asids. */
struct image_fixture {
/* The image. */
struct pt_image image;
/* The test states. */
struct ifix_status status[ifix_nsecs];
/* The test mappings. */
struct ifix_mapping mapping[ifix_nsecs];
/* The sections. */
struct pt_section section[ifix_nsecs];
/* The asids. */
struct pt_asid asid[3];
/* The number of used sections/mappings/states. */
int nsecs;
/* An initially empty image as destination for image copies. */
struct pt_image copy;
/* A test section cache. */
struct pt_image_section_cache iscache;
/* The test fixture initialization and finalization functions. */
struct ptunit_result (*init)(struct image_fixture *);
struct ptunit_result (*fini)(struct image_fixture *);
};
static void ifix_init_section(struct pt_section *section, char *filename,
struct ifix_status *status,
struct ifix_mapping *mapping,
struct image_fixture *ifix)
{
uint8_t i;
memset(section, 0, sizeof(*section));
section->filename = filename;
section->status = status;
section->size = mapping->size = sizeof(mapping->content);
section->offset = 0x10;
for (i = 0; i < mapping->size; ++i)
mapping->content[i] = i;
status->deleted = 0;
status->bad_put = 0;
status->mapping = mapping;
status->ifix = ifix;
}
static int ifix_add_section(struct image_fixture *ifix, char *filename)
{
int index;
if (!ifix)
return -pte_internal;
index = ifix->nsecs;
if (ifix_nsecs <= index)
return -pte_internal;
ifix_init_section(&ifix->section[index], filename, &ifix->status[index],
&ifix->mapping[index], ifix);
ifix->nsecs += 1;
return index;
}
static int ifix_cache_section(struct image_fixture *ifix,
struct pt_section *section, uint64_t laddr)
{
int index;
if (!ifix)
return -pte_internal;
index = ifix->iscache.nsecs;
if (ifix_nsecs <= index)
return -pte_internal;
ifix->iscache.section[index] = section;
ifix->iscache.laddr[index] = laddr;
index += 1;
ifix->iscache.nsecs = index;
return index;
}
int pt_section_clone(struct pt_section **clone,
const struct pt_section *section, uint64_t offset,
uint64_t size)
{
struct image_fixture *ifix;
struct ifix_mapping *mapping, *smapping;
struct ifix_status *status;
uint64_t begin, end, sbegin, send, start;
int index;
if (!clone || !section)
return -pte_internal;
status = section->status;
if (!status || status->deleted)
return -pte_internal;
begin = offset;
end = begin + size;
if (end <= begin)
return -pte_bad_image;
sbegin = pt_section_offset(section);
send = sbegin + pt_section_size(section);
if (send <= sbegin)
return -pte_internal;
if ((begin < sbegin) || (send < end))
return -pte_invalid;
start = begin - sbegin;
ifix = status->ifix;
if (!ifix)
return -pte_internal;
index = ifix_add_section(ifix, section->filename);
if (index < 0)
return index;
mapping = &ifix->mapping[index];
mapping->size = size;
smapping = status->mapping;
if (!smapping)
return -pte_internal;
memset(mapping->content, 0xcd, sizeof(mapping->content));
memcpy(mapping->content, &smapping->content[start], (size_t) size);
ifix->section[index].size = size;
ifix->section[index].offset = offset;
ifix->section[index].ucount = 1;
*clone = &ifix->section[index];
return 0;
}
const char *pt_section_filename(const struct pt_section *section)
{
if (!section)
return NULL;
return section->filename;
}
uint64_t pt_section_offset(const struct pt_section *section)
{
if (!section)
return 0ull;
return section->offset;
}
uint64_t pt_section_size(const struct pt_section *section)
{
if (!section)
return 0ull;
return section->size;
}
struct pt_section *pt_mk_section(const char *file, uint64_t offset,
uint64_t size)
{
(void) file;
(void) offset;
(void) size;
/* This function is not used by our tests. */
return NULL;
}
int pt_section_get(struct pt_section *section)
{
if (!section)
return -pte_internal;
section->ucount += 1;
return 0;
}
int pt_section_put(struct pt_section *section)
{
struct ifix_status *status;
uint16_t ucount;
if (!section)
return -pte_internal;
status = section->status;
if (!status)
return -pte_internal;
ucount = section->ucount;
if (!ucount) {
status->bad_put += 1;
return -pte_internal;
}
ucount = --section->ucount;
if (!ucount) {
status->deleted += 1;
if (status->deleted > 1)
return -pte_internal;
}
return 0;
}
int pt_iscache_lookup(struct pt_image_section_cache *iscache,
struct pt_section **section, uint64_t *laddr, int isid)
{
if (!iscache || !section || !laddr)
return -pte_internal;
if (!isid || iscache->nsecs < isid)
return -pte_bad_image;
isid -= 1;
*section = iscache->section[isid];
*laddr = iscache->laddr[isid];
return pt_section_get(*section);
}
static int ifix_unmap(struct pt_section *section)
{
uint16_t mcount;
if (!section)
return -pte_internal;
mcount = section->mcount;
if (!mcount)
return -pte_internal;
if (!section->mapping)
return -pte_internal;
mcount = --section->mcount;
if (!mcount)
section->mapping = NULL;
return 0;
}
static int ifix_read(const struct pt_section *section, uint8_t *buffer,
uint16_t size, uint64_t offset)
{
struct ifix_mapping *mapping;
uint64_t begin, end;
if (!section || !buffer)
return -pte_internal;
begin = offset;
end = begin + size;
if (end < begin)
return -pte_nomap;
mapping = section->mapping;
if (!mapping)
return -pte_nomap;
if (mapping->errcode)
return mapping->errcode;
if (mapping->size <= begin)
return -pte_nomap;
if (mapping->size < end) {
end = mapping->size;
size = (uint16_t) (end - begin);
}
memcpy(buffer, &mapping->content[begin], size);
return size;
}
int pt_section_map(struct pt_section *section)
{
struct ifix_status *status;
uint16_t mcount;
if (!section)
return -pte_internal;
mcount = section->mcount++;
if (mcount)
return 0;
if (section->mapping)
return -pte_internal;
status = section->status;
if (!status)
return -pte_internal;
section->mapping = status->mapping;
section->unmap = ifix_unmap;
section->read = ifix_read;
return 0;
}
int pt_section_unmap(struct pt_section *section)
{
if (!section)
return -pte_internal;
if (!section->unmap)
return -pte_nomap;
return section->unmap(section);
}
int pt_section_read(const struct pt_section *section, uint8_t *buffer,
uint16_t size, uint64_t offset)
{
if (!section)
return -pte_internal;
if (!section->read)
return -pte_nomap;
return section->read(section, buffer, size, offset);
}
/* A test read memory callback. */
static int image_readmem_callback(uint8_t *buffer, size_t size,
const struct pt_asid *asid,
uint64_t ip, void *context)
{
const uint8_t *memory;
size_t idx;
(void) asid;
if (!buffer)
return -pte_invalid;
/* We use a constant offset of 0x3000. */
if (ip < 0x3000ull)
return -pte_nomap;
ip -= 0x3000ull;
memory = (const uint8_t *) context;
if (!memory)
return -pte_internal;
for (idx = 0; idx < size; ++idx)
buffer[idx] = memory[ip + idx];
return (int) idx;
}
static struct ptunit_result init(void)
{
struct pt_image image;
memset(&image, 0xcd, sizeof(image));
pt_image_init(&image, NULL);
ptu_null(image.name);
ptu_null(image.sections);
ptu_null((void *) (uintptr_t) image.readmem.callback);
ptu_null(image.readmem.context);
return ptu_passed();
}
static struct ptunit_result init_name(struct image_fixture *ifix)
{
memset(&ifix->image, 0xcd, sizeof(ifix->image));
pt_image_init(&ifix->image, "image-name");
ptu_str_eq(ifix->image.name, "image-name");
ptu_null(ifix->image.sections);
ptu_null((void *) (uintptr_t) ifix->image.readmem.callback);
ptu_null(ifix->image.readmem.context);
return ptu_passed();
}
static struct ptunit_result init_null(void)
{
pt_image_init(NULL, NULL);
return ptu_passed();
}
static struct ptunit_result fini(void)
{
struct ifix_mapping mapping;
struct ifix_status status;
struct pt_section section;
struct pt_image image;
struct pt_asid asid;
int errcode;
pt_asid_init(&asid);
ifix_init_section(&section, NULL, &status, &mapping, NULL);
pt_image_init(&image, NULL);
errcode = pt_image_add(&image, &section, &asid, 0x0ull, 0);
ptu_int_eq(errcode, 0);
pt_image_fini(&image);
ptu_int_eq(section.ucount, 0);
ptu_int_eq(section.mcount, 0);
ptu_int_eq(status.deleted, 1);
ptu_int_eq(status.bad_put, 0);
return ptu_passed();
}
static struct ptunit_result fini_empty(void)
{
struct pt_image image;
pt_image_init(&image, NULL);
pt_image_fini(&image);
return ptu_passed();
}
static struct ptunit_result fini_null(void)
{
pt_image_fini(NULL);
return ptu_passed();
}
static struct ptunit_result name(struct image_fixture *ifix)
{
const char *name;
pt_image_init(&ifix->image, "image-name");
name = pt_image_name(&ifix->image);
ptu_str_eq(name, "image-name");
return ptu_passed();
}
static struct ptunit_result name_none(void)
{
struct pt_image image;
const char *name;
pt_image_init(&image, NULL);
name = pt_image_name(&image);
ptu_null(name);
return ptu_passed();
}
static struct ptunit_result name_null(void)
{
const char *name;
name = pt_image_name(NULL);
ptu_null(name);
return ptu_passed();
}
static struct ptunit_result read_empty(struct image_fixture *ifix)
{
struct pt_asid asid;
uint8_t buffer[] = { 0xcc, 0xcc };
int status, isid;
pt_asid_init(&asid);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, sizeof(buffer),
&asid, 0x1000ull);
ptu_int_eq(status, -pte_nomap);
ptu_int_eq(isid, -1);
ptu_uint_eq(buffer[0], 0xcc);
ptu_uint_eq(buffer[1], 0xcc);
return ptu_passed();
}
static struct ptunit_result overlap_front(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc };
int status, isid;
status = pt_image_add(&ifix->image, &ifix->section[0], &ifix->asid[0],
0x1001ull, 1);
ptu_int_eq(status, 0);
status = pt_image_add(&ifix->image, &ifix->section[1], &ifix->asid[0],
0x1000ull, 2);
ptu_int_eq(status, 0);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[0],
0x1010ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 1);
ptu_uint_eq(buffer[0], 0x0f);
ptu_uint_eq(buffer[1], 0xcc);
buffer[0] = 0xcc;
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[0],
0x100full);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 2);
ptu_uint_eq(buffer[0], 0x0f);
ptu_uint_eq(buffer[1], 0xcc);
return ptu_passed();
}
static struct ptunit_result overlap_back(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc };
int status, isid;
status = pt_image_add(&ifix->image, &ifix->section[0], &ifix->asid[0],
0x1000ull, 1);
ptu_int_eq(status, 0);
status = pt_image_add(&ifix->image, &ifix->section[1], &ifix->asid[0],
0x1001ull, 2);
ptu_int_eq(status, 0);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[0],
0x1000ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 1);
ptu_uint_eq(buffer[0], 0x00);
ptu_uint_eq(buffer[1], 0xcc);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[0],
0x1010ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 2);
ptu_uint_eq(buffer[0], 0x0f);
ptu_uint_eq(buffer[1], 0xcc);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 1, &ifix->asid[0],
0x1001ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 2);
ptu_uint_eq(buffer[0], 0x00);
ptu_uint_eq(buffer[1], 0xcc);
return ptu_passed();
}
static struct ptunit_result overlap_multiple(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc };
int status, isid;
status = pt_image_add(&ifix->image, &ifix->section[0], &ifix->asid[0],
0x1000ull, 1);
ptu_int_eq(status, 0);
status = pt_image_add(&ifix->image, &ifix->section[0], &ifix->asid[0],
0x1010ull, 2);
ptu_int_eq(status, 0);
status = pt_image_add(&ifix->image, &ifix->section[0], &ifix->asid[0],
0x1008ull, 3);
ptu_int_eq(status, 0);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[0],
0x1007ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 1);
ptu_uint_eq(buffer[0], 0x07);
ptu_uint_eq(buffer[1], 0xcc);
status = pt_image_read(&ifix->image, &isid, buffer, 1, &ifix->asid[0],
0x1008ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 3);
ptu_uint_eq(buffer[0], 0x00);
ptu_uint_eq(buffer[1], 0xcc);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[0],
0x1017ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 3);
ptu_uint_eq(buffer[0], 0x0f);
ptu_uint_eq(buffer[1], 0xcc);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 1, &ifix->asid[0],
0x1018ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 2);
ptu_uint_eq(buffer[0], 0x08);
ptu_uint_eq(buffer[1], 0xcc);
return ptu_passed();
}
static struct ptunit_result overlap_mid(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc };
int status, isid;
status = pt_image_add(&ifix->image, &ifix->section[0], &ifix->asid[0],
0x1000ull, 1);
ptu_int_eq(status, 0);
ifix->section[1].size = 0x8;
ifix->mapping[1].size = 0x8;
status = pt_image_add(&ifix->image, &ifix->section[1], &ifix->asid[0],
0x1004ull, 2);
ptu_int_eq(status, 0);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[0],
0x1003ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 1);
ptu_uint_eq(buffer[0], 0x03);
ptu_uint_eq(buffer[1], 0xcc);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 1, &ifix->asid[0],
0x1004ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 2);
ptu_uint_eq(buffer[0], 0x00);
ptu_uint_eq(buffer[1], 0xcc);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[0],
0x100bull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 2);
ptu_uint_eq(buffer[0], 0x07);
ptu_uint_eq(buffer[1], 0xcc);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 1, &ifix->asid[0],
0x100cull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 1);
ptu_uint_eq(buffer[0], 0x0c);
ptu_uint_eq(buffer[1], 0xcc);
return ptu_passed();
}
static struct ptunit_result contained(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc };
int status, isid;
ifix->section[0].size = 0x8;
ifix->mapping[0].size = 0x8;
status = pt_image_add(&ifix->image, &ifix->section[0], &ifix->asid[0],
0x1004ull, 1);
ptu_int_eq(status, 0);
status = pt_image_add(&ifix->image, &ifix->section[1], &ifix->asid[0],
0x1000ull, 2);
ptu_int_eq(status, 0);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 1, &ifix->asid[0],
0x1008ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 2);
ptu_uint_eq(buffer[0], 0x08);
ptu_uint_eq(buffer[1], 0xcc);
return ptu_passed();
}
static struct ptunit_result contained_multiple(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc };
int status, isid;
ifix->section[0].size = 0x2;
ifix->mapping[0].size = 0x2;
status = pt_image_add(&ifix->image, &ifix->section[0], &ifix->asid[0],
0x1004ull, 1);
ptu_int_eq(status, 0);
status = pt_image_add(&ifix->image, &ifix->section[0], &ifix->asid[0],
0x1008ull, 2);
ptu_int_eq(status, 0);
status = pt_image_add(&ifix->image, &ifix->section[1], &ifix->asid[0],
0x1000ull, 3);
ptu_int_eq(status, 0);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 1, &ifix->asid[0],
0x1004ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 3);
ptu_uint_eq(buffer[0], 0x04);
ptu_uint_eq(buffer[1], 0xcc);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 1, &ifix->asid[0],
0x1008ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 3);
ptu_uint_eq(buffer[0], 0x08);
ptu_uint_eq(buffer[1], 0xcc);
return ptu_passed();
}
static struct ptunit_result contained_back(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc };
int status, isid;
ifix->section[0].size = 0x8;
ifix->mapping[0].size = 0x8;
status = pt_image_add(&ifix->image, &ifix->section[0], &ifix->asid[0],
0x1004ull, 1);
ptu_int_eq(status, 0);
status = pt_image_add(&ifix->image, &ifix->section[0], &ifix->asid[0],
0x100cull, 2);
ptu_int_eq(status, 0);
status = pt_image_add(&ifix->image, &ifix->section[1], &ifix->asid[0],
0x1000ull, 3);
ptu_int_eq(status, 0);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 1, &ifix->asid[0],
0x1004ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 3);
ptu_uint_eq(buffer[0], 0x04);
ptu_uint_eq(buffer[1], 0xcc);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 1, &ifix->asid[0],
0x100cull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 3);
ptu_uint_eq(buffer[0], 0x0c);
ptu_uint_eq(buffer[1], 0xcc);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[0],
0x100full);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 3);
ptu_uint_eq(buffer[0], 0x0f);
ptu_uint_eq(buffer[1], 0xcc);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 1, &ifix->asid[0],
0x1010ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 2);
ptu_uint_eq(buffer[0], 0x04);
ptu_uint_eq(buffer[1], 0xcc);
return ptu_passed();
}
static struct ptunit_result same(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc };
int status, isid;
status = pt_image_add(&ifix->image, &ifix->section[0], &ifix->asid[0],
0x1000ull, 1);
ptu_int_eq(status, 0);
status = pt_image_add(&ifix->image, &ifix->section[0], &ifix->asid[0],
0x1000ull, 1);
ptu_int_eq(status, 0);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 1, &ifix->asid[0],
0x1008ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 1);
ptu_uint_eq(buffer[0], 0x08);
ptu_uint_eq(buffer[1], 0xcc);
return ptu_passed();
}
static struct ptunit_result same_different_isid(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc };
int status, isid;
status = pt_image_add(&ifix->image, &ifix->section[0], &ifix->asid[0],
0x1000ull, 1);
ptu_int_eq(status, 0);
status = pt_image_add(&ifix->image, &ifix->section[0], &ifix->asid[0],
0x1000ull, 2);
ptu_int_eq(status, 0);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 1, &ifix->asid[0],
0x1008ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 2);
ptu_uint_eq(buffer[0], 0x08);
ptu_uint_eq(buffer[1], 0xcc);
return ptu_passed();
}
static struct ptunit_result adjacent(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc };
int status, isid;
status = pt_image_add(&ifix->image, &ifix->section[0], &ifix->asid[0],
0x1000ull, 1);
ptu_int_eq(status, 0);
status = pt_image_add(&ifix->image, &ifix->section[1], &ifix->asid[0],
0x1000ull - ifix->section[1].size, 2);
ptu_int_eq(status, 0);
status = pt_image_add(&ifix->image, &ifix->section[2], &ifix->asid[0],
0x1000ull + ifix->section[0].size, 3);
ptu_int_eq(status, 0);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 1, &ifix->asid[0],
0x1000ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 1);
ptu_uint_eq(buffer[0], 0x00);
ptu_uint_eq(buffer[1], 0xcc);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 1, &ifix->asid[0],
0xfffull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 2);
ptu_uint_eq(buffer[0],
ifix->mapping[1].content[ifix->mapping[1].size - 1]);
ptu_uint_eq(buffer[1], 0xcc);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 1, &ifix->asid[0],
0x1000ull + ifix->section[0].size);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 3);
ptu_uint_eq(buffer[0], 0x00);
ptu_uint_eq(buffer[1], 0xcc);
return ptu_passed();
}
static struct ptunit_result read_null(struct image_fixture *ifix)
{
uint8_t buffer;
int status, isid;
status = pt_image_read(NULL, &isid, &buffer, 1, &ifix->asid[0],
0x1000ull);
ptu_int_eq(status, -pte_internal);
status = pt_image_read(&ifix->image, NULL, &buffer, 1, &ifix->asid[0],
0x1000ull);
ptu_int_eq(status, -pte_internal);
status = pt_image_read(&ifix->image, &isid, NULL, 1, &ifix->asid[0],
0x1000ull);
ptu_int_eq(status, -pte_internal);
status = pt_image_read(&ifix->image, &isid, &buffer, 1, NULL,
0x1000ull);
ptu_int_eq(status, -pte_internal);
return ptu_passed();
}
static struct ptunit_result read(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc, 0xcc };
int status, isid;
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[1],
0x2003ull);
ptu_int_eq(status, 2);
ptu_int_eq(isid, 11);
ptu_uint_eq(buffer[0], 0x03);
ptu_uint_eq(buffer[1], 0x04);
ptu_uint_eq(buffer[2], 0xcc);
return ptu_passed();
}
static struct ptunit_result read_asid(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc };
int status, isid;
status = pt_image_add(&ifix->image, &ifix->section[0], &ifix->asid[0],
0x1000ull, 1);
ptu_int_eq(status, 0);
status = pt_image_add(&ifix->image, &ifix->section[1], &ifix->asid[1],
0x1008ull, 2);
ptu_int_eq(status, 0);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 1, &ifix->asid[0],
0x1009ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 1);
ptu_uint_eq(buffer[0], 0x09);
ptu_uint_eq(buffer[1], 0xcc);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 1, &ifix->asid[1],
0x1009ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 2);
ptu_uint_eq(buffer[0], 0x01);
ptu_uint_eq(buffer[1], 0xcc);
return ptu_passed();
}
static struct ptunit_result read_bad_asid(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc };
int status, isid;
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, sizeof(buffer),
&ifix->asid[0], 0x2003ull);
ptu_int_eq(status, -pte_nomap);
ptu_int_eq(isid, -1);
ptu_uint_eq(buffer[0], 0xcc);
ptu_uint_eq(buffer[1], 0xcc);
return ptu_passed();
}
static struct ptunit_result read_null_asid(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc, 0xcc };
int status, isid;
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, NULL, 0x2003ull);
ptu_int_eq(status, -pte_internal);
ptu_int_eq(isid, -1);
ptu_uint_eq(buffer[0], 0xcc);
ptu_uint_eq(buffer[1], 0xcc);
return ptu_passed();
}
static struct ptunit_result read_callback(struct image_fixture *ifix)
{
uint8_t memory[] = { 0xdd, 0x01, 0x02, 0xdd };
uint8_t buffer[] = { 0xcc, 0xcc, 0xcc };
int status, isid;
status = pt_image_set_callback(&ifix->image, image_readmem_callback,
memory);
ptu_int_eq(status, 0);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[0],
0x3001ull);
ptu_int_eq(status, 2);
ptu_int_eq(isid, 0);
ptu_uint_eq(buffer[0], 0x01);
ptu_uint_eq(buffer[1], 0x02);
ptu_uint_eq(buffer[2], 0xcc);
return ptu_passed();
}
static struct ptunit_result read_nomem(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc };
int status, isid;
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, sizeof(buffer),
&ifix->asid[1], 0x1010ull);
ptu_int_eq(status, -pte_nomap);
ptu_int_eq(isid, -1);
ptu_uint_eq(buffer[0], 0xcc);
ptu_uint_eq(buffer[1], 0xcc);
return ptu_passed();
}
static struct ptunit_result read_truncated(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc };
int status, isid;
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, sizeof(buffer),
&ifix->asid[0], 0x100full);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 10);
ptu_uint_eq(buffer[0], 0x0f);
ptu_uint_eq(buffer[1], 0xcc);
return ptu_passed();
}
static struct ptunit_result read_error(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc };
int status, isid;
ifix->mapping[0].errcode = -pte_nosync;
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 1, &ifix->asid[0],
0x1000ull);
ptu_int_eq(status, -pte_nosync);
ptu_int_eq(isid, 10);
ptu_uint_eq(buffer[0], 0xcc);
return ptu_passed();
}
static struct ptunit_result read_spurious_error(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc };
int status, isid;
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 1, &ifix->asid[0],
0x1000ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 10);
ptu_uint_eq(buffer[0], 0x00);
ptu_uint_eq(buffer[1], 0xcc);
ifix->mapping[0].errcode = -pte_nosync;
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 1, &ifix->asid[0],
0x1005ull);
ptu_int_eq(status, -pte_nosync);
ptu_int_eq(isid, 10);
ptu_uint_eq(buffer[0], 0x00);
return ptu_passed();
}
static struct ptunit_result remove_section(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc, 0xcc };
int status, isid;
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[0],
0x1001ull);
ptu_int_eq(status, 2);
ptu_int_eq(isid, 10);
ptu_uint_eq(buffer[0], 0x01);
ptu_uint_eq(buffer[1], 0x02);
ptu_uint_eq(buffer[2], 0xcc);
status = pt_image_remove(&ifix->image, &ifix->section[0],
&ifix->asid[0], 0x1000ull);
ptu_int_eq(status, 0);
ptu_int_ne(ifix->status[0].deleted, 0);
ptu_int_eq(ifix->status[1].deleted, 0);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, sizeof(buffer),
&ifix->asid[0], 0x1003ull);
ptu_int_eq(status, -pte_nomap);
ptu_int_eq(isid, -1);
ptu_uint_eq(buffer[0], 0x01);
ptu_uint_eq(buffer[1], 0x02);
ptu_uint_eq(buffer[2], 0xcc);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[1],
0x2003ull);
ptu_int_eq(status, 2);
ptu_int_eq(isid, 11);
ptu_uint_eq(buffer[0], 0x03);
ptu_uint_eq(buffer[1], 0x04);
ptu_uint_eq(buffer[2], 0xcc);
return ptu_passed();
}
static struct ptunit_result remove_bad_vaddr(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc, 0xcc };
int status, isid;
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[0],
0x1001ull);
ptu_int_eq(status, 2);
ptu_int_eq(isid, 10);
ptu_uint_eq(buffer[0], 0x01);
ptu_uint_eq(buffer[1], 0x02);
ptu_uint_eq(buffer[2], 0xcc);
status = pt_image_remove(&ifix->image, &ifix->section[0],
&ifix->asid[0], 0x2000ull);
ptu_int_eq(status, -pte_bad_image);
ptu_int_eq(ifix->status[0].deleted, 0);
ptu_int_eq(ifix->status[1].deleted, 0);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[0],
0x1003ull);
ptu_int_eq(status, 2);
ptu_int_eq(isid, 10);
ptu_uint_eq(buffer[0], 0x03);
ptu_uint_eq(buffer[1], 0x04);
ptu_uint_eq(buffer[2], 0xcc);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[1],
0x2005ull);
ptu_int_eq(status, 2);
ptu_int_eq(isid, 11);
ptu_uint_eq(buffer[0], 0x05);
ptu_uint_eq(buffer[1], 0x06);
ptu_uint_eq(buffer[2], 0xcc);
return ptu_passed();
}
static struct ptunit_result remove_bad_asid(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc, 0xcc };
int status, isid;
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[0],
0x1001ull);
ptu_int_eq(status, 2);
ptu_int_eq(isid, 10);
ptu_uint_eq(buffer[0], 0x01);
ptu_uint_eq(buffer[1], 0x02);
ptu_uint_eq(buffer[2], 0xcc);
status = pt_image_remove(&ifix->image, &ifix->section[0],
&ifix->asid[1], 0x1000ull);
ptu_int_eq(status, -pte_bad_image);
ptu_int_eq(ifix->status[0].deleted, 0);
ptu_int_eq(ifix->status[1].deleted, 0);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[0],
0x1003ull);
ptu_int_eq(status, 2);
ptu_int_eq(isid, 10);
ptu_uint_eq(buffer[0], 0x03);
ptu_uint_eq(buffer[1], 0x04);
ptu_uint_eq(buffer[2], 0xcc);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[1],
0x2005ull);
ptu_int_eq(status, 2);
ptu_int_eq(isid, 11);
ptu_uint_eq(buffer[0], 0x05);
ptu_uint_eq(buffer[1], 0x06);
ptu_uint_eq(buffer[2], 0xcc);
return ptu_passed();
}
static struct ptunit_result remove_by_filename(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc, 0xcc };
int status, isid;
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[0],
0x1001ull);
ptu_int_eq(status, 2);
ptu_int_eq(isid, 10);
ptu_uint_eq(buffer[0], 0x01);
ptu_uint_eq(buffer[1], 0x02);
ptu_uint_eq(buffer[2], 0xcc);
status = pt_image_remove_by_filename(&ifix->image,
ifix->section[0].filename,
&ifix->asid[0]);
ptu_int_eq(status, 1);
ptu_int_ne(ifix->status[0].deleted, 0);
ptu_int_eq(ifix->status[1].deleted, 0);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, sizeof(buffer),
&ifix->asid[0], 0x1003ull);
ptu_int_eq(status, -pte_nomap);
ptu_int_eq(isid, -1);
ptu_uint_eq(buffer[0], 0x01);
ptu_uint_eq(buffer[1], 0x02);
ptu_uint_eq(buffer[2], 0xcc);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[1],
0x2003ull);
ptu_int_eq(status, 2);
ptu_int_eq(isid, 11);
ptu_uint_eq(buffer[0], 0x03);
ptu_uint_eq(buffer[1], 0x04);
ptu_uint_eq(buffer[2], 0xcc);
return ptu_passed();
}
static struct ptunit_result
remove_by_filename_bad_asid(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc, 0xcc };
int status, isid;
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[0],
0x1001ull);
ptu_int_eq(status, 2);
ptu_int_eq(isid, 10);
ptu_uint_eq(buffer[0], 0x01);
ptu_uint_eq(buffer[1], 0x02);
ptu_uint_eq(buffer[2], 0xcc);
status = pt_image_remove_by_filename(&ifix->image,
ifix->section[0].filename,
&ifix->asid[1]);
ptu_int_eq(status, 0);
ptu_int_eq(ifix->status[0].deleted, 0);
ptu_int_eq(ifix->status[1].deleted, 0);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[0],
0x1003ull);
ptu_int_eq(status, 2);
ptu_int_eq(isid, 10);
ptu_uint_eq(buffer[0], 0x03);
ptu_uint_eq(buffer[1], 0x04);
ptu_uint_eq(buffer[2], 0xcc);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[1],
0x2005ull);
ptu_int_eq(status, 2);
ptu_int_eq(isid, 11);
ptu_uint_eq(buffer[0], 0x05);
ptu_uint_eq(buffer[1], 0x06);
ptu_uint_eq(buffer[2], 0xcc);
return ptu_passed();
}
static struct ptunit_result remove_none_by_filename(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc, 0xcc };
int status, isid;
status = pt_image_remove_by_filename(&ifix->image, "bad-name",
&ifix->asid[0]);
ptu_int_eq(status, 0);
ptu_int_eq(ifix->status[0].deleted, 0);
ptu_int_eq(ifix->status[1].deleted, 0);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[0],
0x1003ull);
ptu_int_eq(status, 2);
ptu_int_eq(isid, 10);
ptu_uint_eq(buffer[0], 0x03);
ptu_uint_eq(buffer[1], 0x04);
ptu_uint_eq(buffer[2], 0xcc);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[1],
0x2001ull);
ptu_int_eq(status, 2);
ptu_int_eq(isid, 11);
ptu_uint_eq(buffer[0], 0x01);
ptu_uint_eq(buffer[1], 0x02);
ptu_uint_eq(buffer[2], 0xcc);
return ptu_passed();
}
static struct ptunit_result remove_all_by_filename(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc, 0xcc };
int status, isid;
ifix->section[0].filename = "same-name";
ifix->section[1].filename = "same-name";
status = pt_image_add(&ifix->image, &ifix->section[0], &ifix->asid[0],
0x1000ull, 1);
ptu_int_eq(status, 0);
status = pt_image_add(&ifix->image, &ifix->section[1], &ifix->asid[0],
0x2000ull, 2);
ptu_int_eq(status, 0);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[0],
0x1001ull);
ptu_int_eq(status, 2);
ptu_int_eq(isid, 1);
ptu_uint_eq(buffer[0], 0x01);
ptu_uint_eq(buffer[1], 0x02);
ptu_uint_eq(buffer[2], 0xcc);
status = pt_image_remove_by_filename(&ifix->image, "same-name",
&ifix->asid[0]);
ptu_int_eq(status, 2);
ptu_int_ne(ifix->status[0].deleted, 0);
ptu_int_ne(ifix->status[1].deleted, 0);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, sizeof(buffer),
&ifix->asid[0], 0x1003ull);
ptu_int_eq(status, -pte_nomap);
ptu_int_eq(isid, -1);
ptu_uint_eq(buffer[0], 0x01);
ptu_uint_eq(buffer[1], 0x02);
ptu_uint_eq(buffer[2], 0xcc);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, sizeof(buffer),
&ifix->asid[0], 0x2003ull);
ptu_int_eq(status, -pte_nomap);
ptu_int_eq(isid, -1);
ptu_uint_eq(buffer[0], 0x01);
ptu_uint_eq(buffer[1], 0x02);
ptu_uint_eq(buffer[2], 0xcc);
return ptu_passed();
}
static struct ptunit_result remove_by_asid(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc, 0xcc };
int status, isid;
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[0],
0x1001ull);
ptu_int_eq(status, 2);
ptu_int_eq(isid, 10);
ptu_uint_eq(buffer[0], 0x01);
ptu_uint_eq(buffer[1], 0x02);
ptu_uint_eq(buffer[2], 0xcc);
status = pt_image_remove_by_asid(&ifix->image, &ifix->asid[0]);
ptu_int_eq(status, 1);
ptu_int_ne(ifix->status[0].deleted, 0);
ptu_int_eq(ifix->status[1].deleted, 0);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, sizeof(buffer),
&ifix->asid[0], 0x1003ull);
ptu_int_eq(status, -pte_nomap);
ptu_int_eq(isid, -1);
ptu_uint_eq(buffer[0], 0x01);
ptu_uint_eq(buffer[1], 0x02);
ptu_uint_eq(buffer[2], 0xcc);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[1],
0x2003ull);
ptu_int_eq(status, 2);
ptu_int_eq(isid, 11);
ptu_uint_eq(buffer[0], 0x03);
ptu_uint_eq(buffer[1], 0x04);
ptu_uint_eq(buffer[2], 0xcc);
return ptu_passed();
}
static struct ptunit_result copy_empty(struct image_fixture *ifix)
{
struct pt_asid asid;
uint8_t buffer[] = { 0xcc, 0xcc };
int status, isid;
pt_asid_init(&asid);
status = pt_image_copy(&ifix->copy, &ifix->image);
ptu_int_eq(status, 0);
isid = -1;
status = pt_image_read(&ifix->copy, &isid, buffer, sizeof(buffer),
&asid, 0x1000ull);
ptu_int_eq(status, -pte_nomap);
ptu_int_eq(isid, -1);
ptu_uint_eq(buffer[0], 0xcc);
ptu_uint_eq(buffer[1], 0xcc);
return ptu_passed();
}
static struct ptunit_result copy(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc, 0xcc };
int status, isid;
status = pt_image_copy(&ifix->copy, &ifix->image);
ptu_int_eq(status, 0);
isid = -1;
status = pt_image_read(&ifix->copy, &isid, buffer, 2, &ifix->asid[1],
0x2003ull);
ptu_int_eq(status, 2);
ptu_int_eq(isid, 11);
ptu_uint_eq(buffer[0], 0x03);
ptu_uint_eq(buffer[1], 0x04);
ptu_uint_eq(buffer[2], 0xcc);
return ptu_passed();
}
static struct ptunit_result copy_self(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc, 0xcc };
int status, isid;
status = pt_image_copy(&ifix->image, &ifix->image);
ptu_int_eq(status, 0);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[1],
0x2003ull);
ptu_int_eq(status, 2);
ptu_int_eq(isid, 11);
ptu_uint_eq(buffer[0], 0x03);
ptu_uint_eq(buffer[1], 0x04);
ptu_uint_eq(buffer[2], 0xcc);
return ptu_passed();
}
static struct ptunit_result copy_shrink(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc, 0xcc };
int status, isid;
status = pt_image_add(&ifix->copy, &ifix->section[1], &ifix->asid[1],
0x2000ull, 1);
ptu_int_eq(status, 0);
status = pt_image_copy(&ifix->copy, &ifix->image);
ptu_int_eq(status, 0);
isid = -1;
status = pt_image_read(&ifix->copy, &isid, buffer, 2, &ifix->asid[1],
0x2003ull);
ptu_int_eq(status, 2);
ptu_int_eq(isid, 11);
ptu_uint_eq(buffer[0], 0x03);
ptu_uint_eq(buffer[1], 0x04);
ptu_uint_eq(buffer[2], 0xcc);
return ptu_passed();
}
static struct ptunit_result copy_split(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc };
int status, isid;
status = pt_image_add(&ifix->copy, &ifix->section[0], &ifix->asid[0],
0x2000ull, 1);
ptu_int_eq(status, 0);
ifix->section[1].size = 0x7;
ifix->mapping[1].size = 0x7;
status = pt_image_add(&ifix->image, &ifix->section[1], &ifix->asid[0],
0x2001ull, 2);
ptu_int_eq(status, 0);
ifix->section[2].size = 0x8;
ifix->mapping[2].size = 0x8;
status = pt_image_add(&ifix->image, &ifix->section[2], &ifix->asid[0],
0x2008ull, 3);
ptu_int_eq(status, 0);
status = pt_image_copy(&ifix->copy, &ifix->image);
ptu_int_eq(status, 0);
isid = -1;
status = pt_image_read(&ifix->copy, &isid, buffer, 1, &ifix->asid[0],
0x2003ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 2);
ptu_uint_eq(buffer[0], 0x02);
ptu_uint_eq(buffer[1], 0xcc);
isid = -1;
status = pt_image_read(&ifix->copy, &isid, buffer, 1, &ifix->asid[0],
0x2009ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 3);
ptu_uint_eq(buffer[0], 0x01);
ptu_uint_eq(buffer[1], 0xcc);
isid = -1;
status = pt_image_read(&ifix->copy, &isid, buffer, 1, &ifix->asid[0],
0x2000ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 1);
ptu_uint_eq(buffer[0], 0x00);
ptu_uint_eq(buffer[1], 0xcc);
return ptu_passed();
}
static struct ptunit_result copy_merge(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc };
int status, isid;
ifix->section[1].size = 0x8;
ifix->mapping[1].size = 0x8;
status = pt_image_add(&ifix->copy, &ifix->section[1], &ifix->asid[0],
0x2000ull, 1);
ptu_int_eq(status, 0);
ifix->section[2].size = 0x8;
ifix->mapping[2].size = 0x8;
status = pt_image_add(&ifix->copy, &ifix->section[2], &ifix->asid[0],
0x2008ull, 2);
ptu_int_eq(status, 0);
status = pt_image_add(&ifix->image, &ifix->section[0], &ifix->asid[0],
0x2000ull, 3);
ptu_int_eq(status, 0);
status = pt_image_copy(&ifix->copy, &ifix->image);
ptu_int_eq(status, 0);
isid = -1;
status = pt_image_read(&ifix->copy, &isid, buffer, 1, &ifix->asid[0],
0x2003ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 3);
ptu_uint_eq(buffer[0], 0x03);
ptu_uint_eq(buffer[1], 0xcc);
isid = -1;
status = pt_image_read(&ifix->copy, &isid, buffer, 1, &ifix->asid[0],
0x200aull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 3);
ptu_uint_eq(buffer[0], 0x0a);
ptu_uint_eq(buffer[1], 0xcc);
return ptu_passed();
}
static struct ptunit_result copy_overlap(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc };
int status, isid;
status = pt_image_add(&ifix->copy, &ifix->section[0], &ifix->asid[0],
0x2000ull, 1);
ptu_int_eq(status, 0);
status = pt_image_add(&ifix->copy, &ifix->section[1], &ifix->asid[0],
0x2010ull, 2);
ptu_int_eq(status, 0);
status = pt_image_add(&ifix->image, &ifix->section[2], &ifix->asid[0],
0x2008ull, 3);
ptu_int_eq(status, 0);
status = pt_image_copy(&ifix->copy, &ifix->image);
ptu_int_eq(status, 0);
isid = -1;
status = pt_image_read(&ifix->copy, &isid, buffer, 1, &ifix->asid[0],
0x2003ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 1);
ptu_uint_eq(buffer[0], 0x03);
ptu_uint_eq(buffer[1], 0xcc);
isid = -1;
status = pt_image_read(&ifix->copy, &isid, buffer, 1, &ifix->asid[0],
0x200aull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 3);
ptu_uint_eq(buffer[0], 0x02);
ptu_uint_eq(buffer[1], 0xcc);
isid = -1;
status = pt_image_read(&ifix->copy, &isid, buffer, 1, &ifix->asid[0],
0x2016ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 3);
ptu_uint_eq(buffer[0], 0x0e);
ptu_uint_eq(buffer[1], 0xcc);
isid = -1;
status = pt_image_read(&ifix->copy, &isid, buffer, 1, &ifix->asid[0],
0x2019ull);
ptu_int_eq(status, 1);
ptu_int_eq(isid, 2);
ptu_uint_eq(buffer[0], 0x09);
ptu_uint_eq(buffer[1], 0xcc);
return ptu_passed();
}
static struct ptunit_result copy_replace(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc, 0xcc };
int status, isid;
ifix->section[0].size = 0x8;
ifix->mapping[0].size = 0x8;
status = pt_image_add(&ifix->copy, &ifix->section[0], &ifix->asid[0],
0x1004ull, 1);
ptu_int_eq(status, 0);
status = pt_image_add(&ifix->image, &ifix->section[1], &ifix->asid[0],
0x1000ull, 2);
ptu_int_eq(status, 0);
status = pt_image_copy(&ifix->copy, &ifix->image);
ptu_int_eq(status, 0);
isid = -1;
status = pt_image_read(&ifix->copy, &isid, buffer, 2, &ifix->asid[0],
0x1003ull);
ptu_int_eq(status, 2);
ptu_int_eq(isid, 2);
ptu_uint_eq(buffer[0], 0x03);
ptu_uint_eq(buffer[1], 0x04);
ptu_uint_eq(buffer[2], 0xcc);
return ptu_passed();
}
static struct ptunit_result add_cached_null(void)
{
struct pt_image_section_cache iscache;
struct pt_image image;
int status;
status = pt_image_add_cached(NULL, &iscache, 0, NULL);
ptu_int_eq(status, -pte_invalid);
status = pt_image_add_cached(&image, NULL, 0, NULL);
ptu_int_eq(status, -pte_invalid);
return ptu_passed();
}
static struct ptunit_result add_cached(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc, 0xcc };
int status, isid, risid;
isid = ifix_cache_section(ifix, &ifix->section[0], 0x1000ull);
ptu_int_gt(isid, 0);
status = pt_image_add_cached(&ifix->image, &ifix->iscache, isid,
&ifix->asid[0]);
ptu_int_eq(status, 0);
risid = -1;
status = pt_image_read(&ifix->image, &risid, buffer, 2, &ifix->asid[0],
0x1003ull);
ptu_int_eq(status, 2);
ptu_int_eq(risid, isid);
ptu_uint_eq(buffer[0], 0x03);
ptu_uint_eq(buffer[1], 0x04);
ptu_uint_eq(buffer[2], 0xcc);
return ptu_passed();
}
static struct ptunit_result add_cached_null_asid(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc, 0xcc };
int status, isid, risid;
isid = ifix_cache_section(ifix, &ifix->section[0], 0x1000ull);
ptu_int_gt(isid, 0);
status = pt_image_add_cached(&ifix->image, &ifix->iscache, isid, NULL);
ptu_int_eq(status, 0);
risid = -1;
status = pt_image_read(&ifix->image, &risid, buffer, 2, &ifix->asid[0],
0x1003ull);
ptu_int_eq(status, 2);
ptu_int_eq(risid, isid);
ptu_uint_eq(buffer[0], 0x03);
ptu_uint_eq(buffer[1], 0x04);
ptu_uint_eq(buffer[2], 0xcc);
return ptu_passed();
}
static struct ptunit_result add_cached_twice(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc, 0xcc };
int status, isid, risid;
isid = ifix_cache_section(ifix, &ifix->section[0], 0x1000ull);
ptu_int_gt(isid, 0);
status = pt_image_add_cached(&ifix->image, &ifix->iscache, isid,
&ifix->asid[0]);
ptu_int_eq(status, 0);
status = pt_image_add_cached(&ifix->image, &ifix->iscache, isid,
&ifix->asid[0]);
ptu_int_eq(status, 0);
risid = -1;
status = pt_image_read(&ifix->image, &risid, buffer, 2, &ifix->asid[0],
0x1003ull);
ptu_int_eq(status, 2);
ptu_int_eq(risid, isid);
ptu_uint_eq(buffer[0], 0x03);
ptu_uint_eq(buffer[1], 0x04);
ptu_uint_eq(buffer[2], 0xcc);
return ptu_passed();
}
static struct ptunit_result add_cached_bad_isid(struct image_fixture *ifix)
{
uint8_t buffer[] = { 0xcc, 0xcc, 0xcc };
int status, isid;
status = pt_image_add_cached(&ifix->image, &ifix->iscache, 1,
&ifix->asid[0]);
ptu_int_eq(status, -pte_bad_image);
isid = -1;
status = pt_image_read(&ifix->image, &isid, buffer, 2, &ifix->asid[0],
0x1003ull);
ptu_int_eq(status, -pte_nomap);
ptu_int_eq(isid, -1);
return ptu_passed();
}
static struct ptunit_result find_null(struct image_fixture *ifix)
{
struct pt_section *section;
uint64_t laddr;
int status;
status = pt_image_find(NULL, &section, &laddr, &ifix->asid[0],
0x1000ull);
ptu_int_eq(status, -pte_internal);
status = pt_image_find(&ifix->image, NULL, &laddr, &ifix->asid[0],
0x1000ull);
ptu_int_eq(status, -pte_internal);
status = pt_image_find(&ifix->image, &section, NULL, &ifix->asid[0],
0x1000ull);
ptu_int_eq(status, -pte_internal);
status = pt_image_find(&ifix->image, &section, &laddr, NULL, 0x1000ull);
ptu_int_eq(status, -pte_internal);
return ptu_passed();
}
static struct ptunit_result find(struct image_fixture *ifix)
{
struct pt_section *section;
uint64_t laddr;
int status;
status = pt_image_find(&ifix->image, &section, &laddr, &ifix->asid[1],
0x2003ull);
ptu_int_eq(status, 11);
ptu_ptr_eq(section, &ifix->section[1]);
ptu_uint_eq(laddr, 0x2000ull);
status = pt_section_put(section);
ptu_int_eq(status, 0);
return ptu_passed();
}
static struct ptunit_result find_asid(struct image_fixture *ifix)
{
struct pt_section *section;
uint64_t laddr;
int status;
status = pt_image_add(&ifix->image, &ifix->section[0], &ifix->asid[0],
0x1000ull, 1);
ptu_int_eq(status, 0);
status = pt_image_add(&ifix->image, &ifix->section[0], &ifix->asid[1],
0x1008ull, 2);
ptu_int_eq(status, 0);
status = pt_image_find(&ifix->image, &section, &laddr, &ifix->asid[0],
0x1009ull);
ptu_int_eq(status, 1);
ptu_ptr_eq(section, &ifix->section[0]);
ptu_uint_eq(laddr, 0x1000ull);
status = pt_section_put(section);
ptu_int_eq(status, 0);
status = pt_image_find(&ifix->image, &section, &laddr, &ifix->asid[1],
0x1009ull);
ptu_int_eq(status, 2);
ptu_ptr_eq(section, &ifix->section[0]);
ptu_uint_eq(laddr, 0x1008ull);
status = pt_section_put(section);
ptu_int_eq(status, 0);
return ptu_passed();
}
static struct ptunit_result find_bad_asid(struct image_fixture *ifix)
{
struct pt_section *section;
uint64_t laddr;
int status;
status = pt_image_find(&ifix->image, &section, &laddr, &ifix->asid[0],
0x2003ull);
ptu_int_eq(status, -pte_nomap);
return ptu_passed();
}
static struct ptunit_result find_nomem(struct image_fixture *ifix)
{
struct pt_section *section;
uint64_t laddr;
int status;
status = pt_image_find(&ifix->image, &section, &laddr, &ifix->asid[1],
0x1010ull);
ptu_int_eq(status, -pte_nomap);
return ptu_passed();
}
static struct ptunit_result validate_null(struct image_fixture *ifix)
{
int status;
status = pt_image_validate(NULL, &ifix->asid[0], 0x1004ull,
&ifix->section[0], 0x1000ull, 10);
ptu_int_eq(status, -pte_internal);
status = pt_image_validate(&ifix->image, NULL, 0x1004ull,
&ifix->section[0], 0x1000ull, 10);
ptu_int_eq(status, -pte_internal);
return ptu_passed();
}
static struct ptunit_result validate(struct image_fixture *ifix)
{
int status;
/* This test depends on the order in which sections are stored when
* added to the image.
*
* Since pt_image_validate() only looks at the top of the LRU stack we
* can only validate that section - i.e. the one that was added first.
*/
status = pt_image_validate(&ifix->image, &ifix->asid[0], 0x1004ull,
&ifix->section[0], 0x1000ull, 10);
ptu_int_eq(status, 0);
return ptu_passed();
}
static struct ptunit_result validate_bad_asid(struct image_fixture *ifix)
{
int status;
status = pt_image_validate(&ifix->image, &ifix->asid[1], 0x1004ull,
&ifix->section[0], 0x1000ull, 10);
ptu_int_eq(status, -pte_nomap);
return ptu_passed();
}
static struct ptunit_result validate_bad_laddr(struct image_fixture *ifix)
{
int status;
status = pt_image_validate(&ifix->image, &ifix->asid[0], 0x1004ull,
&ifix->section[0], 0x2000ull, 10);
ptu_int_eq(status, -pte_nomap);
return ptu_passed();
}
static struct ptunit_result validate_bad_isid(struct image_fixture *ifix)
{
int status;
status = pt_image_validate(&ifix->image, &ifix->asid[0], 0x1004ull,
&ifix->section[0], 0x1000ull, 11);
ptu_int_eq(status, -pte_nomap);
return ptu_passed();
}
struct ptunit_result ifix_init(struct image_fixture *ifix)
{
int index;
pt_image_init(&ifix->image, NULL);
pt_image_init(&ifix->copy, NULL);
memset(ifix->status, 0, sizeof(ifix->status));
memset(ifix->mapping, 0, sizeof(ifix->mapping));
memset(ifix->section, 0, sizeof(ifix->section));
memset(&ifix->iscache, 0, sizeof(ifix->iscache));
ifix->nsecs = 0;
index = ifix_add_section(ifix, "file-0");
ptu_int_eq(index, 0);
index = ifix_add_section(ifix, "file-1");
ptu_int_eq(index, 1);
index = ifix_add_section(ifix, "file-2");
ptu_int_eq(index, 2);
pt_asid_init(&ifix->asid[0]);
ifix->asid[0].cr3 = 0xa000;
pt_asid_init(&ifix->asid[1]);
ifix->asid[1].cr3 = 0xb000;
pt_asid_init(&ifix->asid[2]);
ifix->asid[2].cr3 = 0xc000;
return ptu_passed();
}
struct ptunit_result rfix_init(struct image_fixture *ifix)
{
int status;
ptu_check(ifix_init, ifix);
status = pt_image_add(&ifix->image, &ifix->section[0], &ifix->asid[0],
0x1000ull, 10);
ptu_int_eq(status, 0);
status = pt_image_add(&ifix->image, &ifix->section[1], &ifix->asid[1],
0x2000ull, 11);
ptu_int_eq(status, 0);
return ptu_passed();
}
struct ptunit_result dfix_fini(struct image_fixture *ifix)
{
pt_image_fini(&ifix->image);
return ptu_passed();
}
struct ptunit_result ifix_fini(struct image_fixture *ifix)
{
int sec;
ptu_check(dfix_fini, ifix);
pt_image_fini(&ifix->copy);
for (sec = 0; sec < ifix_nsecs; ++sec) {
ptu_int_eq(ifix->section[sec].ucount, 0);
ptu_int_eq(ifix->section[sec].mcount, 0);
ptu_int_le(ifix->status[sec].deleted, 1);
ptu_int_eq(ifix->status[sec].bad_put, 0);
}
return ptu_passed();
}
int main(int argc, char **argv)
{
struct image_fixture dfix, ifix, rfix;
struct ptunit_suite suite;
/* Dfix provides image destruction. */
dfix.init = NULL;
dfix.fini = dfix_fini;
/* Ifix provides an empty image. */
ifix.init = ifix_init;
ifix.fini = ifix_fini;
/* Rfix provides an image with two sections added. */
rfix.init = rfix_init;
rfix.fini = ifix_fini;
suite = ptunit_mk_suite(argc, argv);
ptu_run(suite, init);
ptu_run_f(suite, init_name, dfix);
ptu_run(suite, init_null);
ptu_run(suite, fini);
ptu_run(suite, fini_empty);
ptu_run(suite, fini_null);
ptu_run_f(suite, name, dfix);
ptu_run(suite, name_none);
ptu_run(suite, name_null);
ptu_run_f(suite, read_empty, ifix);
ptu_run_f(suite, overlap_front, ifix);
ptu_run_f(suite, overlap_back, ifix);
ptu_run_f(suite, overlap_multiple, ifix);
ptu_run_f(suite, overlap_mid, ifix);
ptu_run_f(suite, contained, ifix);
ptu_run_f(suite, contained_multiple, ifix);
ptu_run_f(suite, contained_back, ifix);
ptu_run_f(suite, same, ifix);
ptu_run_f(suite, same_different_isid, ifix);
ptu_run_f(suite, adjacent, ifix);
ptu_run_f(suite, read_null, rfix);
ptu_run_f(suite, read, rfix);
ptu_run_f(suite, read_null, rfix);
ptu_run_f(suite, read_asid, ifix);
ptu_run_f(suite, read_bad_asid, rfix);
ptu_run_f(suite, read_null_asid, rfix);
ptu_run_f(suite, read_callback, rfix);
ptu_run_f(suite, read_nomem, rfix);
ptu_run_f(suite, read_truncated, rfix);
ptu_run_f(suite, read_error, rfix);
ptu_run_f(suite, read_spurious_error, rfix);
ptu_run_f(suite, remove_section, rfix);
ptu_run_f(suite, remove_bad_vaddr, rfix);
ptu_run_f(suite, remove_bad_asid, rfix);
ptu_run_f(suite, remove_by_filename, rfix);
ptu_run_f(suite, remove_by_filename_bad_asid, rfix);
ptu_run_f(suite, remove_none_by_filename, rfix);
ptu_run_f(suite, remove_all_by_filename, ifix);
ptu_run_f(suite, remove_by_asid, rfix);
ptu_run_f(suite, copy_empty, ifix);
ptu_run_f(suite, copy, rfix);
ptu_run_f(suite, copy_self, rfix);
ptu_run_f(suite, copy_shrink, rfix);
ptu_run_f(suite, copy_split, ifix);
ptu_run_f(suite, copy_merge, ifix);
ptu_run_f(suite, copy_overlap, ifix);
ptu_run_f(suite, copy_replace, ifix);
ptu_run(suite, add_cached_null);
ptu_run_f(suite, add_cached, ifix);
ptu_run_f(suite, add_cached_null_asid, ifix);
ptu_run_f(suite, add_cached_twice, ifix);
ptu_run_f(suite, add_cached_bad_isid, ifix);
ptu_run_f(suite, find_null, rfix);
ptu_run_f(suite, find, rfix);
ptu_run_f(suite, find_asid, ifix);
ptu_run_f(suite, find_bad_asid, rfix);
ptu_run_f(suite, find_nomem, rfix);
ptu_run_f(suite, validate_null, rfix);
ptu_run_f(suite, validate, rfix);
ptu_run_f(suite, validate_bad_asid, rfix);
ptu_run_f(suite, validate_bad_laddr, rfix);
ptu_run_f(suite, validate_bad_isid, rfix);
ptunit_report(&suite);
return suite.nr_fails;
}