336 lines
8.3 KiB
C
336 lines
8.3 KiB
C
|
/*
|
||
|
*
|
||
|
* Copyright (c) 2019 Jonathan Afek <jonyafek@me.com>
|
||
|
*
|
||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
||
|
* of this software and associated documentation files (the "Software"), to deal
|
||
|
* in the Software without restriction, including without limitation the rights
|
||
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
||
|
* copies of the Software, and to permit persons to whom the Software is
|
||
|
* furnished to do so, subject to the following conditions:
|
||
|
*
|
||
|
* The above copyright notice and this permission notice shall be included in
|
||
|
* all copies or substantial portions of the Software.
|
||
|
*
|
||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
||
|
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
||
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
||
|
* THE SOFTWARE.
|
||
|
*/
|
||
|
|
||
|
#include "qemu/osdep.h"
|
||
|
#include "qapi/error.h"
|
||
|
#include "qemu-common.h"
|
||
|
#include "hw/arm/boot.h"
|
||
|
#include "sysemu/sysemu.h"
|
||
|
#include "qemu/error-report.h"
|
||
|
#include "hw/arm/xnu.h"
|
||
|
#include "hw/loader.h"
|
||
|
#include "hw/arm/xnu_dtb.h"
|
||
|
|
||
|
static uint64_t align_4_high_num(uint64_t num)
|
||
|
{
|
||
|
return (num + (4 - 1)) & ~(4 - 1);
|
||
|
}
|
||
|
|
||
|
static void *align_4_high_ptr(void *ptr)
|
||
|
{
|
||
|
uint64_t num = align_4_high_num((uint64_t)ptr);
|
||
|
return (void *)num;
|
||
|
}
|
||
|
|
||
|
|
||
|
static DTBProp *read_dtb_prop(uint8_t **dtb_blob)
|
||
|
{
|
||
|
if ((NULL == dtb_blob) || (NULL == *dtb_blob)) {
|
||
|
abort();
|
||
|
}
|
||
|
*dtb_blob = align_4_high_ptr(*dtb_blob);
|
||
|
DTBProp *prop = g_new0(DTBProp, 1);
|
||
|
memcpy(&prop->name[0], *dtb_blob, DTB_PROP_NAME_LEN);
|
||
|
*dtb_blob += DTB_PROP_NAME_LEN;
|
||
|
//zero out this flag which sometimes appears in the DT
|
||
|
//normally done by iboot
|
||
|
prop->length = *(uint32_t *)*dtb_blob & ~DT_PROP_FLAG_PLACEHOLDER;
|
||
|
*dtb_blob += sizeof(uint32_t);
|
||
|
if (0 != prop->length) {
|
||
|
prop->value = g_malloc0(prop->length);
|
||
|
if (NULL == prop->value) {
|
||
|
abort();
|
||
|
}
|
||
|
memcpy(&prop->value[0], *dtb_blob, prop->length);
|
||
|
*dtb_blob += prop->length;
|
||
|
}
|
||
|
|
||
|
return prop;
|
||
|
}
|
||
|
|
||
|
static void delete_prop(DTBProp *prop)
|
||
|
{
|
||
|
if (NULL == prop) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if (NULL != prop->value) {
|
||
|
g_free(prop->value);
|
||
|
}
|
||
|
|
||
|
g_free(prop);
|
||
|
}
|
||
|
|
||
|
static DTBNode *read_dtb_node(uint8_t **dtb_blob)
|
||
|
{
|
||
|
if ((NULL == dtb_blob) || (NULL == *dtb_blob)) {
|
||
|
abort();
|
||
|
}
|
||
|
|
||
|
uint32_t i = 0;
|
||
|
|
||
|
*dtb_blob = align_4_high_ptr(*dtb_blob);
|
||
|
DTBNode *node = g_new0(DTBNode, 1);
|
||
|
node->prop_count = *(uint32_t *)*dtb_blob;
|
||
|
*dtb_blob += sizeof(uint32_t);
|
||
|
node->child_node_count = *(uint32_t *)*dtb_blob;
|
||
|
*dtb_blob += sizeof(uint32_t);
|
||
|
|
||
|
if (0 == node->prop_count) {
|
||
|
abort();
|
||
|
}
|
||
|
for (i = 0; i < node->prop_count; i++) {
|
||
|
DTBProp *prop = read_dtb_prop(dtb_blob);
|
||
|
node->props = g_list_append(node->props, prop);
|
||
|
}
|
||
|
for (i = 0; i < node->child_node_count; i++) {
|
||
|
DTBNode *child = read_dtb_node(dtb_blob);
|
||
|
node->child_nodes = g_list_append(node->child_nodes, child);
|
||
|
}
|
||
|
return node;
|
||
|
}
|
||
|
|
||
|
void delete_dtb_node(DTBNode *node)
|
||
|
{
|
||
|
if (NULL == node) {
|
||
|
return;
|
||
|
}
|
||
|
if (NULL != node->props) {
|
||
|
g_list_free_full(node->props, (GDestroyNotify)delete_prop);
|
||
|
}
|
||
|
if (NULL != node->child_nodes) {
|
||
|
g_list_free_full(node->child_nodes, (GDestroyNotify)delete_dtb_node);
|
||
|
}
|
||
|
g_free(node);
|
||
|
}
|
||
|
|
||
|
DTBNode *load_dtb(uint8_t *dtb_blob)
|
||
|
{
|
||
|
DTBNode *root = read_dtb_node(&dtb_blob);
|
||
|
return root;
|
||
|
}
|
||
|
|
||
|
static void save_prop(DTBProp *prop, uint8_t **buf)
|
||
|
{
|
||
|
if ((NULL == prop) || (NULL == buf) || (NULL ==*buf)) {
|
||
|
abort();
|
||
|
}
|
||
|
|
||
|
*buf = align_4_high_ptr(*buf);
|
||
|
memcpy(*buf, &prop->name[0], DTB_PROP_NAME_LEN);
|
||
|
*buf += DTB_PROP_NAME_LEN;
|
||
|
memcpy(*buf, &prop->length, sizeof(uint32_t));
|
||
|
*buf += sizeof(uint32_t);
|
||
|
memcpy(*buf, prop->value, prop->length);
|
||
|
*buf += prop->length;
|
||
|
}
|
||
|
|
||
|
static void save_node(DTBNode *node, uint8_t **buf)
|
||
|
{
|
||
|
if ((NULL == node) || (NULL == buf) || (NULL ==*buf)) {
|
||
|
abort();
|
||
|
}
|
||
|
|
||
|
*buf = align_4_high_ptr(*buf);
|
||
|
|
||
|
memcpy(*buf, &node->prop_count, sizeof(uint32_t));
|
||
|
*buf += sizeof(uint32_t);
|
||
|
memcpy(*buf, &node->child_node_count, sizeof(uint32_t));
|
||
|
*buf += sizeof(uint32_t);
|
||
|
g_list_foreach(node->props, (GFunc)save_prop, buf);
|
||
|
g_list_foreach(node->child_nodes, (GFunc)save_node, buf);
|
||
|
}
|
||
|
|
||
|
void remove_dtb_prop(DTBNode *node, DTBProp *prop)
|
||
|
{
|
||
|
if ((NULL == node) || (NULL == prop)) {
|
||
|
abort();
|
||
|
}
|
||
|
GList *iter;
|
||
|
bool found = false;
|
||
|
for (iter = node->props; iter != NULL; iter = iter->next) {
|
||
|
if (prop == iter->data) {
|
||
|
found = true;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
if (!found) {
|
||
|
abort();
|
||
|
return;
|
||
|
}
|
||
|
delete_prop(prop);
|
||
|
node->props = g_list_delete_link(node->props, iter);
|
||
|
|
||
|
//sanity
|
||
|
if (0 == node->prop_count) {
|
||
|
abort();
|
||
|
}
|
||
|
|
||
|
node->prop_count--;
|
||
|
}
|
||
|
|
||
|
void add_dtb_prop(DTBNode *n, const char *name, uint32_t size, uint8_t *val)
|
||
|
{
|
||
|
if ((NULL == n) || (NULL == name) || (NULL == val)) {
|
||
|
abort();
|
||
|
}
|
||
|
DTBProp *prop = g_new0(DTBProp, 1);
|
||
|
memcpy(&prop->name[0], name, DTB_PROP_NAME_LEN);
|
||
|
prop->length = size;
|
||
|
prop->value = g_malloc0(size);
|
||
|
memcpy(&prop->value[0], val, size);
|
||
|
n->props = g_list_append(n->props, prop);
|
||
|
n->prop_count++;
|
||
|
}
|
||
|
|
||
|
void save_dtb(uint8_t *buf, DTBNode *root)
|
||
|
{
|
||
|
if ((NULL == root) || (NULL == buf)) {
|
||
|
abort();
|
||
|
}
|
||
|
|
||
|
//TODO: handle cases where the buffer is not 4 bytes aligned
|
||
|
//though this is never expected to happen and the code is simpler this
|
||
|
//way
|
||
|
if (align_4_high_ptr(buf) != buf) {
|
||
|
abort();
|
||
|
}
|
||
|
|
||
|
save_node(root, &buf);
|
||
|
}
|
||
|
|
||
|
static uint64_t get_dtb_prop_size(DTBProp *prop)
|
||
|
{
|
||
|
uint64_t size = 0;
|
||
|
|
||
|
if (NULL == prop) {
|
||
|
abort();
|
||
|
}
|
||
|
|
||
|
size = align_4_high_num(sizeof(prop->name) + sizeof(prop->length) +
|
||
|
prop->length);
|
||
|
return size;
|
||
|
}
|
||
|
|
||
|
uint64_t get_dtb_node_buffer_size(DTBNode *node)
|
||
|
{
|
||
|
uint64_t size = 0;
|
||
|
DTBProp *prop = NULL;
|
||
|
DTBNode *child = NULL;
|
||
|
GList *iter = NULL;
|
||
|
|
||
|
if (NULL == node) {
|
||
|
abort();
|
||
|
}
|
||
|
|
||
|
size += sizeof(node->prop_count) + sizeof(node->child_node_count);
|
||
|
|
||
|
for (iter = node->props; iter != NULL; iter = iter->next) {
|
||
|
prop = (DTBProp *)iter->data;
|
||
|
if (NULL == prop) {
|
||
|
abort();
|
||
|
}
|
||
|
size += get_dtb_prop_size(prop);
|
||
|
}
|
||
|
for (iter = node->child_nodes; iter != NULL; iter = iter->next) {
|
||
|
child = (DTBNode *)iter->data;
|
||
|
if (NULL == child) {
|
||
|
abort();
|
||
|
}
|
||
|
size += get_dtb_node_buffer_size(child);
|
||
|
}
|
||
|
return size;
|
||
|
}
|
||
|
|
||
|
DTBProp *get_dtb_prop(DTBNode *node, const char *name)
|
||
|
{
|
||
|
if ((NULL == node) || (NULL == name)) {
|
||
|
abort();
|
||
|
}
|
||
|
|
||
|
GList *iter = NULL;
|
||
|
DTBProp *prop = NULL;
|
||
|
|
||
|
for (iter = node->props; iter != NULL; iter = iter->next) {
|
||
|
prop = (DTBProp *)iter->data;
|
||
|
|
||
|
if (NULL == prop) {
|
||
|
abort();
|
||
|
}
|
||
|
|
||
|
if (0 == strncmp((const char *)&prop->name[0], name,
|
||
|
DTB_PROP_NAME_LEN)) {
|
||
|
return prop;
|
||
|
}
|
||
|
}
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
DTBNode *get_dtb_child_node_by_name(DTBNode *node, const char *name)
|
||
|
{
|
||
|
if ((NULL == node) || (NULL == name)) {
|
||
|
abort();
|
||
|
}
|
||
|
|
||
|
GList *iter = NULL;
|
||
|
DTBProp *prop = NULL;
|
||
|
DTBNode *child = NULL;
|
||
|
|
||
|
for (iter = node->child_nodes; iter != NULL; iter = iter->next) {
|
||
|
child = (DTBNode *)iter->data;
|
||
|
|
||
|
if (NULL == child) {
|
||
|
abort();
|
||
|
}
|
||
|
|
||
|
prop = get_dtb_prop(child, "name");
|
||
|
|
||
|
if (NULL == prop) {
|
||
|
abort();
|
||
|
}
|
||
|
|
||
|
if (0 == strncmp((const char *)prop->value, name, prop->length)) {
|
||
|
return child;
|
||
|
}
|
||
|
}
|
||
|
return NULL;
|
||
|
}
|
||
|
|
||
|
void overwrite_dtb_prop_val(DTBProp *prop, uint8_t chr)
|
||
|
{
|
||
|
uint64_t i = 0;
|
||
|
uint8_t *ptr = prop->value;
|
||
|
for (i = 0; i < prop->length - 1; i++) {
|
||
|
ptr[i] = chr;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void overwrite_dtb_prop_name(DTBProp *prop, uint8_t chr)
|
||
|
{
|
||
|
uint64_t i = 0;
|
||
|
uint8_t *ptr = &prop->name[0];
|
||
|
for (i = 0; i < DTB_PROP_NAME_LEN; i++) {
|
||
|
ptr[i] = chr;
|
||
|
}
|
||
|
}
|