Blame src/xb-stack.c

Packit caecb6
/*
Packit caecb6
 * Copyright (C) 2018 Richard Hughes <richard@hughsie.com>
Packit caecb6
 *
Packit caecb6
 * SPDX-License-Identifier: LGPL-2.1+
Packit caecb6
 */
Packit caecb6
Packit caecb6
#define G_LOG_DOMAIN				"XbMachine"
Packit caecb6
Packit caecb6
#include "config.h"
Packit caecb6
Packit caecb6
#include <gio/gio.h>
Packit caecb6
Packit caecb6
#include "xb-opcode-private.h"
Packit caecb6
#include "xb-stack-private.h"
Packit caecb6
Packit caecb6
struct _XbStack {
Packit caecb6
	gint		 ref;
Packit caecb6
	guint		 pos;
Packit caecb6
	guint		 max_size;
Packit caecb6
	XbOpcode	*opcodes[];	/* allocated as part of XbStack */
Packit caecb6
};
Packit caecb6
Packit caecb6
/**
Packit caecb6
 * xb_stack_unref:
Packit caecb6
 * @self: a #XbStack
Packit caecb6
 *
Packit caecb6
 * Decrements the reference count of the stack, freeing the object when the
Packit caecb6
 * refcount drops to zero.
Packit caecb6
 *
Packit caecb6
 * Since: 0.1.3
Packit caecb6
 **/
Packit caecb6
void
Packit caecb6
xb_stack_unref (XbStack *self)
Packit caecb6
{
Packit caecb6
	g_assert (self->ref > 0);
Packit caecb6
	if (--self->ref > 0)
Packit caecb6
		return;
Packit caecb6
	for (guint i = 0; i < self->pos; i++)
Packit caecb6
		xb_opcode_unref (self->opcodes[i]);
Packit caecb6
	g_free (self);
Packit caecb6
}
Packit caecb6
Packit caecb6
/**
Packit caecb6
 * xb_stack_ref:
Packit caecb6
 * @self: a #XbStack
Packit caecb6
 *
Packit caecb6
 * Increments the refcount of the stack.
Packit caecb6
 *
Packit caecb6
 * Returns: (transfer none): the original @self #XbStack instance
Packit caecb6
 *
Packit caecb6
 * Since: 0.1.3
Packit caecb6
 **/
Packit caecb6
XbStack *
Packit caecb6
xb_stack_ref (XbStack *self)
Packit caecb6
{
Packit caecb6
	self->ref++;
Packit caecb6
	return self;
Packit caecb6
}
Packit caecb6
Packit caecb6
Packit caecb6
/**
Packit caecb6
 * xb_stack_pop:
Packit caecb6
 * @self: a #XbStack
Packit caecb6
 *
Packit caecb6
 * Pops an opcode off the stack.
Packit caecb6
 *
Packit caecb6
 * Returns: (transfer full): a #XbOpcode
Packit caecb6
 *
Packit caecb6
 * Since: 0.1.3
Packit caecb6
 **/
Packit caecb6
XbOpcode *
Packit caecb6
xb_stack_pop (XbStack *self)
Packit caecb6
{
Packit caecb6
	if (self->pos == 0)
Packit caecb6
		return NULL;
Packit caecb6
	return self->opcodes[--self->pos];
Packit caecb6
}
Packit caecb6
Packit caecb6
/* private */
Packit caecb6
GPtrArray *
Packit caecb6
xb_stack_steal_all (XbStack *self)
Packit caecb6
{
Packit caecb6
	GPtrArray *array;
Packit caecb6
Packit caecb6
	/* array takes ownership of the opcodes */
Packit caecb6
	array = g_ptr_array_new_with_free_func ((GDestroyNotify) xb_opcode_unref);
Packit caecb6
	for (guint i = 0; i < self->pos; i++)
Packit caecb6
		g_ptr_array_add (array, self->opcodes[i]);
Packit caecb6
	self->pos = 0;
Packit caecb6
	return array;
Packit caecb6
}
Packit caecb6
Packit caecb6
/**
Packit caecb6
 * xb_stack_peek:
Packit caecb6
 * @self: a #XbStack
Packit caecb6
 * @idx: index
Packit caecb6
 *
Packit caecb6
 * Peeks an opcode from the stack.
Packit caecb6
 *
Packit caecb6
 * Returns: (transfer none): a #XbOpcode
Packit caecb6
 *
Packit caecb6
 * Since: 0.1.3
Packit caecb6
 **/
Packit caecb6
XbOpcode *
Packit caecb6
xb_stack_peek (XbStack *self, guint idx)
Packit caecb6
{
Packit caecb6
	if (idx >= self->pos)
Packit caecb6
		return NULL;
Packit caecb6
	return self->opcodes[idx];
Packit caecb6
}
Packit caecb6
Packit caecb6
/* private */
Packit caecb6
gboolean
Packit caecb6
xb_stack_push_bool (XbStack *self, gboolean val)
Packit caecb6
{
Packit caecb6
	return xb_stack_push_steal (self, xb_opcode_bool_new (val));
Packit caecb6
}
Packit caecb6
Packit caecb6
/* private */
Packit caecb6
XbOpcode *
Packit caecb6
xb_stack_peek_head (XbStack *self)
Packit caecb6
{
Packit caecb6
	if (self->pos == 0)
Packit caecb6
		return NULL;
Packit caecb6
	return self->opcodes[0];
Packit caecb6
}
Packit caecb6
Packit caecb6
/* private */
Packit caecb6
XbOpcode *
Packit caecb6
xb_stack_peek_tail (XbStack *self)
Packit caecb6
{
Packit caecb6
	if (self->pos == 0)
Packit caecb6
		return NULL;
Packit caecb6
	return self->opcodes[self->pos - 1];
Packit caecb6
}
Packit caecb6
Packit caecb6
/**
Packit caecb6
 * xb_stack_push:
Packit caecb6
 * @self: a #XbStack
Packit caecb6
 * @opcode: a #XbOpcode
Packit caecb6
 *
Packit caecb6
 * Pushes a new opcode onto the end of the stack
Packit caecb6
 *
Packit caecb6
 * Returns: %TRUE if the opcode was stored on the stack
Packit caecb6
 *
Packit caecb6
 * Since: 0.1.3
Packit caecb6
 **/
