Logo Search packages:      
Sourcecode: jackd2 version File versions  Download package

controller_iface_configure.c

/* -*- Mode: C ; c-basic-offset: 4 -*- */
/*
    Copyright (C) 2007,2008 Nedko Arnaudov
    Copyright (C) 2007-2008 Juuso Alasuutari

    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 2 of the License.

    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, write to the Free Software
    Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.

*/

#if defined(HAVE_CONFIG_H)
#include "config.h"
#endif

#include <stdint.h>
#include <string.h>
#include <assert.h>
#include <dbus/dbus.h>

#include "jackdbus.h"
#include "controller_internal.h"
#include "xml.h"

#define PTNODE_ENGINE    "engine"
#define PTNODE_DRIVER    "driver"
#define PTNODE_DRIVERS   "drivers"
#define PTNODE_INTERNALS "internals"

#define ENGINE_DRIVER_PARAMETER_NAME         "driver"
#define ENGINE_DRIVER_PARAMETER_TYPE         JackParamString
#define ENGINE_DRIVER_PARAMETER_SHORT_DESCR  "Driver to use"
#define ENGINE_DRIVER_PARAMETER_LONG_DESCR   ""

00044 struct parameter_info
{
    unsigned char type;
    const char * name;
    const char * short_decr;
    const char * long_descr;
};

unsigned char jack_controller_dbus_types[JACK_PARAM_MAX] =
{
    [JackParamInt] = DBUS_TYPE_INT32,
    [JackParamUInt] = DBUS_TYPE_UINT32,
    [JackParamChar] = DBUS_TYPE_BYTE,
    [JackParamString] = DBUS_TYPE_STRING,
    [JackParamBool] = DBUS_TYPE_BOOLEAN,
};

const char *jack_controller_dbus_type_signatures[JACK_PARAM_MAX] =
{
    [JackParamInt] = DBUS_TYPE_INT32_AS_STRING,
    [JackParamUInt] = DBUS_TYPE_UINT32_AS_STRING,
    [JackParamChar] = DBUS_TYPE_BYTE_AS_STRING,
    [JackParamString] = DBUS_TYPE_STRING_AS_STRING,
    [JackParamBool] = DBUS_TYPE_BOOLEAN_AS_STRING,
};

#define PARAM_TYPE_JACK_TO_DBUS(_) jack_controller_dbus_types[_]
#define PARAM_TYPE_JACK_TO_DBUS_SIGNATURE(_) jack_controller_dbus_type_signatures[_]

static
bool
jack_controller_jack_to_dbus_variant(
    jackctl_param_type_t type,
    const union jackctl_parameter_value *value_ptr,
    message_arg_t  *dbusv_ptr)
{
    switch (type)
    {
    case JackParamInt:
        dbusv_ptr->int32 = (dbus_int32_t)value_ptr->i;
        return true;
    case JackParamUInt:
        dbusv_ptr->uint32 = (dbus_uint32_t)value_ptr->ui;
        return true;
    case JackParamChar:
        dbusv_ptr->byte = value_ptr->c;
        return true;
    case JackParamString:
        dbusv_ptr->string = value_ptr->str;
        return true;
    case JackParamBool:
        dbusv_ptr->boolean = (dbus_bool_t)value_ptr->b;
        return true;
    }

    jack_error("Unknown JACK parameter type %i", (int)type);
    assert(0);
    return false;
}

static
bool
jack_controller_dbus_to_jack_variant(
    int type,
    const message_arg_t *dbusv_ptr,
    union jackctl_parameter_value *value_ptr)
{
    size_t len;

    switch (type)
    {
    case DBUS_TYPE_INT32:
        value_ptr->i = dbusv_ptr->int32;
        return true;
    case DBUS_TYPE_UINT32:
        value_ptr->ui = dbusv_ptr->uint32;
        return true;
    case DBUS_TYPE_BYTE:
        value_ptr->c = dbusv_ptr->byte;
        return true;
    case DBUS_TYPE_STRING:
        len = strlen(dbusv_ptr->string);
        if (len > JACK_PARAM_STRING_MAX)
        {
            jack_error("Parameter string value is too long (%u)", (unsigned int)len);
            return false;
        }
        memcpy(value_ptr->str, dbusv_ptr->string, len + 1);

        return true;
    case DBUS_TYPE_BOOLEAN:
        value_ptr->b = dbusv_ptr->boolean;
        return true;
    }

    jack_error("Unknown D-Bus parameter type %i", (int)type);
    return false;
}

/*
 * Construct a return message for a Get[Driver|Engine]ParameterValue method call.
 *
 * The operation can only fail due to lack of memory, in which case
 * there's no sense in trying to construct an error return. Instead,
 * call->reply will be set to NULL and handled in send_method_return().
 */
static void
jack_dbus_construct_method_return_parameter(
    struct jack_dbus_method_call * call,
    dbus_bool_t is_set,
    int type,
    const char *signature,
    message_arg_t default_value,
    message_arg_t value)
{
    DBusMessageIter iter;

    /* Create a new method return message. */
    call->reply = dbus_message_new_method_return (call->message);
    if (!call->reply)
    {
        goto fail;
    }

    dbus_message_iter_init_append (call->reply, &iter);

    /* Append the is_set argument. */
    if (!dbus_message_iter_append_basic (&iter, DBUS_TYPE_BOOLEAN, (const void *) &is_set))
    {
        goto fail_unref;
    }

    /* Append the 'default' and 'value' arguments. */
    if (!jack_dbus_message_append_variant(&iter, type, signature, &default_value))
    {
        goto fail_unref;
    }
    if (!jack_dbus_message_append_variant(&iter, type, signature, &value))
    {
        goto fail_unref;
    }

    return;

fail_unref:
    dbus_message_unref (call->reply);
    call->reply = NULL;

fail:
    jack_error ("Ran out of memory trying to construct method return");
}

static
bool
jack_controller_dbus_get_parameter_address_ex(
    struct jack_dbus_method_call * call,
    DBusMessageIter * iter_ptr,
    const char ** address_array)
{
    const char * signature;
    DBusMessageIter array_iter;
    int type;
    int index;

    if (!dbus_message_iter_init(call->message, iter_ptr))
    {
        jack_dbus_error(
            call,
            JACK_DBUS_ERROR_INVALID_ARGS,
            "Invalid arguments to method '%s'. No input arguments found.",
            call->method_name);
        return false;
    }

    signature = dbus_message_iter_get_signature(iter_ptr);
    if (signature == NULL)
    {
        jack_error("dbus_message_iter_get_signature() failed");
        return false;
    }

    if (strcmp(signature, "as") != 0)
    {
        jack_dbus_error(
            call,
            JACK_DBUS_ERROR_INVALID_ARGS,
            "Invalid arguments to method '%s'. Input arguments signature '%s', must begin with 'as'.",
            call->method_name,
            signature);
        return false;
    }

    dbus_message_iter_recurse(iter_ptr, &array_iter);

    index = 0;
    while ((type = dbus_message_iter_get_arg_type(&array_iter)) != DBUS_TYPE_INVALID)
    {
        if (index == 3)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_INVALID_ARGS,
                "Invalid arguments to method '%s'. Parameter address array must contain not more than three elements.",
                call->method_name);
            return false;
        }

        ;
        if (type != DBUS_TYPE_STRING)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_FATAL,
                "Internal error when parsing parameter address of method '%s'. Address array element type '%c' is not string type.",
                call->method_name,
                type);
            return false;
        }

        dbus_message_iter_get_basic(&array_iter, address_array + index);
        //jack_info("address component: '%s'", address_array[index]);

        dbus_message_iter_next(&array_iter);
        index++;
    }

    while (index < 3)
    {
        address_array[index] = NULL;
        index++;
    }

    return true;
}

