Blame radeon/bof.c

Packit 631bab
/*
Packit 631bab
 * Copyright 2010 Jerome Glisse <glisse@freedesktop.org>
Packit 631bab
 *
Packit 631bab
 * Permission is hereby granted, free of charge, to any person obtaining a
Packit 631bab
 * copy of this software and associated documentation files (the "Software"),
Packit 631bab
 * to deal in the Software without restriction, including without limitation
Packit 631bab
 * on the rights to use, copy, modify, merge, publish, distribute, sub
Packit 631bab
 * license, and/or sell copies of the Software, and to permit persons to whom
Packit 631bab
 * the Software is furnished to do so, subject to the following conditions:
Packit 631bab
 *
Packit 631bab
 * The above copyright notice and this permission notice (including the next
Packit 631bab
 * paragraph) shall be included in all copies or substantial portions of the
Packit 631bab
 * Software.
Packit 631bab
 *
Packit 631bab
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
Packit 631bab
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
Packit 631bab
 * FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL
Packit 631bab
 * THE AUTHOR(S) AND/OR THEIR SUPPLIERS BE LIABLE FOR ANY CLAIM,
Packit 631bab
 * DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
Packit 631bab
 * OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
Packit 631bab
 * USE OR OTHER DEALINGS IN THE SOFTWARE.
Packit 631bab
 *
Packit 631bab
 * Authors:
Packit 631bab
 *      Jerome Glisse
Packit 631bab
 */
Packit 631bab
#include <errno.h>
Packit 631bab
#include <stdlib.h>
Packit 631bab
#include <string.h>
Packit 631bab
#include "bof.h"
Packit 631bab
Packit 631bab
/*
Packit 631bab
 * helpers
Packit 631bab
 */
Packit 631bab
static int bof_entry_grow(bof_t *bof)
Packit 631bab
{
Packit 631bab
	bof_t **array;
Packit 631bab
Packit 631bab
	if (bof->array_size < bof->nentry)
Packit 631bab
		return 0;
Packit 631bab
	array = realloc(bof->array, (bof->nentry + 16) * sizeof(void*));
Packit 631bab
	if (array == NULL)
Packit 631bab
		return -ENOMEM;
Packit 631bab
	bof->array = array;
Packit 631bab
	bof->nentry += 16;
Packit 631bab
	return 0;
Packit 631bab
}
Packit 631bab
Packit 631bab
/*
Packit 631bab
 * object 
Packit 631bab
 */
Packit 631bab
bof_t *bof_object(void)
Packit 631bab
{
Packit 631bab
	bof_t *object;
Packit 631bab
Packit 631bab
	object = calloc(1, sizeof(bof_t));
Packit 631bab
	if (object == NULL)
Packit 631bab
		return NULL;
Packit 631bab
	object->refcount = 1;
Packit 631bab
	object->type = BOF_TYPE_OBJECT;
Packit 631bab
	object->size = 12;
Packit 631bab
	return object;
Packit 631bab
}
Packit 631bab
Packit 631bab
bof_t *bof_object_get(bof_t *object, const char *keyname)
Packit 631bab
{
Packit 631bab
	unsigned i;
Packit 631bab
Packit 631bab
	for (i = 0; i < object->array_size; i += 2) {
Packit 631bab
		if (!strcmp(object->array[i]->value, keyname)) {
Packit 631bab
			return object->array[i + 1];
Packit 631bab
		}
Packit 631bab
	}
Packit 631bab
	return NULL;
Packit 631bab
}
Packit 631bab
Packit 631bab
int bof_object_set(bof_t *object, const char *keyname, bof_t *value)
Packit 631bab
{
Packit 631bab
	bof_t *key;
Packit 631bab
	int r;
Packit 631bab
Packit 631bab
	if (object->type != BOF_TYPE_OBJECT)
Packit 631bab
		return -EINVAL;
Packit 631bab
	r = bof_entry_grow(object);
Packit 631bab
	if (r)
Packit 631bab
		return r;
Packit 631bab
	key = bof_string(keyname);
Packit 631bab
	if (key == NULL)
Packit 631bab
		return -ENOMEM;
Packit 631bab
	object->array[object->array_size++] = key;
Packit 631bab
	object->array[object->array_size++] = value;
Packit 631bab
	object->size += value->size;
Packit 631bab
	object->size += key->size;
Packit 631bab
	bof_incref(value);
Packit 631bab
	return 0;
Packit 631bab
}
Packit 631bab
Packit 631bab
/*
Packit 631bab
 * array
Packit 631bab
 */
