1 #include <lunaix/device.h>
2 #include <lunaix/mm/valloc.h>
3 #include <lunaix/spike.h>
4 #include <lunaix/status.h>
6 #include <sys/mm/mempart.h>
8 #include <hal/serial.h>
10 #define lock_sdev(sdev) device_lock((sdev)->dev)
11 #define unlock_sdev(sdev) device_unlock((sdev)->dev)
12 #define unlock_and_wait(sdev, wq) \
19 static DEFINE_LLIST(serial_devs);
20 static int serial_idx = 0;
22 #define serial_device(dev) ((struct serial_dev*)(dev)->underlay)
25 serial_accept_one(struct serial_dev* sdev, u8_t val)
27 return !!rbuffer_put(&sdev->rxbuf, val);
31 serial_accept_buffer(struct serial_dev* sdev, void* val, size_t len)
33 return !!rbuffer_puts(&sdev->rxbuf, val, len);
37 serial_end_recv(struct serial_dev* sdev)
39 mark_device_done_read(sdev->dev);
41 pwake_one(&sdev->wq_rxdone);
45 serial_end_xmit(struct serial_dev* sdev, size_t len)
47 mark_device_done_write(sdev->dev);
50 pwake_one(&sdev->wq_txdone);
54 serial_readone_nowait(struct serial_dev* sdev, u8_t* val)
58 int rd_len = rbuffer_get(&sdev->rxbuf, (char*)val);
66 serial_readone(struct serial_dev* sdev, u8_t* val)
70 mark_device_doing_read(sdev->dev);
72 while (!rbuffer_get(&sdev->rxbuf, (char*)val)) {
73 unlock_and_wait(sdev, wq_rxdone);
80 serial_readbuf(struct serial_dev* sdev, u8_t* buf, size_t len)
84 mark_device_doing_read(sdev->dev);
87 while (!(rdlen = rbuffer_gets(&sdev->rxbuf, (char*)buf, len))) {
88 unlock_and_wait(sdev, wq_rxdone);
97 serial_readbuf_nowait(struct serial_dev* sdev, u8_t* buf, size_t len)
101 mark_device_doing_read(sdev->dev);
103 int rdlen = rbuffer_gets(&sdev->rxbuf, (char*)buf, len);
111 serial_writebuf(struct serial_dev* sdev, u8_t* buf, size_t len)
115 mark_device_doing_write(sdev->dev);
117 if (sdev->write(sdev, buf, len) == RXTX_DONE) {
121 unlock_and_wait(sdev, wq_txdone);
124 int rdlen = sdev->wr_len;
131 serial_writebuf_nowait(struct serial_dev* sdev, u8_t* buf, size_t len)
135 mark_device_doing_write(sdev->dev);
137 sdev->write(sdev, buf, len);
138 int rdlen = sdev->wr_len;
146 __serial_read(struct device* dev, void* buf, size_t offset, size_t len)
148 return serial_readbuf(serial_device(dev), &((u8_t*)buf)[offset], len);
152 __serial_read_async(struct device* dev, void* buf, size_t offset, size_t len)
154 return serial_readbuf_nowait(
155 serial_device(dev), &((u8_t*)buf)[offset], len);
159 __serial_read_page(struct device* dev, void* buf, size_t offset)
161 return serial_readbuf(serial_device(dev), &((u8_t*)buf)[offset], MEM_PAGE);
165 __serial_write(struct device* dev, void* buf, size_t offset, size_t len)
167 return serial_writebuf(serial_device(dev), &((u8_t*)buf)[offset], len);
171 __serial_write_async(struct device* dev, void* buf, size_t offset, size_t len)
173 return serial_writebuf_nowait(
174 serial_device(dev), &((u8_t*)buf)[offset], len);
178 __serial_write_page(struct device* dev, void* buf, size_t offset)
180 return serial_writebuf(serial_device(dev), &((u8_t*)buf)[offset], MEM_PAGE);
184 __serial_exec_command(struct device* dev, u32_t req, va_list args)
186 struct serial_dev* sdev = serial_device(dev);
188 if (!sdev->exec_cmd) {
192 return sdev->exec_cmd(sdev, req, args);
196 __serial_poll_event(struct device* dev)
198 struct serial_dev* sdev = serial_device(dev);
200 return sdev->dev->poll_evflags;
203 #define RXBUF_SIZE 512
206 serial_create(struct devclass* class, char* if_ident)
208 struct serial_dev* sdev = valloc(sizeof(struct serial_dev));
209 struct device* dev = device_allocseq(NULL, sdev);
210 dev->ops.read = __serial_read;
211 dev->ops.read_page = __serial_read_page;
212 dev->ops.write = __serial_write;
213 dev->ops.write_page = __serial_write_page;
214 dev->ops.exec_cmd = __serial_exec_command;
215 dev->ops.poll = __serial_poll_event;
218 dev->underlay = sdev;
220 waitq_init(&sdev->wq_rxdone);
221 waitq_init(&sdev->wq_txdone);
222 rbuffer_init(&sdev->rxbuf, valloc(RXBUF_SIZE), RXBUF_SIZE);
223 llist_append(&serial_devs, &sdev->sdev_list);
225 device_register(dev, class, "port%s%d", if_ident, class->variant);
227 sdev->at_term = term_create(dev, if_ident);
234 serial_get_avilable()
236 struct serial_dev *pos, *n;
237 llist_for_each(pos, n, &serial_devs, sdev_list)
239 if (!device_locked(pos->dev)) {