static
bool
jack_controller_dbus_get_parameter_address(
    struct jack_dbus_method_call * call,
    const char ** address_array)
{
    DBusMessageIter iter;
    bool ret;

    ret = jack_controller_dbus_get_parameter_address_ex(call, &iter, address_array);
    if (ret && dbus_message_iter_has_next(&iter))
    {
        jack_dbus_error(
            call,
            JACK_DBUS_ERROR_INVALID_ARGS,
            "Invalid arguments to method '%s'. Input arguments signature must be 'as'.",
            call->method_name);
        return false;
    }

    return ret;
}

#define controller_ptr ((struct jack_controller *)call->context)

static
bool
jack_controller_fill_parameter_names(
    struct jack_dbus_method_call * call,
    DBusMessageIter * iter_ptr,
    const char * special_first,
    const JSList * parameters_list)
{
    DBusMessageIter array_iter;
    const char * param_name;

    if (!dbus_message_iter_open_container(iter_ptr, DBUS_TYPE_ARRAY, "s", &array_iter))
    {
        return false;
    }

    if (special_first != NULL)
    {
        if (!dbus_message_iter_append_basic(&array_iter, DBUS_TYPE_STRING, &special_first))
        {
            dbus_message_iter_close_container(iter_ptr, &array_iter);
            return false;
        }
    }

    /* Append parameter descriptions to the array. */
    while (parameters_list != NULL)
    {
        param_name = jackctl_parameter_get_name(parameters_list->data);
        if (!dbus_message_iter_append_basic(&array_iter, DBUS_TYPE_STRING, &param_name))
        {
            dbus_message_iter_close_container(iter_ptr, &array_iter);
            return false;
        }

        parameters_list = jack_slist_next(parameters_list);
    }

    return dbus_message_iter_close_container(iter_ptr, &array_iter);
}

static
void
jack_controller_dbus_read_container(
    struct jack_dbus_method_call * call)
{
    const char * address[3];
    dbus_bool_t leaf;
    DBusMessageIter iter;
    DBusMessageIter array_iter;
    const char * child_name;
    unsigned int index;
    jackctl_internal_t * internal;
    jackctl_driver_t * driver;

    //jack_info("jack_controller_dbus_read_container() called");

    if (!jack_controller_dbus_get_parameter_address(call, address))
    {
        /* The method call had invalid arguments meaning that
         * jack_controller_dbus_get_parameter_address() has
         * constructed an error for us. */
        return;
    }

    //jack_info("address is '%s':'%s':'%s'", address[0], address[1], address[2]);

    /* Create a new method return message. */
    call->reply = dbus_message_new_method_return(call->message);
    if (!call->reply)
    {
        goto oom;
    }

    dbus_message_iter_init_append(call->reply, &iter);

    if (address[0] == NULL)     /* root node */
    {
        //jack_info("reading root container");

        leaf = false;
        if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_BOOLEAN, &leaf))
        {
            goto oom_unref;
        }

        if (!dbus_message_iter_open_container(&iter, DBUS_TYPE_ARRAY, "s", &array_iter))
        {
            goto oom_unref;
        }

        child_name = PTNODE_ENGINE;
        if (!dbus_message_iter_append_basic(&array_iter, DBUS_TYPE_STRING, &child_name))
        {
            goto oom_close_unref;
        }

        child_name = PTNODE_DRIVER;
        if (!dbus_message_iter_append_basic(&array_iter, DBUS_TYPE_STRING, &child_name))
        {
            goto oom_close_unref;
        }

        child_name = PTNODE_DRIVERS;
        if (!dbus_message_iter_append_basic(&array_iter, DBUS_TYPE_STRING, &child_name))
        {
            goto oom_close_unref;
        }

        child_name = PTNODE_INTERNALS;
        if (!dbus_message_iter_append_basic(&array_iter, DBUS_TYPE_STRING, &child_name))
        {
            goto oom_close_unref;
        }

        dbus_message_iter_close_container(&iter, &array_iter);

        return;
    }

    if (address[0] != NULL &&
        address[1] == NULL &&
        strcmp(address[0], PTNODE_ENGINE) == 0) /* engine parameters requested */
    {
        //jack_info("reading engine params container");

        leaf = true;
        if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_BOOLEAN, &leaf))
        {
            goto oom_unref;
        }

        if (!jack_controller_fill_parameter_names(
                call,
                &iter,
                ENGINE_DRIVER_PARAMETER_NAME,
                jackctl_server_get_parameters(controller_ptr->server)))
        {
            goto oom_unref;
        }

        return;
    }

    if (address[0] != NULL &&
        address[1] == NULL &&
        strcmp(address[0], PTNODE_DRIVER) == 0) /* current driver parameters requested */
    {
        //jack_info("reading current driver params container");

        leaf = true;
        if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_BOOLEAN, &leaf))
        {
            goto oom_unref;
        }

        if (!jack_controller_fill_parameter_names(
                call,
                &iter,
                NULL,
                jackctl_driver_get_parameters(controller_ptr->driver)))
        {
            goto oom_unref;
        }

        return;
    }

    if (address[0] != NULL &&
        strcmp(address[0], PTNODE_DRIVERS) == 0)
    {
        leaf = address[1] != NULL;
        if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_BOOLEAN, &leaf))
        {
            goto oom_unref;
        }

        if (!leaf)              /* available drivers requested */
        {
            //jack_info("reading drivers container");

            if (!dbus_message_iter_open_container(&iter, DBUS_TYPE_ARRAY, "s", &array_iter))
            {
                goto oom_unref;
            }

            for (index = 0; index < controller_ptr->drivers_count; index++)
            {
                if (!dbus_message_iter_append_basic(&array_iter, DBUS_TYPE_STRING, controller_ptr->driver_names + index))
                {
                    goto oom_close_unref;
                }
            }

            dbus_message_iter_close_container(&iter, &array_iter);
        }
        else                    /* specified driver parameters requested */
        {
            //jack_info("reading driver '%s' params container", address[1]);

            driver = jack_controller_find_driver(controller_ptr->server, address[1]);
            if (driver == NULL)
            {
                jack_dbus_error(
                    call,
                    JACK_DBUS_ERROR_UNKNOWN_INTERNAL,
                    "Unknown driver '%s'",
                    address[1]);
                return;
            }

            if (!jack_controller_fill_parameter_names(
                    call,
                    &iter,
                    NULL,
                    jackctl_driver_get_parameters(driver)))
            {
                goto oom_unref;
            }
        }

        return;
    }

    if (address[0] != NULL &&
        strcmp(address[0], PTNODE_INTERNALS) == 0)
    {
        leaf = address[1] != NULL;
        if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_BOOLEAN, &leaf))
        {
            goto oom_unref;
        }

        if (!leaf)              /* available internals requested */
        {
            //jack_info("reading internals container");

            if (!dbus_message_iter_open_container(&iter, DBUS_TYPE_ARRAY, "s", &array_iter))
            {
                goto oom_unref;
            }

            for (index = 0; index < controller_ptr->internals_count; index++)
            {
                if (!dbus_message_iter_append_basic(&array_iter, DBUS_TYPE_STRING, controller_ptr->internal_names + index))
                {
                    goto oom_close_unref;
                }
            }

            dbus_message_iter_close_container(&iter, &array_iter);
        }
        else                    /* specified driver parameters requested */
        {
            //jack_info("reading internal '%s' params container", address[1]);

            internal = jack_controller_find_internal(controller_ptr->server, address[1]);
            if (internal == NULL)
            {
                jack_dbus_error(
                    call,
                    JACK_DBUS_ERROR_UNKNOWN_INTERNAL,
                    "Unknown internal '%s'",
                    address[1]);
                return;
            }

            if (!jack_controller_fill_parameter_names(
                    call,
                    &iter,
                    NULL,
                    jackctl_internal_get_parameters(internal)))
            {
                goto oom_unref;
            }
        }

        return;
    }

    jack_dbus_error(
        call,
        JACK_DBUS_ERROR_INVALID_ARGS,
        "Invalid container address '%s':'%s':'%s' supplied to method '%s'.",
        address[0],
        address[1],
        address[2],
        call->method_name);

    return;