Packit 631bab
bof_t *bof_array(void)
Packit 631bab
{
Packit 631bab
	bof_t *array = bof_object();
Packit 631bab
Packit 631bab
	if (array == NULL)
Packit 631bab
		return NULL;
Packit 631bab
	array->type = BOF_TYPE_ARRAY;
Packit 631bab
	array->size = 12;
Packit 631bab
	return array;
Packit 631bab
}
Packit 631bab
Packit 631bab
int bof_array_append(bof_t *array, bof_t *value)
Packit 631bab
{
Packit 631bab
	int r;
Packit 631bab
	if (array->type != BOF_TYPE_ARRAY)
Packit 631bab
		return -EINVAL;
Packit 631bab
	r = bof_entry_grow(array);
Packit 631bab
	if (r)
Packit 631bab
		return r;
Packit 631bab
	array->array[array->array_size++] = value;
Packit 631bab
	array->size += value->size;
Packit 631bab
	bof_incref(value);
Packit 631bab
	return 0;
Packit 631bab
}
Packit 631bab
Packit 631bab
bof_t *bof_array_get(bof_t *bof, unsigned i)
Packit 631bab
{
Packit 631bab
	if (!bof_is_array(bof) || i >= bof->array_size)
Packit 631bab
		return NULL;
Packit 631bab
	return bof->array[i];
Packit 631bab
}
Packit 631bab
Packit 631bab
unsigned bof_array_size(bof_t *bof)
Packit 631bab
{
Packit 631bab
	if (!bof_is_array(bof))
Packit 631bab
		return 0;
Packit 631bab
	return bof->array_size;
Packit 631bab
}
Packit 631bab
Packit 631bab
/*
Packit 631bab
 * blob
Packit 631bab
 */
Packit 631bab
bof_t *bof_blob(unsigned size, void *value)
Packit 631bab
{
Packit 631bab
	bof_t *blob = bof_object();
Packit 631bab
Packit 631bab
	if (blob == NULL)
Packit 631bab
		return NULL;
Packit 631bab
	blob->type = BOF_TYPE_BLOB;
Packit 631bab
	blob->value = calloc(1, size);
Packit 631bab
	if (blob->value == NULL) {
Packit 631bab
		bof_decref(blob);
Packit 631bab
		return NULL;
Packit 631bab
	}
Packit 631bab
	blob->size = size;
Packit 631bab
	memcpy(blob->value, value, size);
Packit 631bab
	blob->size += 12;
Packit 631bab
	return blob;
Packit 631bab
}
Packit 631bab
Packit 631bab
unsigned bof_blob_size(bof_t *bof)
Packit 631bab
{
Packit 631bab
	if (!bof_is_blob(bof))
Packit 631bab
		return 0;
Packit 631bab
	return bof->size - 12;
Packit 631bab
}
Packit 631bab
Packit 631bab
void *bof_blob_value(bof_t *bof)
Packit 631bab
{
Packit 631bab
	if (!bof_is_blob(bof))
Packit 631bab
		return NULL;
Packit 631bab
	return bof->value;
Packit 631bab
}
Packit 631bab
Packit 631bab
/*
Packit 631bab
 * string
Packit 631bab
 */
Packit 631bab
bof_t *bof_string(const char *value)
Packit 631bab
{
Packit 631bab
	bof_t *string = bof_object();
Packit 631bab
Packit 631bab
	if (string == NULL)
Packit 631bab
		return NULL;
Packit 631bab
	string->type = BOF_TYPE_STRING;
Packit 631bab
	string->size = strlen(value) + 1;
Packit 631bab
	string->value = calloc(1, string->size);
Packit 631bab
	if (string->value == NULL) {
Packit 631bab
		bof_decref(string);
Packit 631bab
		return NULL;
Packit 631bab
	}
Packit 631bab
	strcpy(string->value, value);
Packit 631bab
	string->size += 12;
Packit 631bab
	return string;
Packit 631bab
}
Packit 631bab
Packit 631bab
/*
Packit 631bab
 *  int32
Packit 631bab
 */
