/* * This file is part of the Palacios Virtual Machine Monitor developed * by the V3VEE Project with funding from the United States National * Science Foundation and the Department of Energy. * * The V3VEE Project is a joint project between Northwestern University * and the University of New Mexico. You can find out more at * http://www.v3vee.org * * Copyright (c) 2010, Peter Dinda * Copyright (c) 2010, The V3VEE Project * All rights reserved. * * Author: Peter Dinda * * This is free software. You are permitted to use, * redistribute, and modify it as specified in the file "V3VEE_LICENSE". */ #include #include #include #include #include #include #include struct stream_state { struct v3_stream * stream; struct v3_dev_char_ops char_ops; struct v3_vm_info * vm; void * push_fn_arg; }; static uint64_t stream_input(struct v3_stream * stream, uint8_t * buf, uint64_t len) { struct stream_state * state = stream->guest_stream_data; return state->char_ops.input(state->vm, buf, len, state->push_fn_arg); } static uint64_t stream_output(uint8_t * buf, uint64_t length, void * private_data) { struct stream_state * state = (struct stream_state *)private_data; return v3_stream_output(state->stream, buf, length); } static int stream_free(struct stream_state * state) { v3_stream_close(state->stream); // detach host event V3_Free(state); return 0; } static struct v3_device_ops dev_ops = { .free = (int (*)(void *))stream_free, }; static int stream_init(struct v3_vm_info * vm, v3_cfg_tree_t * cfg) { char * dev_id = v3_cfg_val(cfg, "ID"); char * stream_name = v3_cfg_val(cfg, "name"); struct stream_state * state = NULL; v3_cfg_tree_t * frontend_cfg = v3_cfg_subtree(cfg, "frontend"); state = (struct stream_state *)V3_Malloc(sizeof(struct stream_state)); if (state == NULL) { PrintError("Could not allocate stream backend device\n"); return -1; } memset(state, 0, sizeof(struct stream_state)); struct vm_device * dev = v3_add_device(vm, dev_id, &dev_ops, state); if (dev == NULL) { PrintError("Could not allocate device %s\n", dev_id); V3_Free(state); return -1; } state->stream = v3_stream_open(vm, stream_name, stream_input, state); if (state->stream == NULL) { PrintError("Could not open stream %s\n", stream_name); v3_remove_device(dev); return -1; } state->vm = vm; state->char_ops.output = stream_output; if (v3_dev_connect_char(vm, v3_cfg_val(frontend_cfg, "tag"), &(state->char_ops), frontend_cfg, state, &(state->push_fn_arg)) == -1) { PrintError("Could not connect %s to frontend %s\n", dev_id, v3_cfg_val(frontend_cfg, "tag")); v3_remove_device(dev); return -1; } return 0; } device_register("CHAR_STREAM", stream_init);