oom_close_unref:
    dbus_message_iter_close_container(&iter, &array_iter);

oom_unref:
    dbus_message_unref(call->reply);
    call->reply = NULL;

oom:
    jack_error ("Ran out of memory trying to construct method return");
}

static
void
jack_controller_get_parameters_info(
    struct jack_dbus_method_call * call,
    struct parameter_info * special_parameter_info_ptr,
    const JSList * parameters_list)
{
    DBusMessageIter iter, array_iter, struct_iter;
    unsigned char type;
    const char *str;

    call->reply = dbus_message_new_method_return (call->message);
    if (!call->reply)
    {
        goto fail;
    }

    dbus_message_iter_init_append (call->reply, &iter);

    /* Open the array. */
    if (!dbus_message_iter_open_container (&iter, DBUS_TYPE_ARRAY, "(ysss)", &array_iter))
    {
        goto fail_unref;
    }

    if (special_parameter_info_ptr != NULL)
    {
        /* Open the struct. */
        if (!dbus_message_iter_open_container (&array_iter, DBUS_TYPE_STRUCT, NULL, &struct_iter))
        {
            goto fail_close_unref;
        }

        /* Append parameter type. */
        type = PARAM_TYPE_JACK_TO_DBUS(special_parameter_info_ptr->type);
        if (!dbus_message_iter_append_basic (&struct_iter, DBUS_TYPE_BYTE, &type))
        {
            goto fail_close2_unref;
        }

        /* Append parameter name. */
        if (!dbus_message_iter_append_basic (&struct_iter, DBUS_TYPE_STRING, &special_parameter_info_ptr->name))
        {
            goto fail_close2_unref;
        }

        /* Append parameter short description. */
        if (!dbus_message_iter_append_basic (&struct_iter, DBUS_TYPE_STRING, &special_parameter_info_ptr->short_decr))
        {
            goto fail_close2_unref;
        }

        /* Append parameter long description. */
        if (!dbus_message_iter_append_basic (&struct_iter, DBUS_TYPE_STRING, &special_parameter_info_ptr->long_descr))
        {
            goto fail_close2_unref;
        }

        /* Close the struct. */
        if (!dbus_message_iter_close_container (&array_iter, &struct_iter))
        {
            goto fail_close_unref;
        }
    }

    /* Append parameter descriptions to the array. */
    while (parameters_list != NULL)
    {
        /* Open the struct. */
        if (!dbus_message_iter_open_container (&array_iter, DBUS_TYPE_STRUCT, NULL, &struct_iter))
        {
            goto fail_close_unref;
        }

        /* Append parameter type. */
        type = PARAM_TYPE_JACK_TO_DBUS(jackctl_parameter_get_type(parameters_list->data));
        if (!dbus_message_iter_append_basic (&struct_iter, DBUS_TYPE_BYTE, &type))
        {
            goto fail_close2_unref;
        }

        /* Append parameter name. */
        str = jackctl_parameter_get_name(parameters_list->data);
        if (!dbus_message_iter_append_basic (&struct_iter, DBUS_TYPE_STRING, &str))
        {
            goto fail_close2_unref;
        }

        /* Append parameter short description. */
        str = jackctl_parameter_get_short_description(parameters_list->data);
        if (!dbus_message_iter_append_basic (&struct_iter, DBUS_TYPE_STRING, &str))
        {
            goto fail_close2_unref;
        }

        /* Append parameter long description. */
        str = jackctl_parameter_get_long_description(parameters_list->data);
        if (!dbus_message_iter_append_basic (&struct_iter, DBUS_TYPE_STRING, &str))
        {
            goto fail_close2_unref;
        }

        /* Close the struct. */
        if (!dbus_message_iter_close_container (&array_iter, &struct_iter))
        {
            goto fail_close_unref;
        }

        parameters_list = jack_slist_next(parameters_list);
    }

    /* Close the array. */
    if (!dbus_message_iter_close_container (&iter, &array_iter))
    {
        goto fail_unref;
    }

    return;

fail_close2_unref:
    dbus_message_iter_close_container (&iter, &struct_iter);

fail_close_unref:
    dbus_message_iter_close_container (&iter, &array_iter);

fail_unref:
    dbus_message_unref (call->reply);
    call->reply = NULL;

fail:
    jack_error ("Ran out of memory trying to construct method return");
}

static
void
jack_controller_dbus_get_parameters_info(
    struct jack_dbus_method_call * call)
{
    const char * address[3];
    jackctl_internal_t * internal;
    jackctl_driver_t * driver;
    struct parameter_info driver_parameter_info;

    //jack_info("jack_controller_dbus_get_parameters_info() called");

    if (!jack_controller_dbus_get_parameter_address(call, address))
    {
        /* The method call had invalid arguments meaning that
         * jack_controller_dbus_get_parameter_address() has
         * constructed an error for us. */
        return;
    }

    //jack_info("address is '%s':'%s':'%s'", address[0], address[1], address[2]);

    if (address[0] != NULL &&
        address[1] == NULL &&
        strcmp(address[0], PTNODE_ENGINE) == 0) /* engine parameters requested */
    {
        driver_parameter_info.type = ENGINE_DRIVER_PARAMETER_TYPE;
        driver_parameter_info.name = ENGINE_DRIVER_PARAMETER_NAME;
        driver_parameter_info.short_decr = ENGINE_DRIVER_PARAMETER_SHORT_DESCR;
        driver_parameter_info.long_descr = ENGINE_DRIVER_PARAMETER_LONG_DESCR;

        jack_controller_get_parameters_info(
            call,
            &driver_parameter_info,
            jackctl_server_get_parameters(controller_ptr->server));

        return;
    }

    if (address[0] != NULL &&
        address[1] == NULL &&
        strcmp(address[0], PTNODE_DRIVER) == 0) /* current driver parameters requested */
    {
        jack_controller_get_parameters_info(
            call,
            NULL,
            jackctl_driver_get_parameters(controller_ptr->driver));

        return;
    }

    if (address[0] != NULL &&
        address[1] != NULL &&
        address[2] == NULL &&
        strcmp(address[0], PTNODE_DRIVERS) == 0)
    {
        driver = jack_controller_find_driver(controller_ptr->server, address[1]);
        if (driver == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_INTERNAL,
                "Unknown driver '%s'",
                address[1]);
            return;
        }

        jack_controller_get_parameters_info(
            call,
            NULL,
            jackctl_driver_get_parameters(driver));

        return;
    }

    if (address[0] != NULL &&
        address[1] != NULL &&
        address[2] == NULL &&
        strcmp(address[0], PTNODE_INTERNALS) == 0)
    {
        internal = jack_controller_find_internal(controller_ptr->server, address[1]);
        if (internal == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_INTERNAL,
                "Unknown internal '%s'",
                address[1]);
            return;
        }

        jack_controller_get_parameters_info(
            call,
            NULL,
            jackctl_internal_get_parameters(internal));

        return;
    }

    jack_dbus_error(
        call,
        JACK_DBUS_ERROR_INVALID_ARGS,
        "Invalid container address '%s':'%s':'%s' supplied to method '%s'.",
        address[0],
        address[1],
        address[2],
        call->method_name);
}

static
void
jack_controller_get_parameter_info_ex(
    struct jack_dbus_method_call * call,
    struct parameter_info * info_ptr)
{
    DBusMessageIter iter, struct_iter;
    unsigned char type;

    call->reply = dbus_message_new_method_return(call->message);
    if (!call->reply)
    {
        goto fail;
    }

    dbus_message_iter_init_append(call->reply, &iter);