Packit caecb6
gboolean
Packit caecb6
xb_stack_push (XbStack *self, XbOpcode *opcode)
Packit caecb6
{
Packit caecb6
	if (self->pos >= self->max_size)
Packit caecb6
		return FALSE;
Packit caecb6
	self->opcodes[self->pos++] = xb_opcode_ref (opcode);
Packit caecb6
	return TRUE;
Packit caecb6
}
Packit caecb6
Packit caecb6
/**
Packit caecb6
 * xb_stack_push_steal:
Packit caecb6
 * @self: a #XbStack
Packit caecb6
 * @opcode: a #XbOpcode, which is consumed
Packit caecb6
 *
Packit caecb6
 * Pushes a new opcode onto the end of the stack
Packit caecb6
 *
Packit caecb6
 * Returns: %TRUE if the opcode was stored on the stack
Packit caecb6
 *
Packit caecb6
 * Since: 0.1.3
Packit caecb6
 **/
Packit caecb6
gboolean
Packit caecb6
xb_stack_push_steal (XbStack *self, XbOpcode *opcode)
Packit caecb6
{
Packit caecb6
	if (self->pos >= self->max_size)
Packit caecb6
		return FALSE;
Packit caecb6
	self->opcodes[self->pos++] = opcode;
Packit caecb6
	return TRUE;
Packit caecb6
}
Packit caecb6
Packit caecb6
/**
Packit caecb6
 * xb_stack_get_size:
Packit caecb6
 * @self: a #XbStack
Packit caecb6
 *
Packit caecb6
 * Gets the current size of the stack.
Packit caecb6
 *
Packit caecb6
 * Returns: integer, where 0 is "empty"
Packit caecb6
 *
Packit caecb6
 * Since: 0.1.3
Packit caecb6
 **/
Packit caecb6
guint
Packit caecb6
xb_stack_get_size (XbStack *self)
Packit caecb6
{
Packit caecb6
	return self->pos;
Packit caecb6
}
Packit caecb6
Packit caecb6
/**
Packit caecb6
 * xb_stack_get_max_size:
Packit caecb6
 * @self: a #XbStack
Packit caecb6
 *
Packit caecb6
 * Gets the maximum size of the stack.
Packit caecb6
 *
Packit caecb6
 * Returns: integer
Packit caecb6
 *
Packit caecb6
 * Since: 0.1.3
Packit caecb6
 **/
Packit caecb6
guint
Packit caecb6
xb_stack_get_max_size (XbStack *self)
Packit caecb6
{
Packit caecb6
	return self->max_size;
Packit caecb6
}
Packit caecb6
Packit caecb6
/**
Packit caecb6
 * xb_stack_to_string:
Packit caecb6
 * @self: a #XbStack
Packit caecb6
 *
Packit caecb6
 * Returns a string representing a stack.
Packit caecb6
 *
Packit caecb6
 * Returns: text
Packit caecb6
 *
Packit caecb6
 * Since: 0.1.4
Packit caecb6
 **/
Packit caecb6
gchar *
Packit caecb6
xb_stack_to_string (XbStack *self)
Packit caecb6
{
Packit caecb6
	GString *str = g_string_new (NULL);
Packit caecb6
	for (guint i = 0; i < self->pos; i++) {
Packit caecb6
		g_autofree gchar *tmp = xb_opcode_to_string (self->opcodes[i]);
Packit caecb6
		g_string_append_printf (str, "%s,", tmp);
Packit caecb6
	}
Packit caecb6
	if (str->len > 0)
Packit caecb6
		g_string_truncate (str, str->len - 1);
Packit caecb6
	return g_string_free (str, FALSE);
Packit caecb6
}
Packit caecb6
Packit caecb6
/**
Packit caecb6
 * xb_stack_new:
Packit caecb6
 * @max_size: maximum size of the stack
Packit caecb6
 *
Packit caecb6
 * Creates a stack for the XbMachine request. Only #XbOpcode's can be pushed and
Packit caecb6
 * popped from the stack.
Packit caecb6
 *
Packit caecb6
 * Returns: (transfer full): a #XbStack
Packit caecb6
 *
Packit caecb6
 * Since: 0.1.3
Packit caecb6
 **/
Packit caecb6
XbStack *
Packit caecb6
xb_stack_new (guint max_size)
Packit caecb6
{
Packit caecb6
	XbStack *self = g_malloc0 (sizeof(XbStack) + max_size * sizeof(XbOpcode*));
Packit caecb6
	self->ref = 1;
Packit caecb6
	self->max_size = max_size;
Packit caecb6
	return self;
Packit caecb6
}
Packit caecb6
Packit caecb6
GType
Packit caecb6
xb_stack_get_type (void)
Packit caecb6
{
Packit caecb6
	static GType type = 0;
Packit caecb6
	if (G_UNLIKELY (!type)) {
Packit caecb6
		type = g_boxed_type_register_static ("XbStack",
Packit caecb6
						     (GBoxedCopyFunc) xb_stack_ref,
Packit caecb6
						     (GBoxedFreeFunc) xb_stack_unref);
Packit caecb6
	}
Packit caecb6
	return type;
Packit caecb6
}