/* * Server-side serial port communications management * * Copyright (C) 1998 Alexandre Julliard * Copyright (C) 2000 Mike McCormack * * TODO: * Add async read, write and WaitCommEvent handling. * */ #include "config.h" #include #include #include #include #include #include #ifdef HAVE_SYS_ERRNO_H #include #endif #include #include #include #include #include #include #include #include #include "winerror.h" #include "winbase.h" #include "handle.h" #include "thread.h" #include "request.h" static void serial_dump( struct object *obj, int verbose ); static int serial_get_fd( struct object *obj ); static int serial_get_info( struct object *obj, struct get_file_info_request *req ); static int serial_get_poll_events( struct object *obj ); struct serial { struct object obj; unsigned int access; /* timeout values */ unsigned int readinterval; unsigned int readconst; unsigned int readmult; unsigned int writeconst; unsigned int writemult; unsigned int eventmask; unsigned int commerror; struct termios original; /* FIXME: add dcb, comm status, handler module, sharing */ }; static const struct object_ops serial_ops = { sizeof(struct serial), /* size */ serial_dump, /* dump */ default_poll_add_queue, /* add_queue */ default_poll_remove_queue, /* remove_queue */ default_poll_signaled, /* signaled */ no_satisfied, /* satisfied */ serial_get_poll_events, /* get_poll_events */ default_poll_event, /* poll_event */ serial_get_fd, /* get_fd */ no_flush, /* flush */ serial_get_info, /* get_file_info */ no_destroy /* destroy */ }; /* SERIAL PORT functions */ static struct serial *create_serial( const char *nameptr, size_t len, unsigned int access ) { struct serial *serial; struct termios tios; int fd, flags = 0; char *name; if (!(name = mem_alloc( len + 1 ))) return NULL; memcpy( name, nameptr, len ); name[len] = 0; switch(access & (GENERIC_READ | GENERIC_WRITE)) { case GENERIC_READ: flags |= O_RDONLY; break; case GENERIC_WRITE: flags |= O_WRONLY; break; case GENERIC_READ|GENERIC_WRITE: flags |= O_RDWR; break; default: break; } fd = open( name, flags ); free( name ); if (fd < 0) { file_set_error(); return NULL; } /* check its really a serial port */ if (tcgetattr(fd,&tios)) { file_set_error(); close( fd ); return NULL; } if ((serial = alloc_object( &serial_ops, fd ))) { serial->access = access; serial->readinterval = 0; serial->readmult = 0; serial->readconst = 0; serial->writemult = 0; serial->writeconst = 0; serial->eventmask = 0; serial->commerror = 0; } return serial; } static void serial_dump( struct object *obj, int verbose ) { struct serial *serial = (struct serial *)obj; assert( obj->ops == &serial_ops ); fprintf( stderr, "Port fd=%d mask=%x\n", serial->obj.fd, serial->eventmask ); } struct serial *get_serial_obj( struct process *process, handle_t handle, unsigned int access ) { return (struct serial *)get_handle_obj( process, handle, access, &serial_ops ); } static int serial_get_poll_events( struct object *obj ) { struct serial *serial = (struct serial *)obj; int events = 0; assert( obj->ops == &serial_ops ); if (serial->access & GENERIC_READ) events |= POLLIN; if (serial->access & GENERIC_WRITE) events |= POLLOUT; return events; } static int serial_get_fd( struct object *obj ) { struct serial *serial = (struct serial *)obj; assert( obj->ops == &serial_ops ); return serial->obj.fd; } static int serial_get_info( struct object *obj, struct get_file_info_request *req ) { assert( obj->ops == &serial_ops ); req->type = FILE_TYPE_CHAR; req->attr = 0; req->access_time = 0; req->write_time = 0; req->size_high = 0; req->size_low = 0; req->links = 0; req->index_high = 0; req->index_low = 0; req->serial = 0; return 1; } /* these functions are for interaction with asynchronous i/o objects */ int serial_async_setup(struct object *obj, struct async *ov) { struct serial *serial = (struct serial *)obj; int timeout; if(obj->ops != &serial_ops) return 0; switch(async_type(ov)) { case ASYNC_TYPE_READ: timeout = serial->readconst + serial->readmult*async_count(ov); async_add_timeout(ov, timeout); async_set_eventmask(ov,EV_RXCHAR); break; case ASYNC_TYPE_WRITE: timeout = serial->writeconst + serial->writemult*async_count(ov); async_add_timeout(ov, timeout); async_set_eventmask(ov,EV_TXEMPTY); break; case ASYNC_TYPE_WAIT: async_set_eventmask(ov,serial->eventmask); break; } return 1; } int serial_async_get_poll_events( struct async *ov ) { int events=0,mask; switch(async_type(ov)) { case ASYNC_TYPE_READ: events |= POLLIN; break; case ASYNC_TYPE_WRITE: events |= POLLOUT; break; case ASYNC_TYPE_WAIT: /* * FIXME: here is the spot to implement other WaitCommEvent flags */ mask = async_get_eventmask(ov); if(mask&EV_RXCHAR) events |= POLLIN; if(mask&EV_TXEMPTY) events |= POLLOUT; break; } return events; } /* receive a select event, and output a windows event */ int serial_async_poll_event(struct object *obj, int event) { int r=0; /* * FIXME: here is the spot to implement other WaitCommEvent flags */ if(event & POLLIN) r |= EV_RXCHAR; if(event & POLLOUT) r |= EV_TXEMPTY; return r; } /* create a serial */ DECL_HANDLER(create_serial) { struct serial *serial; req->handle = 0; if ((serial = create_serial( get_req_data(req), get_req_data_size(req), req->access ))) { req->handle = alloc_handle( current->process, serial, req->access, req->inherit ); release_object( serial ); } } DECL_HANDLER(get_serial_info) { struct serial *serial; if ((serial = get_serial_obj( current->process, req->handle, 0 ))) { /* timeouts */ req->readinterval = serial->readinterval; req->readconst = serial->readconst; req->readmult = serial->readmult; req->writeconst = serial->writeconst; req->writemult = serial->writemult; /* event mask */ req->eventmask = serial->eventmask; /* comm port error status */ req->commerror = serial->commerror; release_object( serial ); } } DECL_HANDLER(set_serial_info) { struct serial *serial; if ((serial = get_serial_obj( current->process, req->handle, 0 ))) { /* timeouts */ if(req->flags & SERIALINFO_SET_TIMEOUTS) { serial->readinterval = req->readinterval; serial->readconst = req->readconst; serial->readmult = req->readmult; serial->writeconst = req->writeconst; serial->writemult = req->writemult; } /* event mask */ if(req->flags & SERIALINFO_SET_MASK) { serial->eventmask = req->eventmask; } /* comm port error status */ if(req->flags & SERIALINFO_SET_ERROR) { serial->commerror = req->commerror; } release_object( serial ); } }