    /* Open the struct. */
    if (!dbus_message_iter_open_container(&iter, DBUS_TYPE_STRUCT, NULL, &struct_iter))
    {
        goto fail_unref;
    }

    /* Append parameter type. */
    type = PARAM_TYPE_JACK_TO_DBUS(info_ptr->type);
    if (!dbus_message_iter_append_basic(&struct_iter, DBUS_TYPE_BYTE, &type))
    {
        goto fail_close_unref;
    }

    /* Append parameter name. */
    if (!dbus_message_iter_append_basic(&struct_iter, DBUS_TYPE_STRING, &info_ptr->name))
    {
        goto fail_close_unref;
    }

    /* Append parameter short description. */
    if (!dbus_message_iter_append_basic(&struct_iter, DBUS_TYPE_STRING, &info_ptr->short_decr))
    {
        goto fail_close_unref;
    }

    /* Append parameter long description. */
    if (!dbus_message_iter_append_basic(&struct_iter, DBUS_TYPE_STRING, &info_ptr->long_descr))
    {
        goto fail_close_unref;
    }

    /* Close the struct. */
    if (!dbus_message_iter_close_container(&iter, &struct_iter))
    {
        goto fail_unref;
    }

    return;

fail_close_unref:
    dbus_message_iter_close_container(&iter, &struct_iter);

fail_unref:
    dbus_message_unref(call->reply);
    call->reply = NULL;

fail:
    jack_error("Ran out of memory trying to construct method return");
}

static
void
jack_controller_get_parameter_info(
    struct jack_dbus_method_call * call,
    jackctl_parameter_t * parameter)
{
    struct parameter_info info;

    info.type = jackctl_parameter_get_type(parameter);
    info.name = jackctl_parameter_get_name(parameter);
    info.short_decr = jackctl_parameter_get_short_description(parameter);
    info.long_descr = jackctl_parameter_get_long_description(parameter);

    jack_controller_get_parameter_info_ex(call, &info);
}

static
void
jack_controller_dbus_get_parameter_info(
    struct jack_dbus_method_call * call)
{
    const char * address[3];
    jackctl_internal_t * internal;
    jackctl_driver_t * driver;
    jackctl_parameter_t * parameter;
    struct parameter_info driver_parameter_info;

    //jack_info("jack_controller_dbus_get_parameter_info() called");

    if (!jack_controller_dbus_get_parameter_address(call, address))
    {
        /* The method call had invalid arguments meaning that
         * jack_controller_dbus_get_parameter_address() has
         * constructed an error for us. */
        return;
    }

    //jack_info("address is '%s':'%s':'%s'", address[0], address[1], address[2]);

    if (address[0] != NULL &&
        address[1] != NULL &&
        address[2] == NULL &&
        strcmp(address[0], PTNODE_ENGINE) == 0) /* engine parameters requested */
    {
        if (strcmp(address[1], ENGINE_DRIVER_PARAMETER_NAME) == 0)
        {
            driver_parameter_info.type = ENGINE_DRIVER_PARAMETER_TYPE;
            driver_parameter_info.name = ENGINE_DRIVER_PARAMETER_NAME;
            driver_parameter_info.short_decr = ENGINE_DRIVER_PARAMETER_SHORT_DESCR;
            driver_parameter_info.long_descr = ENGINE_DRIVER_PARAMETER_LONG_DESCR;

            jack_controller_get_parameter_info_ex(call, &driver_parameter_info);

            return;
        }

        parameter = jack_controller_find_parameter(jackctl_server_get_parameters(controller_ptr->server), address[1]);
        if (parameter == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_PARAMETER,
                "Unknown engine parameter '%s'",
                address[1]);
            return;
        }

        jack_controller_get_parameter_info(call, parameter);

        return;
    }

    if (address[0] != NULL &&
        address[1] != NULL &&
        address[2] == NULL &&
        strcmp(address[0], PTNODE_DRIVER) == 0) /* current driver parameters requested */
    {
        parameter = jack_controller_find_parameter(jackctl_driver_get_parameters(controller_ptr->driver), address[1]);
        if (parameter == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_PARAMETER,
                "Unknown parameter '%s' for driver '%s'",
                address[1],
                jackctl_driver_get_name(controller_ptr->driver));
            return;
        }

        jack_controller_get_parameter_info(call, parameter);

        return;
    }

    if (address[0] != NULL &&
        address[1] != NULL &&
        address[2] != NULL &&
        strcmp(address[0], PTNODE_DRIVERS) == 0)
    {
        driver = jack_controller_find_driver(controller_ptr->server, address[1]);
        if (driver == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_INTERNAL,
                "Unknown driver '%s'",
                address[1]);
            return;
        }

        parameter = jack_controller_find_parameter(jackctl_driver_get_parameters(driver), address[2]);
        if (parameter == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_PARAMETER,
                "Unknown parameter '%s' for driver '%s'",
                address[2],
                address[1]);
            return;
        }

        jack_controller_get_parameter_info(call, parameter);

        return;
    }

    if (address[0] != NULL &&
        address[1] != NULL &&
        address[2] != NULL &&
        strcmp(address[0], PTNODE_INTERNALS) == 0)
    {
        internal = jack_controller_find_internal(controller_ptr->server, address[1]);
        if (internal == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_INTERNAL,
                "Unknown internal '%s'",
                address[1]);
            return;
        }

        parameter = jack_controller_find_parameter(jackctl_internal_get_parameters(internal), address[2]);
        if (parameter == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_PARAMETER,
                "Unknown parameter '%s' for internal '%s'",
                address[2],
                address[1]);
            return;
        }

        jack_controller_get_parameter_info(call, parameter);

        return;
    }

    jack_dbus_error(
        call,
        JACK_DBUS_ERROR_INVALID_ARGS,
        "Invalid container address '%s':'%s':'%s' supplied to method '%s'.",
        address[0],
        address[1],
        address[2],
        call->method_name);
}

