Subversion Repositories HelenOS

Rev

Rev 2550 | Rev 2594 | Go to most recent revision | Blame | Compare with Previous | Last modification | View Log | Download | RSS feed

/*
 * Copyright (c) 2007 Josef Cejka
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 *
 * - Redistributions of source code must retain the above copyright
 *   notice, this list of conditions and the following disclaimer.
 * - Redistributions in binary form must reproduce the above copyright
 *   notice, this list of conditions and the following disclaimer in the
 *   documentation and/or other materials provided with the distribution.
 * - The name of the author may not be used to endorse or promote products
 *   derived from this software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */

/**
 * @defgroup devmap Device mapper.
 * @brief   HelenOS device mapper.
 * @{
 */ 

/** @file
 */

#include <ipc/services.h>
#include <ipc/ns.h>
#include <async.h>
#include <stdio.h>
#include <errno.h>
#include <bool.h>
#include <libadt/list.h>
#include <futex.h>

#include "devmap.h"

LIST_INITIALIZE(device_list);

atomic_t device_list_futex = FUTEX_INITIALIZER;

/** Initialize device mapper. 
 *
 *
 */
static int devmap_init()
{
    /* */

    return EOK;
}

static int devmap_register()
{
    ipc_callid_t callid2;
    size_t size;
    char buffer[DEVMAP_NAME_MAXLEN + 1];
    ipc_callid_t callid;
    ipc_call_t call;
    int retval;
    
    if (ipc_data_receive(&callid, &call, NULL, &size) != 0) {
    //  retval = 
    }

    if (size > DEVMAP_NAME_MAXLEN) {
        retval = ELIMIT;
    } else {
        ipc_data_deliver(callid2, &call, buffer, size);
        buffer[DEVMAP_NAME_MAXLEN] = 0;
    }
    return EOK;
}

static int devmap_unregister()
{
    return EOK;
}

static int devmap_forward(int handle, ipc_call_t *call, ipc_callid_t callid)
{
    link_t *item;
    ipcarg_t phone;
    devmap_device_t *dev;

    /* FIXME: add futex */
    futex_down(&device_list_futex);

    item = (&device_list)->next;

    while (item != &device_list) {

        dev = list_get_instance(item, devmap_device_t, list);
        if (dev->handle == handle) {
            break;
        }
        item = item->next;
    }

    if (item == &device_list) {
        return ENOENT;
    }

    dev = list_get_instance(item, devmap_device_t, list);
    phone = dev->phone;

    futex_up(&device_list_futex);
    
    return ipc_forward_fast(callid, phone, 0, 0);
}

static int devmap_get_handle()
{
    return EOK;
}

/** Function for handling connections to devmap 
 *
 */
static void
devmap_client_connection(ipc_callid_t iid, ipc_call_t *icall)
{
    ipc_callid_t callid;
    ipc_call_t call;
    int retval;
    bool cont = true;

    printf("DevMap: new connection.");

    ipc_answer_fast(iid, EOK, 0, 0); /* Accept connection */

    while (cont) {
        callid = async_get_call(&call);

        switch (IPC_GET_METHOD(call)) {
        case IPC_M_PHONE_HUNGUP:
                /* TODO: if its a device connection, remove it from table */
            devmap_unregister();
            printf("DevMap: connection hung up.");
            cont = false;
            continue; /* Exit thread */

        case DEVMAP_REGISTER:
            
            if ((retval = devmap_register()) != EOK) {
                cont = false;
            }
            break;
        case DEVMAP_UNREGISTER:
            /* TODO: remove device (if registred) */
            retval = devmap_unregister();
            cont = false;
            break;
        case DEVMAP_CONNECT_TO_DEVICE:
            retval = devmap_forward(IPC_GET_ARG1(call), &call, callid);
            cont = false;
            break;
        case DEVMAP_GET_HANDLE:
            
            break;
        default:
            retval = ENOENT;
        }
        ipc_answer_fast(callid, retval, 0, 0);
    }
        
    printf("DevMap: connection closed.");
    return;
}


int main(int argc, char *argv[])
{

    ipcarg_t phonead;

    printf("DevMap: HelenOS device mapper.\n");

    if (devmap_init() != 0) {
        printf("Error while initializing DevMap service.");
        return -1;
    }

        /* Set a handler of incomming connections */
    async_set_client_connection(devmap_client_connection);

        /* Register device mapper at naming service */
    if (ipc_connect_to_me(PHONE_NS, SERVICE_DEVMAP, 0, &phonead) != 0) 
        return -1;
    
    async_manager();
    /* Never reached */
    return 0;
}

/** 
 * @}
 */