|  | /* | 
|  | * QEMU System Emulator | 
|  | * | 
|  | * Copyright (c) 2003-2008 Fabrice Bellard | 
|  | * | 
|  | * Permission is hereby granted, free of charge, to any person obtaining a copy | 
|  | * of this software and associated documentation files (the "Software"), to deal | 
|  | * in the Software without restriction, including without limitation the rights | 
|  | * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell | 
|  | * copies of the Software, and to permit persons to whom the Software is | 
|  | * furnished to do so, subject to the following conditions: | 
|  | * | 
|  | * The above copyright notice and this permission notice shall be included in | 
|  | * all copies or substantial portions of the Software. | 
|  | * | 
|  | * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR | 
|  | * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | 
|  | * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL | 
|  | * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER | 
|  | * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, | 
|  | * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN | 
|  | * THE SOFTWARE. | 
|  | */ | 
|  |  | 
|  | #include "qemu/osdep.h" | 
|  | #include "qemu/main-loop.h" | 
|  | #include "qemu/module.h" | 
|  | #include "qapi/error.h" | 
|  | #include "chardev/char-win.h" | 
|  |  | 
|  | static void win_chr_read(Chardev *chr, DWORD len) | 
|  | { | 
|  | WinChardev *s = WIN_CHARDEV(chr); | 
|  | int max_size = qemu_chr_be_can_write(chr); | 
|  | int ret, err; | 
|  | uint8_t buf[CHR_READ_BUF_LEN]; | 
|  | DWORD size; | 
|  |  | 
|  | if (len > max_size) { | 
|  | len = max_size; | 
|  | } | 
|  | if (len == 0) { | 
|  | return; | 
|  | } | 
|  |  | 
|  | ZeroMemory(&s->orecv, sizeof(s->orecv)); | 
|  | s->orecv.hEvent = s->hrecv; | 
|  | ret = ReadFile(s->file, buf, len, &size, &s->orecv); | 
|  | if (!ret) { | 
|  | err = GetLastError(); | 
|  | if (err == ERROR_IO_PENDING) { | 
|  | ret = GetOverlappedResult(s->file, &s->orecv, &size, TRUE); | 
|  | } | 
|  | } | 
|  |  | 
|  | if (size > 0) { | 
|  | qemu_chr_be_write(chr, buf, size); | 
|  | } | 
|  | } | 
|  |  | 
|  | static int win_chr_serial_poll(void *opaque) | 
|  | { | 
|  | Chardev *chr = CHARDEV(opaque); | 
|  | WinChardev *s = WIN_CHARDEV(opaque); | 
|  | COMSTAT status; | 
|  | DWORD comerr; | 
|  |  | 
|  | ClearCommError(s->file, &comerr, &status); | 
|  | if (status.cbInQue > 0) { | 
|  | win_chr_read(chr, status.cbInQue); | 
|  | return 1; | 
|  | } | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | int win_chr_serial_init(Chardev *chr, const char *filename, Error **errp) | 
|  | { | 
|  | WinChardev *s = WIN_CHARDEV(chr); | 
|  | COMMCONFIG comcfg; | 
|  | COMMTIMEOUTS cto = { 0, 0, 0, 0, 0}; | 
|  | COMSTAT comstat; | 
|  | DWORD size; | 
|  | DWORD err; | 
|  |  | 
|  | s->hsend = CreateEvent(NULL, TRUE, FALSE, NULL); | 
|  | if (!s->hsend) { | 
|  | error_setg(errp, "Failed CreateEvent"); | 
|  | goto fail; | 
|  | } | 
|  | s->hrecv = CreateEvent(NULL, TRUE, FALSE, NULL); | 
|  | if (!s->hrecv) { | 
|  | error_setg(errp, "Failed CreateEvent"); | 
|  | goto fail; | 
|  | } | 
|  |  | 
|  | s->file = CreateFile(filename, GENERIC_READ | GENERIC_WRITE, 0, NULL, | 
|  | OPEN_EXISTING, FILE_FLAG_OVERLAPPED, 0); | 
|  | if (s->file == INVALID_HANDLE_VALUE) { | 
|  | error_setg_win32(errp, GetLastError(), "Failed CreateFile"); | 
|  | s->file = NULL; | 
|  | goto fail; | 
|  | } | 
|  |  | 
|  | if (!SetupComm(s->file, NRECVBUF, NSENDBUF)) { | 
|  | error_setg(errp, "Failed SetupComm"); | 
|  | goto fail; | 
|  | } | 
|  |  | 
|  | ZeroMemory(&comcfg, sizeof(COMMCONFIG)); | 
|  | size = sizeof(COMMCONFIG); | 
|  | GetDefaultCommConfig(filename, &comcfg, &size); | 
|  | comcfg.dcb.DCBlength = sizeof(DCB); | 
|  | CommConfigDialog(filename, NULL, &comcfg); | 
|  |  | 
|  | if (!SetCommState(s->file, &comcfg.dcb)) { | 
|  | error_setg(errp, "Failed SetCommState"); | 
|  | goto fail; | 
|  | } | 
|  |  | 
|  | if (!SetCommMask(s->file, EV_ERR)) { | 
|  | error_setg(errp, "Failed SetCommMask"); | 
|  | goto fail; | 
|  | } | 
|  |  | 
|  | cto.ReadIntervalTimeout = MAXDWORD; | 
|  | if (!SetCommTimeouts(s->file, &cto)) { | 
|  | error_setg(errp, "Failed SetCommTimeouts"); | 
|  | goto fail; | 
|  | } | 
|  |  | 
|  | if (!ClearCommError(s->file, &err, &comstat)) { | 
|  | error_setg(errp, "Failed ClearCommError"); | 
|  | goto fail; | 
|  | } | 
|  | qemu_add_polling_cb(win_chr_serial_poll, chr); | 
|  | return 0; | 
|  |  | 
|  | fail: | 
|  | return -1; | 
|  | } | 
|  |  | 
|  | int win_chr_pipe_poll(void *opaque) | 
|  | { | 
|  | Chardev *chr = CHARDEV(opaque); | 
|  | WinChardev *s = WIN_CHARDEV(opaque); | 
|  | DWORD size; | 
|  |  | 
|  | PeekNamedPipe(s->file, NULL, 0, NULL, &size, NULL); | 
|  | if (size > 0) { | 
|  | win_chr_read(chr, size); | 
|  | return 1; | 
|  | } | 
|  | return 0; | 
|  | } | 
|  |  | 
|  | /* Called with chr_write_lock held.  */ | 
|  | static int win_chr_write(Chardev *chr, const uint8_t *buf, int len1) | 
|  | { | 
|  | WinChardev *s = WIN_CHARDEV(chr); | 
|  | DWORD len, ret, size, err; | 
|  |  | 
|  | len = len1; | 
|  | ZeroMemory(&s->osend, sizeof(s->osend)); | 
|  | s->osend.hEvent = s->hsend; | 
|  | while (len > 0) { | 
|  | if (s->hsend) { | 
|  | ret = WriteFile(s->file, buf, len, &size, &s->osend); | 
|  | } else { | 
|  | ret = WriteFile(s->file, buf, len, &size, NULL); | 
|  | } | 
|  | if (!ret) { | 
|  | err = GetLastError(); | 
|  | if (err == ERROR_IO_PENDING) { | 
|  | ret = GetOverlappedResult(s->file, &s->osend, &size, TRUE); | 
|  | if (ret) { | 
|  | buf += size; | 
|  | len -= size; | 
|  | } else { | 
|  | break; | 
|  | } | 
|  | } else { | 
|  | break; | 
|  | } | 
|  | } else { | 
|  | buf += size; | 
|  | len -= size; | 
|  | } | 
|  | } | 
|  | return len1 - len; | 
|  | } | 
|  |  | 
|  | static void char_win_finalize(Object *obj) | 
|  | { | 
|  | Chardev *chr = CHARDEV(obj); | 
|  | WinChardev *s = WIN_CHARDEV(chr); | 
|  |  | 
|  | if (s->hsend) { | 
|  | CloseHandle(s->hsend); | 
|  | } | 
|  | if (s->hrecv) { | 
|  | CloseHandle(s->hrecv); | 
|  | } | 
|  | if (!s->keep_open && s->file) { | 
|  | CloseHandle(s->file); | 
|  | } | 
|  | if (s->fpipe) { | 
|  | qemu_del_polling_cb(win_chr_pipe_poll, chr); | 
|  | } else { | 
|  | qemu_del_polling_cb(win_chr_serial_poll, chr); | 
|  | } | 
|  |  | 
|  | qemu_chr_be_event(chr, CHR_EVENT_CLOSED); | 
|  | } | 
|  |  | 
|  | void win_chr_set_file(Chardev *chr, HANDLE file, bool keep_open) | 
|  | { | 
|  | WinChardev *s = WIN_CHARDEV(chr); | 
|  |  | 
|  | s->keep_open = keep_open; | 
|  | s->file = file; | 
|  | } | 
|  |  | 
|  | static void char_win_class_init(ObjectClass *oc, void *data) | 
|  | { | 
|  | ChardevClass *cc = CHARDEV_CLASS(oc); | 
|  |  | 
|  | cc->chr_write = win_chr_write; | 
|  | } | 
|  |  | 
|  | static const TypeInfo char_win_type_info = { | 
|  | .name = TYPE_CHARDEV_WIN, | 
|  | .parent = TYPE_CHARDEV, | 
|  | .instance_size = sizeof(WinChardev), | 
|  | .instance_finalize = char_win_finalize, | 
|  | .class_init = char_win_class_init, | 
|  | .abstract = true, | 
|  | }; | 
|  |  | 
|  | static void register_types(void) | 
|  | { | 
|  | type_register_static(&char_win_type_info); | 
|  | } | 
|  |  | 
|  | type_init(register_types); |