static
void
jack_controller_get_parameter_constraint(
    struct jack_dbus_method_call * call,
    jackctl_parameter_t * parameter)
{
    uint32_t index;
    uint32_t count;
      union jackctl_parameter_value min;
      union jackctl_parameter_value max;
    union jackctl_parameter_value jackctl_value;
    DBusMessageIter iter, array_iter, struct_iter;
    const char * descr;
    jackctl_param_type_t type;
    message_arg_t value;
    bool is_range;

    type = jackctl_parameter_get_type(parameter);

    call->reply = dbus_message_new_method_return(call->message);
    if (!call->reply)
    {
        goto fail;
    }

    dbus_message_iter_init_append(call->reply, &iter);

    is_range = jackctl_parameter_has_range_constraint(parameter);
    value.boolean = is_range;
    if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_BOOLEAN, &value))
    {
        goto fail_unref;
    }

    value.boolean = jackctl_parameter_constraint_is_strict(parameter);
    if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_BOOLEAN, &value))
    {
        goto fail_unref;
    }

    value.boolean = jackctl_parameter_constraint_is_fake_value(parameter);
    if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_BOOLEAN, &value))
    {
        goto fail_unref;
    }

    /* Open the array. */
    if (!dbus_message_iter_open_container(&iter, DBUS_TYPE_ARRAY, "(vs)", &array_iter))
    {
        goto fail_unref;
    }

    if (is_range)
    {
        jackctl_parameter_get_range_constraint(parameter, &min, &max);

        /* Open the struct. */
        if (!dbus_message_iter_open_container(&array_iter, DBUS_TYPE_STRUCT, NULL, &struct_iter))
        {
            goto fail_close_unref;
        }

        jack_controller_jack_to_dbus_variant(type, &min, &value);

        if (!jack_dbus_message_append_variant(&struct_iter, PARAM_TYPE_JACK_TO_DBUS(type), PARAM_TYPE_JACK_TO_DBUS_SIGNATURE(type), &value))
        {
            goto fail_close2_unref;
        }

        descr = "min";

        if (!dbus_message_iter_append_basic(&struct_iter, DBUS_TYPE_STRING, &descr))
        {
            goto fail_close2_unref;
        }

        /* Close the struct. */
        if (!dbus_message_iter_close_container(&array_iter, &struct_iter))
        {
            goto fail_close_unref;
        }

        /* Open the struct. */
        if (!dbus_message_iter_open_container(&array_iter, DBUS_TYPE_STRUCT, NULL, &struct_iter))
        {
            goto fail_close_unref;
        }

        jack_controller_jack_to_dbus_variant(type, &max, &value);
 
        if (!jack_dbus_message_append_variant(&struct_iter, PARAM_TYPE_JACK_TO_DBUS(type), PARAM_TYPE_JACK_TO_DBUS_SIGNATURE(type), &value))
        {
            goto fail_close2_unref;
        }

        descr = "max";

        if (!dbus_message_iter_append_basic(&struct_iter, DBUS_TYPE_STRING, &descr))
        {
            goto fail_close2_unref;
        }

        /* Close the struct. */
        if (!dbus_message_iter_close_container(&array_iter, &struct_iter))
        {
            goto fail_close_unref;
        }
    }
    else
    {
        count = jackctl_parameter_get_enum_constraints_count(parameter);

        /* Append enum values to the array. */
        for (index = 0 ; index < count ; index++)
        {
            jackctl_value = jackctl_parameter_get_enum_constraint_value(parameter, index);
            descr = jackctl_parameter_get_enum_constraint_description(parameter, index);

            jack_controller_jack_to_dbus_variant(type, &jackctl_value, &value);

            /* Open the struct. */
            if (!dbus_message_iter_open_container(&array_iter, DBUS_TYPE_STRUCT, NULL, &struct_iter))
            {
                goto fail_close_unref;
            }

            if (!jack_dbus_message_append_variant(&struct_iter, PARAM_TYPE_JACK_TO_DBUS(type), PARAM_TYPE_JACK_TO_DBUS_SIGNATURE(type), &value))
            {
                goto fail_close2_unref;
            }

            if (!dbus_message_iter_append_basic(&struct_iter, DBUS_TYPE_STRING, &descr))
            {
                goto fail_close2_unref;
            }

            /* Close the struct. */
            if (!dbus_message_iter_close_container(&array_iter, &struct_iter))
            {
                goto fail_close_unref;
            }
        }
    }

    /* Close the array. */
    if (!dbus_message_iter_close_container(&iter, &array_iter))
    {
        goto fail_unref;
    }

    return;

fail_close2_unref:
    dbus_message_iter_close_container(&array_iter, &struct_iter);

fail_close_unref:
    dbus_message_iter_close_container(&iter, &array_iter);

fail_unref:
    dbus_message_unref(call->reply);
    call->reply = NULL;

fail:
    jack_error ("Ran out of memory trying to construct method return");
}

static
void
jack_controller_get_parameter_constraint_engine_driver(
    struct jack_dbus_method_call * call)
{
    unsigned int index;
    DBusMessageIter iter, array_iter, struct_iter;
    jackctl_param_type_t type;
    dbus_bool_t bval;
    message_arg_t value;

    type = ENGINE_DRIVER_PARAMETER_TYPE;

    call->reply = dbus_message_new_method_return(call->message);
    if (!call->reply)
    {
        goto fail;
    }

    dbus_message_iter_init_append(call->reply, &iter);

    /* is_range */
    bval = false;
    if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_BOOLEAN, &bval))
    {
        goto fail_unref;
    }

    /* is_strict */
    bval = true;
    if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_BOOLEAN, &bval))
    {
        goto fail_unref;
    }

    /* is_fake_value */
    bval = true;
    if (!dbus_message_iter_append_basic(&iter, DBUS_TYPE_BOOLEAN, &bval))
    {
        goto fail_unref;
    }

    /* Open the array. */
    if (!dbus_message_iter_open_container(&iter, DBUS_TYPE_ARRAY, "(vs)", &array_iter))
    {
        goto fail_unref;
    }

    /* Append enum values to the array. */
    for (index = 0 ; index < controller_ptr->drivers_count ; index++)
    {
        /* Open the struct. */
        if (!dbus_message_iter_open_container(&array_iter, DBUS_TYPE_STRUCT, NULL, &struct_iter))
        {
            goto fail_close_unref;
        }

        value.string = controller_ptr->driver_names[index];
        if (!jack_dbus_message_append_variant(
                &struct_iter,
                PARAM_TYPE_JACK_TO_DBUS(type),
                PARAM_TYPE_JACK_TO_DBUS_SIGNATURE(type),
                &value))
        {
            goto fail_close2_unref;
        }

        if (!dbus_message_iter_append_basic(&struct_iter, DBUS_TYPE_STRING, &value))
        {
            goto fail_close2_unref;
        }

        /* Close the struct. */
        if (!dbus_message_iter_close_container(&array_iter, &struct_iter))
        {
            goto fail_close_unref;
        }
    }

    /* Close the array. */
    if (!dbus_message_iter_close_container(&iter, &array_iter))
    {
        goto fail_unref;
    }

    return;

fail_close2_unref:
    dbus_message_iter_close_container(&array_iter, &struct_iter);

fail_close_unref:
    dbus_message_iter_close_container(&iter, &array_iter);

fail_unref:
    dbus_message_unref(call->reply);
    call->reply = NULL;

fail:
    jack_error ("Ran out of memory trying to construct method return");
}

static
void
jack_controller_dbus_get_parameter_constraint(
    struct jack_dbus_method_call * call)
{
    const char * address[3];
    jackctl_internal_t * internal;
    jackctl_driver_t * driver;
    jackctl_parameter_t * parameter;

    //jack_info("jack_controller_dbus_get_parameter_constraint() called");

    if (!jack_controller_dbus_get_parameter_address(call, address))
    {
        /* The method call had invalid arguments meaning that
         * jack_controller_dbus_get_parameter_address() has
         * constructed an error for us. */
        return;
    }

    //jack_info("address is '%s':'%s':'%s'", address[0], address[1], address[2]);

    if (address[0] != NULL &&
        address[1] != NULL &&
        address[2] == NULL &&
        strcmp(address[0], PTNODE_ENGINE) == 0) /* engine parameters requested */
    {
        if (strcmp(address[1], ENGINE_DRIVER_PARAMETER_NAME) == 0)
        {
            jack_controller_get_parameter_constraint_engine_driver(call);

            return;
        }

        parameter = jack_controller_find_parameter(jackctl_server_get_parameters(controller_ptr->server), address[1]);
        if (parameter == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_PARAMETER,
                "Unknown engine parameter '%s'",
                address[1]);
            return;
        }

        jack_controller_get_parameter_constraint(call, parameter);