Packit 631bab
bof_t *bof_int32(int32_t value)
Packit 631bab
{
Packit 631bab
	bof_t *int32 = bof_object();
Packit 631bab
Packit 631bab
	if (int32 == NULL)
Packit 631bab
		return NULL;
Packit 631bab
	int32->type = BOF_TYPE_INT32;
Packit 631bab
	int32->size = 4;
Packit 631bab
	int32->value = calloc(1, int32->size);
Packit 631bab
	if (int32->value == NULL) {
Packit 631bab
		bof_decref(int32);
Packit 631bab
		return NULL;
Packit 631bab
	}
Packit 631bab
	memcpy(int32->value, &value, 4);
Packit 631bab
	int32->size += 12;
Packit 631bab
	return int32;
Packit 631bab
}
Packit 631bab
Packit 631bab
int32_t bof_int32_value(bof_t *bof)
Packit 631bab
{
Packit 631bab
	return *((uint32_t*)bof->value);
Packit 631bab
}
Packit 631bab
Packit 631bab
/*
Packit 631bab
 *  common
Packit 631bab
 */
Packit 631bab
static void bof_indent(int level)
Packit 631bab
{
Packit 631bab
	int i;
Packit 631bab
Packit 631bab
	for (i = 0; i < level; i++)
Packit 631bab
		fprintf(stderr, " ");
Packit 631bab
}
Packit 631bab
Packit 631bab
static void bof_print_bof(bof_t *bof, int level, int entry)
Packit 631bab
{
Packit 631bab
	bof_indent(level);
Packit 631bab
	if (bof == NULL) {
Packit 631bab
		fprintf(stderr, "--NULL-- for entry %d\n", entry);
Packit 631bab
		return;
Packit 631bab
	}
Packit 631bab
	switch (bof->type) {
Packit 631bab
	case BOF_TYPE_STRING:
Packit 631bab
		fprintf(stderr, "%p string [%s %d]\n", bof, (char*)bof->value, bof->size);
Packit 631bab
		break;
Packit 631bab
	case BOF_TYPE_INT32:
Packit 631bab
		fprintf(stderr, "%p int32 [%d %d]\n", bof, *(int*)bof->value, bof->size);
Packit 631bab
		break;
Packit 631bab
	case BOF_TYPE_BLOB:
Packit 631bab
		fprintf(stderr, "%p blob [%d]\n", bof, bof->size);
Packit 631bab
		break;
Packit 631bab
	case BOF_TYPE_NULL:
Packit 631bab
		fprintf(stderr, "%p null [%d]\n", bof, bof->size);
Packit 631bab
		break;
Packit 631bab
	case BOF_TYPE_OBJECT:
Packit 631bab
		fprintf(stderr, "%p object [%d %d]\n", bof, bof->array_size / 2, bof->size);
Packit 631bab
		break;
Packit 631bab
	case BOF_TYPE_ARRAY:
Packit 631bab
		fprintf(stderr, "%p array [%d %d]\n", bof, bof->array_size, bof->size);
Packit 631bab
		break;
Packit 631bab
	default:
Packit 631bab
		fprintf(stderr, "%p unknown [%d]\n", bof, bof->type);
Packit 631bab
		return;
Packit 631bab
	}
Packit 631bab
}
Packit 631bab
Packit 631bab
static void bof_print_rec(bof_t *bof, int level, int entry)
Packit 631bab
{
Packit 631bab
	unsigned i;
Packit 631bab
Packit 631bab
	bof_print_bof(bof, level, entry);
Packit 631bab
	for (i = 0; i < bof->array_size; i++) {
Packit 631bab
		bof_print_rec(bof->array[i], level + 2, i);
Packit 631bab
	}
Packit 631bab
}
Packit 631bab
Packit 631bab
void bof_print(bof_t *bof)
Packit 631bab
{
Packit 631bab
	bof_print_rec(bof, 0, 0);
Packit 631bab
}
Packit 631bab
Packit 631bab
static int bof_read(bof_t *root, FILE *file, long end, int level)
Packit 631bab
{
Packit 631bab
	bof_t *bof = NULL;
Packit 631bab
	int r;
Packit 631bab
Packit 631bab
	if (ftell(file) >= end) {
Packit 631bab
		return 0;
Packit 631bab
	}
Packit 631bab
	r = bof_entry_grow(root);
Packit 631bab
	if (r)
Packit 631bab
		return r;
Packit 631bab
	bof = bof_object();
Packit 631bab
	if (bof == NULL)
Packit 631bab
		return -ENOMEM;
Packit 631bab
	bof->offset = ftell(file);
Packit 631bab
	r = fread(&bof->type, 4, 1, file);
Packit 631bab
	if (r != 1)
Packit 631bab
		goto out_err;
Packit 631bab
	r = fread(&bof->size, 4, 1, file);
Packit 631bab
	if (r != 1)
Packit 631bab
		goto out_err;
Packit 631bab
	r = fread(&bof->array_size, 4, 1, file);
Packit 631bab
	if (r != 1)
Packit 631bab
		goto out_err;
Packit 631bab
	switch (bof->type) {
Packit 631bab
	case BOF_TYPE_STRING:
Packit 631bab
	case BOF_TYPE_INT32:
Packit 631bab
	case BOF_TYPE_BLOB:
Packit 631bab
		bof->value = calloc(1, bof->size - 12);
Packit 631bab
		if (bof->value == NULL) {
Packit 631bab
			goto out_err;
Packit 631bab
		}
Packit 631bab
		r = fread(bof->value, bof->size - 12, 1, file);
Packit 631bab
		if (r != 1) {
Packit 631bab
			fprintf(stderr, "error reading %d\n", bof->size - 12);
Packit 631bab
			goto out_err;
Packit 631bab
		}
Packit 631bab
		break;
Packit 631bab
	case BOF_TYPE_NULL:
Packit 631bab
		return 0;
Packit 631bab
	case BOF_TYPE_OBJECT:
Packit 631bab
	case BOF_TYPE_ARRAY:
Packit 631bab
		r = bof_read(bof, file, bof->offset + bof->size, level + 2);
Packit 631bab
		if (r)
Packit 631bab
			goto out_err;
Packit 631bab
		break;
Packit 631bab
	default:
Packit 631bab
		fprintf(stderr, "invalid type %d\n", bof->type);
Packit 631bab
		goto out_err;
Packit 631bab
	}
Packit 631bab
	root->array[root->centry++] = bof;
Packit 631bab
	return bof_read(root, file, end, level);
Packit 631bab
out_err:
Packit 631bab
	bof_decref(bof);
Packit 631bab
	return -EINVAL;
Packit 631bab
}
Packit 631bab
Packit 631bab
bof_t *bof_load_file(const char *filename)
Packit 631bab
{
Packit 631bab
	bof_t *root = bof_object();
Packit 631bab
	int r;
Packit 631bab
Packit 631bab
	if (root == NULL) {
Packit 631bab
		fprintf(stderr, "%s failed to create root object\n", __func__);
Packit 631bab
		return NULL;
Packit 631bab
	}
Packit 631bab
	root->file = fopen(filename, "r");
Packit 631bab
	if (root->file == NULL)
Packit 631bab
		goto out_err;
Packit 631bab
	r = fseek(root->file, 0L, SEEK_SET);
Packit 631bab
	if (r) {
Packit 631bab
		fprintf(stderr, "%s failed to seek into file %s\n", __func__, filename);
Packit 631bab
		goto out_err;
Packit 631bab
	}
Packit 631bab
	root->offset = ftell(root->file);
Packit 631bab
	r = fread(&root->type, 4, 1, root->file);
Packit 631bab
	if (r != 1)
Packit 631bab
		goto out_err;
Packit 631bab
	r = fread(&root->size, 4, 1, root->file);
Packit 631bab
	if (r != 1)
Packit 631bab
		goto out_err;
Packit 631bab
	r = fread(&root->array_size, 4, 1, root->file);
Packit 631bab
	if (r != 1)
Packit 631bab
		goto out_err;
Packit 631bab
	r = bof_read(root, root->file, root->offset + root->size, 2);
Packit 631bab
	if (r)
Packit 631bab
		goto out_err;
Packit 631bab
	return root;
Packit 631bab
out_err:
Packit 631bab
	bof_decref(root);
Packit 631bab
	return NULL;
Packit 631bab
}
Packit 631bab
Packit 631bab
void bof_incref(bof_t *bof)
Packit 631bab
{
Packit 631bab
	bof->refcount++;
Packit 631bab
}
Packit 631bab
Packit 631bab
void bof_decref(bof_t *bof)
Packit 631bab
{
Packit 631bab
	unsigned i;
Packit 631bab
Packit 631bab
	if (bof == NULL)
Packit 631bab
		return;
Packit 631bab
	if (--bof->refcount > 0)
Packit 631bab
		return;
Packit 631bab
	for (i = 0; i < bof->array_size; i++) {
Packit 631bab
		bof_decref(bof->array[i]);
Packit 631bab
		bof->array[i] = NULL;
Packit 631bab
	}
Packit 631bab
	bof->array_size = 0;
Packit 631bab
	if (bof->file) {
Packit 631bab
		fclose(bof->file);
Packit 631bab
		bof->file = NULL;
Packit 631bab
	}
Packit 631bab
	free(bof->array);
Packit 631bab
	free(bof->value);
Packit 631bab
	free(bof);
Packit 631bab
}
Packit 631bab
Packit 631bab
static int bof_file_write(bof_t *bof, FILE *file)
Packit 631bab
{
Packit 631bab
	unsigned i;
Packit 631bab
	int r;
Packit 631bab
Packit 631bab
	r = fwrite(&bof->type, 4, 1, file);
Packit 631bab
	if (r != 1)
Packit 631bab
		return -EINVAL;
Packit 631bab
	r = fwrite(&bof->size, 4, 1, file);
Packit 631bab
	if (r != 1)
Packit 631bab
		return -EINVAL;
Packit 631bab
	r = fwrite(&bof->array_size, 4, 1, file);
Packit 631bab
	if (r != 1)
Packit 631bab
		return -EINVAL;
Packit 631bab
	switch (bof->type) {
Packit 631bab
	case BOF_TYPE_NULL:
Packit 631bab
		if (bof->size)
Packit 631bab
			return -EINVAL;
Packit 631bab
		break;
Packit 631bab
	case BOF_TYPE_STRING:
Packit 631bab
	case BOF_TYPE_INT32:
Packit 631bab
	case BOF_TYPE_BLOB:
Packit 631bab
		r = fwrite(bof->value, bof->size - 12, 1, file);
Packit 631bab
		if (r != 1)
Packit 631bab
			return -EINVAL;
Packit 631bab
		break;
Packit 631bab
	case BOF_TYPE_OBJECT:
Packit 631bab
	case BOF_TYPE_ARRAY:
Packit 631bab
		for (i = 0; i < bof->array_size; i++) {
Packit 631bab
			r = bof_file_write(bof->array[i], file);
Packit 631bab
			if (r)
Packit 631bab
				return r;
Packit 631bab
		}
Packit 631bab
		break;
Packit 631bab
	default:
Packit 631bab
		return -EINVAL;
Packit 631bab
	}
Packit 631bab
	return 0;
Packit 631bab
}
Packit 631bab
Packit 631bab
int bof_dump_file(bof_t *bof, const char *filename)
Packit 631bab
{
Packit 631bab
	unsigned i;
Packit 631bab
	int r = 0;
Packit 631bab
Packit 631bab
	if (bof->file) {
Packit 631bab
		fclose(bof->file);
Packit 631bab
		bof->file = NULL;
Packit 631bab
	}
Packit 631bab
	bof->file = fopen(filename, "w");
Packit 631bab
	if (bof->file == NULL) {
Packit 631bab
		fprintf(stderr, "%s failed to open file %s\n", __func__, filename);
Packit 631bab
		r = -EINVAL;
Packit 631bab
		goto out_err;
Packit 631bab
	}
Packit 631bab
	r = fseek(bof->file, 0L, SEEK_SET);
Packit 631bab
	if (r) {
Packit 631bab
		fprintf(stderr, "%s failed to seek into file %s\n", __func__, filename);
Packit 631bab
		goto out_err;
Packit 631bab
	}
Packit 631bab
	r = fwrite(&bof->type, 4, 1, bof->file);
Packit 631bab
	if (r != 1)
Packit 631bab
		goto out_err;
Packit 631bab
	r = fwrite(&bof->size, 4, 1, bof->file);
Packit 631bab
	if (r != 1)
Packit 631bab
		goto out_err;
Packit 631bab
	r = fwrite(&bof->array_size, 4, 1, bof->file);
Packit 631bab
	if (r != 1)
Packit 631bab
		goto out_err;
Packit 631bab
	for (i = 0; i < bof->array_size; i++) {
Packit 631bab
		r = bof_file_write(bof->array[i], bof->file);
Packit 631bab
		if (r)
Packit 631bab
			return r;
Packit 631bab
	}
Packit 631bab
out_err:
Packit 631bab
	fclose(bof->file);
Packit 631bab
	bof->file = NULL;
Packit 631bab
	return r;
Packit 631bab
}