        return;
    }

    if (address[0] != NULL &&
        address[1] != NULL &&
        address[2] == NULL &&
        strcmp(address[0], PTNODE_DRIVER) == 0) /* current driver parameters requested */
    {
        parameter = jack_controller_find_parameter(jackctl_driver_get_parameters(controller_ptr->driver), address[1]);
        if (parameter == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_PARAMETER,
                "Unknown parameter '%s' for driver '%s'",
                address[1],
                jackctl_driver_get_name(controller_ptr->driver));
            return;
        }

        jack_controller_get_parameter_constraint(call, parameter);

        return;
    }

    if (address[0] != NULL &&
        address[1] != NULL &&
        address[2] != NULL &&
        strcmp(address[0], PTNODE_DRIVERS) == 0)
    {
        driver = jack_controller_find_driver(controller_ptr->server, address[1]);
        if (driver == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_INTERNAL,
                "Unknown driver '%s'",
                address[1]);
            return;
        }

        parameter = jack_controller_find_parameter(jackctl_driver_get_parameters(driver), address[2]);
        if (parameter == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_PARAMETER,
                "Unknown parameter '%s' for driver '%s'",
                address[2],
                address[1]);
            return;
        }

        jack_controller_get_parameter_constraint(call, parameter);

        return;
    }

    if (address[0] != NULL &&
        address[1] != NULL &&
        address[2] != NULL &&
        strcmp(address[0], PTNODE_INTERNALS) == 0)
    {
        internal = jack_controller_find_internal(controller_ptr->server, address[1]);
        if (internal == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_INTERNAL,
                "Unknown internal '%s'",
                address[1]);
            return;
        }

        parameter = jack_controller_find_parameter(jackctl_internal_get_parameters(internal), address[2]);
        if (parameter == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_PARAMETER,
                "Unknown parameter '%s' for internal '%s'",
                address[2],
                address[1]);
            return;
        }

        jack_controller_get_parameter_constraint(call, parameter);

        return;
    }

    jack_dbus_error(
        call,
        JACK_DBUS_ERROR_INVALID_ARGS,
        "Invalid container address '%s':'%s':'%s' supplied to method '%s'.",
        address[0],
        address[1],
        address[2],
        call->method_name);
}

static
void
jack_controller_get_parameter_value(
    struct jack_dbus_method_call * call,
    jackctl_parameter_t * parameter)
{
    int type;
    union jackctl_parameter_value jackctl_value;
    union jackctl_parameter_value jackctl_default_value;
    message_arg_t value;
    message_arg_t default_value;

    type = jackctl_parameter_get_type(parameter);
    jackctl_default_value = jackctl_parameter_get_default_value(parameter);
    jackctl_value = jackctl_parameter_get_value(parameter);

    jack_controller_jack_to_dbus_variant(type, &jackctl_value, &value);
    jack_controller_jack_to_dbus_variant(type, &jackctl_default_value, &default_value);

    /* Construct the reply. */
    jack_dbus_construct_method_return_parameter(
        call,
        (dbus_bool_t)(jackctl_parameter_is_set(parameter) ? TRUE : FALSE),
        PARAM_TYPE_JACK_TO_DBUS(type),
        PARAM_TYPE_JACK_TO_DBUS_SIGNATURE(type),
        default_value,
        value);
}

static
void
jack_controller_get_parameter_value_engine_driver(
    struct jack_dbus_method_call * call)
{
    message_arg_t value;
    message_arg_t default_value;

    default_value.string = DEFAULT_DRIVER;
    value.string = jackctl_driver_get_name(controller_ptr->driver);

    /* Construct the reply. */
    jack_dbus_construct_method_return_parameter(
        call,
        controller_ptr->driver_set,
        DBUS_TYPE_STRING,
        DBUS_TYPE_STRING_AS_STRING,
        default_value,
        value);
}


static void
jack_controller_dbus_get_parameter_value(
    struct jack_dbus_method_call * call)
{
    const char * address[3];
    jackctl_internal_t * internal;
    jackctl_driver_t * driver;
    jackctl_parameter_t * parameter;

    //jack_info("jack_controller_dbus_get_parameter_value() called");

    if (!jack_controller_dbus_get_parameter_address(call, address))
    {
        /* The method call had invalid arguments meaning that
         * jack_controller_dbus_get_parameter_address() has
         * constructed an error for us. */
        return;
    }

    //jack_info("address is '%s':'%s':'%s'", address[0], address[1], address[2]);

    if (address[0] != NULL &&
        address[1] != NULL &&
        address[2] == NULL &&
        strcmp(address[0], PTNODE_ENGINE) == 0) /* engine parameters requested */
    {
        if (strcmp(address[1], ENGINE_DRIVER_PARAMETER_NAME) == 0)
        {
            jack_controller_get_parameter_value_engine_driver(call);

            return;
        }

        parameter = jack_controller_find_parameter(jackctl_server_get_parameters(controller_ptr->server), address[1]);
        if (parameter == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_PARAMETER,
                "Unknown engine parameter '%s'",
                address[1]);
            return;
        }

        jack_controller_get_parameter_value(call, parameter);

        return;
    }

    if (address[0] != NULL &&
        address[1] != NULL &&
        address[2] == NULL &&
        strcmp(address[0], PTNODE_DRIVER) == 0) /* current driver parameters requested */
    {
        parameter = jack_controller_find_parameter(jackctl_driver_get_parameters(controller_ptr->driver), address[1]);
        if (parameter == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_PARAMETER,
                "Unknown parameter '%s' for driver '%s'",
                address[1],
                jackctl_driver_get_name(controller_ptr->driver));
            return;
        }

        jack_controller_get_parameter_value(call, parameter);

        return;
    }

    if (address[0] != NULL &&
        address[1] != NULL &&
        address[2] != NULL &&
        strcmp(address[0], PTNODE_DRIVERS) == 0)
    {
        driver = jack_controller_find_driver(controller_ptr->server, address[1]);
        if (driver == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_INTERNAL,
                "Unknown driver '%s'",
                address[1]);
            return;
        }

        parameter = jack_controller_find_parameter(jackctl_driver_get_parameters(driver), address[2]);
        if (parameter == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_PARAMETER,
                "Unknown parameter '%s' for driver '%s'",
                address[2],
                address[1]);
            return;
        }

        jack_controller_get_parameter_value(call, parameter);

        return;
    }

    if (address[0] != NULL &&
        address[1] != NULL &&
        address[2] != NULL &&
        strcmp(address[0], PTNODE_INTERNALS) == 0)
    {
        internal = jack_controller_find_internal(controller_ptr->server, address[1]);
        if (internal == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_INTERNAL,
                "Unknown internal '%s'",
                address[1]);
            return;
        }

        parameter = jack_controller_find_parameter(jackctl_internal_get_parameters(internal), address[2]);
        if (parameter == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_PARAMETER,
                "Unknown parameter '%s' for internal '%s'",
                address[2],
                address[1]);
            return;
        }

        jack_controller_get_parameter_value(call, parameter);

        return;
    }

    jack_dbus_error(
        call,
        JACK_DBUS_ERROR_INVALID_ARGS,
        "Invalid container address '%s':'%s':'%s' supplied to method '%s'.",
        address[0],
        address[1],
        address[2],
        call->method_name);
}

static
void
jack_controller_set_parameter_value(
    struct jack_dbus_method_call * call,
    jackctl_parameter_t * parameter,
    message_arg_t * arg_ptr,
    int arg_type)
{
    jackctl_param_type_t type;
    union jackctl_parameter_value value;

    type = jackctl_parameter_get_type(parameter);

    if (PARAM_TYPE_JACK_TO_DBUS(type) != arg_type)
    {
        jack_dbus_error(
            call,
            JACK_DBUS_ERROR_INVALID_ARGS,
            "Parameter value type mismatch: was expecting '%c', got '%c'",
            (char)PARAM_TYPE_JACK_TO_DBUS(type),
            (char)arg_type);
        return;
    }

    if (!jack_controller_dbus_to_jack_variant(
            arg_type,
            arg_ptr,
            &value))
    {
        jack_dbus_error(
            call,
            JACK_DBUS_ERROR_INVALID_ARGS,
            "Cannot convert parameter value");
        return;
    }

    jackctl_parameter_set_value(parameter, &value);

    jack_controller_settings_save_auto(controller_ptr);

    jack_dbus_construct_method_return_empty(call);
}

static
void
jack_controller_set_parameter_value_engine_driver(
    struct jack_dbus_method_call * call,
    message_arg_t * arg_ptr,
    int arg_type)
{
    union jackctl_parameter_value value;

    if (arg_type != DBUS_TYPE_STRING)
    {
        jack_dbus_error(
            call,
            JACK_DBUS_ERROR_INVALID_ARGS,
            "Engine parameter value type mismatch: was expecting '%c', got '%c'",
            (char)DBUS_TYPE_STRING,
            (char)arg_type);
        return;
    }

    if (!jack_controller_dbus_to_jack_variant(
            arg_type,
            arg_ptr,
            &value))
    {
        jack_dbus_error(
            call,
            JACK_DBUS_ERROR_INVALID_ARGS,
            "Cannot convert engine parameter value");
        return;
    }

    if (!jack_controller_select_driver(controller_ptr, value.str))
    {
        /* Couldn't find driver with the specified name. */
        jack_dbus_error(
            call,
            JACK_DBUS_ERROR_UNKNOWN_DRIVER,
            "Unknown driver '%s'",
            value.str);
        return;
    }

    jack_controller_settings_save_auto(controller_ptr);

    jack_dbus_construct_method_return_empty(call);
}

static
void
jack_controller_dbus_set_parameter_value(
    struct jack_dbus_method_call * call)
{
    const char * address[3];
    DBusMessageIter iter;
    DBusMessageIter variant_iter;
    message_arg_t arg;
    int arg_type;
    jackctl_internal_t * internal;
    jackctl_driver_t * driver;
    jackctl_parameter_t * parameter;

    //jack_info("jack_controller_dbus_set_parameter_value() called");

    if (!jack_controller_dbus_get_parameter_address_ex(call, &iter, address))
    {
        /* The method call had invalid arguments meaning that
         * jack_controller_dbus_get_parameter_address() has
         * constructed an error for us. */
        return;
    }

    //jack_info("address is '%s':'%s':'%s'", address[0], address[1], address[2]);

    dbus_message_iter_next(&iter);

    if (dbus_message_iter_has_next(&iter))
    {
        jack_dbus_error(
            call,
            JACK_DBUS_ERROR_INVALID_ARGS,
            "Invalid arguments to method '%s'. Too many arguments.",
            call->method_name);
        return;
    }

    if (dbus_message_iter_get_arg_type(&iter) != DBUS_TYPE_VARIANT)
    {
        jack_dbus_error(
            call,
            JACK_DBUS_ERROR_INVALID_ARGS,
            "Invalid arguments to method '%s'. Value to set must be variant.",
            call->method_name);
        return;
    }

    dbus_message_iter_recurse (&iter, &variant_iter);
    dbus_message_iter_get_basic(&variant_iter, &arg);
    arg_type = dbus_message_iter_get_arg_type(&variant_iter);

    //jack_info("argument of type '%c'", arg_type);

    if (address[0] != NULL &&
        address[1] != NULL &&
        address[2] == NULL &&
        strcmp(address[0], PTNODE_ENGINE) == 0) /* engine parameters requested */
    {
        if (strcmp(address[1], ENGINE_DRIVER_PARAMETER_NAME) == 0)
        {
            jack_controller_set_parameter_value_engine_driver(call, &arg, arg_type);

            return;
        }

        parameter = jack_controller_find_parameter(jackctl_server_get_parameters(controller_ptr->server), address[1]);
        if (parameter == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_PARAMETER,
                "Unknown engine parameter '%s'",
                address[1]);
            return;
        }

        jack_controller_set_parameter_value(call, parameter, &arg, arg_type);

        return;
    }

    if (address[0] != NULL &&
        address[1] != NULL &&
        address[2] == NULL &&
        strcmp(address[0], PTNODE_DRIVER) == 0) /* current driver parameters requested */
    {
        parameter = jack_controller_find_parameter(jackctl_driver_get_parameters(controller_ptr->driver), address[1]);
        if (parameter == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_PARAMETER,
                "Unknown parameter '%s' for driver '%s'",
                address[1],
                jackctl_driver_get_name(controller_ptr->driver));
            return;
        }

        jack_controller_set_parameter_value(call, parameter, &arg, arg_type);

        return;
    }

    if (address[0] != NULL &&
        address[1] != NULL &&
        address[2] != NULL &&
        strcmp(address[0], PTNODE_DRIVERS) == 0)
    {
        driver = jack_controller_find_driver(controller_ptr->server, address[1]);
        if (driver == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_INTERNAL,
                "Unknown driver '%s'",
                address[1]);
            return;
        }

        parameter = jack_controller_find_parameter(jackctl_driver_get_parameters(driver), address[2]);
        if (parameter == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_PARAMETER,
                "Unknown parameter '%s' for driver '%s'",
                address[2],
                address[1]);
            return;
        }

        jack_controller_set_parameter_value(call, parameter, &arg, arg_type);

        return;
    }

    if (address[0] != NULL &&
        address[1] != NULL &&
        address[2] != NULL &&
        strcmp(address[0], PTNODE_INTERNALS) == 0)
    {
        internal = jack_controller_find_internal(controller_ptr->server, address[1]);
        if (internal == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_INTERNAL,
                "Unknown internal '%s'",
                address[1]);
            return;
        }

        parameter = jack_controller_find_parameter(jackctl_internal_get_parameters(internal), address[2]);
        if (parameter == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_PARAMETER,
                "Unknown parameter '%s' for internal '%s'",
                address[2],
                address[1]);
            return;
        }

        jack_controller_set_parameter_value(call, parameter, &arg, arg_type);

        return;
    }

    jack_dbus_error(
        call,
        JACK_DBUS_ERROR_INVALID_ARGS,
        "Invalid container address '%s':'%s':'%s' supplied to method '%s'.",
        address[0],
        address[1],
        address[2],
        call->method_name);
}

static
void
jack_controller_reset_parameter_value(
    struct jack_dbus_method_call * call,
    jackctl_parameter_t * parameter)
{
    jackctl_parameter_reset(parameter);

    jack_controller_settings_save_auto(controller_ptr);

    jack_dbus_construct_method_return_empty(call);
}

static
void
jack_controller_reset_parameter_value_engine_driver(
    struct jack_dbus_method_call * call)
{
    if (!jack_controller_select_driver(controller_ptr, DEFAULT_DRIVER))
    {
        /* Couldn't find driver with the specified name. */
        jack_dbus_error(
            call,
            JACK_DBUS_ERROR_UNKNOWN_DRIVER,
            "Default driver '%s' is unknown",
            DEFAULT_DRIVER);
        return;
    }

    controller_ptr->driver_set = false;

    jack_controller_settings_save_auto(controller_ptr);

    jack_dbus_construct_method_return_empty(call);
}

static
void
jack_controller_dbus_reset_parameter_value(
    struct jack_dbus_method_call * call)
{
    const char * address[3];
    jackctl_internal_t * internal;
    jackctl_driver_t * driver;
    jackctl_parameter_t * parameter;

    //jack_info("jack_controller_dbus_reset_parameter_value() called");

    if (!jack_controller_dbus_get_parameter_address(call, address))
    {
        /* The method call had invalid arguments meaning that
         * jack_controller_dbus_get_parameter_address() has
         * constructed an error for us. */
        return;
    }

    //jack_info("address is '%s':'%s':'%s'", address[0], address[1], address[2]);

    if (address[0] != NULL &&
        address[1] != NULL &&
        address[2] == NULL &&
        strcmp(address[0], PTNODE_ENGINE) == 0) /* engine parameters requested */
    {
        if (strcmp(address[1], ENGINE_DRIVER_PARAMETER_NAME) == 0)
        {
            jack_controller_reset_parameter_value_engine_driver(call);

            return;
        }

        parameter = jack_controller_find_parameter(jackctl_server_get_parameters(controller_ptr->server), address[1]);
        if (parameter == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_PARAMETER,
                "Unknown engine parameter '%s'",
                address[1]);
            return;
        }

        jack_controller_reset_parameter_value(call, parameter);

        return;
    }

    if (address[0] != NULL &&
        address[1] != NULL &&
        address[2] == NULL &&
        strcmp(address[0], PTNODE_DRIVER) == 0) /* current driver parameters requested */
    {
        parameter = jack_controller_find_parameter(jackctl_driver_get_parameters(controller_ptr->driver), address[1]);
        if (parameter == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_PARAMETER,
                "Unknown parameter '%s' for driver '%s'",
                address[1],
                jackctl_driver_get_name(controller_ptr->driver));
            return;
        }

        jack_controller_reset_parameter_value(call, parameter);

        return;
    }

    if (address[0] != NULL &&
        address[1] != NULL &&
        address[2] != NULL &&
        strcmp(address[0], PTNODE_DRIVERS) == 0)
    {
        driver = jack_controller_find_driver(controller_ptr->server, address[1]);
        if (driver == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_INTERNAL,
                "Unknown driver '%s'",
                address[1]);
            return;
        }

        parameter = jack_controller_find_parameter(jackctl_driver_get_parameters(driver), address[2]);
        if (parameter == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_PARAMETER,
                "Unknown parameter '%s' for driver '%s'",
                address[2],
                address[1]);
            return;
        }

        jack_controller_reset_parameter_value(call, parameter);

        return;
    }

    if (address[0] != NULL &&
        address[1] != NULL &&
        address[2] != NULL &&
        strcmp(address[0], PTNODE_INTERNALS) == 0)
    {
        internal = jack_controller_find_internal(controller_ptr->server, address[1]);
        if (internal == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_INTERNAL,
                "Unknown internal '%s'",
                address[1]);
            return;
        }

        parameter = jack_controller_find_parameter(jackctl_internal_get_parameters(internal), address[2]);
        if (parameter == NULL)
        {
            jack_dbus_error(
                call,
                JACK_DBUS_ERROR_UNKNOWN_PARAMETER,
                "Unknown parameter '%s' for internal '%s'",
                address[2],
                address[1]);
            return;
        }

        jack_controller_reset_parameter_value(call, parameter);

        return;
    }

    jack_dbus_error(
        call,
        JACK_DBUS_ERROR_INVALID_ARGS,
        "Invalid container address '%s':'%s':'%s' supplied to method '%s'.",
        address[0],
        address[1],
        address[2],
        call->method_name);
}

#undef controller_ptr

JACK_DBUS_METHOD_ARGUMENTS_BEGIN_EX(ReadContainer, "Get names of child parameters or containers")
    JACK_DBUS_METHOD_ARGUMENT_IN("parent", "as", "Address of parent container")
    JACK_DBUS_METHOD_ARGUMENT_OUT("leaf", "b", "Whether children are parameters (true) or containers (false)")
    JACK_DBUS_METHOD_ARGUMENT_OUT("children", "as", "Array of child names")
JACK_DBUS_METHOD_ARGUMENTS_END

JACK_DBUS_METHOD_ARGUMENTS_BEGIN_EX(GetParametersInfo, "Retrieve info about parameters")
    JACK_DBUS_METHOD_ARGUMENT_IN("parent", "as", "Address of parameters parent")
    JACK_DBUS_METHOD_ARGUMENT_OUT("parameter_info_array", "a(ysss)", "Array of parameter info structs. Each info struct contains: type char, name, short and long description")
JACK_DBUS_METHOD_ARGUMENTS_END

JACK_DBUS_METHOD_ARGUMENTS_BEGIN_EX(GetParameterInfo, "Retrieve info about parameter")
    JACK_DBUS_METHOD_ARGUMENT_IN("parameter", "as", "Address of parameter")
    JACK_DBUS_METHOD_ARGUMENT_OUT("parameter_info", "(ysss)", "Parameter info struct that contains: type char, name, short and long description")
JACK_DBUS_METHOD_ARGUMENTS_END

JACK_DBUS_METHOD_ARGUMENTS_BEGIN_EX(GetParameterConstraint, "Get constraint of parameter")
    JACK_DBUS_METHOD_ARGUMENT_IN("parameter", "as", "Address of parameter")
    JACK_DBUS_METHOD_ARGUMENT_OUT("is_range", "b", "Whether constrinat is a range. If so, values parameter will contain two values, min and max")
    JACK_DBUS_METHOD_ARGUMENT_OUT("is_strict", "b", "Whether enum constraint is strict. I.e. value not listed in values array will not work")
    JACK_DBUS_METHOD_ARGUMENT_OUT("is_fake_value", "b", "Whether enum values are fake. I.e. have no user meaningful meaning")
    JACK_DBUS_METHOD_ARGUMENT_OUT("values", "a(vs)", "Values. If there is no constraint, this array will be empty. For range constraint there will be two values, min and max. For enum constraint there will be 2 or more values.")
JACK_DBUS_METHOD_ARGUMENTS_END

JACK_DBUS_METHOD_ARGUMENTS_BEGIN_EX(GetParameterValue, "Get value of parameter")
    JACK_DBUS_METHOD_ARGUMENT_IN("parameter", "as", "Address of parameter")
    JACK_DBUS_METHOD_ARGUMENT_OUT("is_set", "b", "Whether parameter is set or its default value is used")
    JACK_DBUS_METHOD_ARGUMENT_OUT("default", "v", "Default value of parameter")
    JACK_DBUS_METHOD_ARGUMENT_OUT("value", "v", "Actual value of parameter")
JACK_DBUS_METHOD_ARGUMENTS_END

JACK_DBUS_METHOD_ARGUMENTS_BEGIN_EX(SetParameterValue, "Set value of parameter")
    JACK_DBUS_METHOD_ARGUMENT_IN("parameter", "as", "Address of parameter")
    JACK_DBUS_METHOD_ARGUMENT_IN("value", "v", "New value for parameter")
JACK_DBUS_METHOD_ARGUMENTS_END

JACK_DBUS_METHOD_ARGUMENTS_BEGIN_EX(ResetParameterValue, "Reset parameter to default value")
    JACK_DBUS_METHOD_ARGUMENT_IN("parameter", "as", "Address of parameter")
JACK_DBUS_METHOD_ARGUMENTS_END

JACK_DBUS_METHODS_BEGIN
    JACK_DBUS_METHOD_DESCRIBE(ReadContainer, jack_controller_dbus_read_container)
    JACK_DBUS_METHOD_DESCRIBE(GetParametersInfo, jack_controller_dbus_get_parameters_info)
    JACK_DBUS_METHOD_DESCRIBE(GetParameterInfo, jack_controller_dbus_get_parameter_info)
    JACK_DBUS_METHOD_DESCRIBE(GetParameterConstraint, jack_controller_dbus_get_parameter_constraint)
    JACK_DBUS_METHOD_DESCRIBE(GetParameterValue, jack_controller_dbus_get_parameter_value)
    JACK_DBUS_METHOD_DESCRIBE(SetParameterValue, jack_controller_dbus_set_parameter_value)
    JACK_DBUS_METHOD_DESCRIBE(ResetParameterValue, jack_controller_dbus_reset_parameter_value)
JACK_DBUS_METHODS_END

/*
 * Parameter addresses:
 *
 * "engine"
 * "engine", "driver"
 * "engine", "realtime"
 * "engine", ...more engine parameters
 *
 * "driver", "device"
 * "driver", ...more driver parameters
 *
 * "drivers", "alsa", "device"
 * "drivers", "alsa", ...more alsa driver parameters
 *
 * "drivers", ...more drivers
 *
 * "internals", "netmanager", "multicast_ip"
 * "internals", "netmanager", ...more netmanager parameters
 *
 * "internals", ...more internals
 *
 */

JACK_DBUS_IFACE_BEGIN(g_jack_controller_iface_configure, "org.jackaudio.Configure")
    JACK_DBUS_IFACE_EXPOSE_METHODS
JACK_DBUS_IFACE_END

Generated by  Doxygen 1.6.0   Back to index