1 #include <lunaix/device.h>
2 #include <lunaix/mm/valloc.h>
3 #include <lunaix/spike.h>
4 #include <lunaix/owloysius.h>
5 #include <lunaix/status.h>
7 #include <sys/mm/mempart.h>
9 #include <hal/serial.h>
12 #define lock_sdev(sdev) device_lock((sdev)->dev)
13 #define unlock_sdev(sdev) device_unlock((sdev)->dev)
14 #define unlock_and_wait(sdev, wq) \
21 static DEFINE_LLIST(serial_devs);
22 static int serial_idx = 0;
24 static struct device_cat* serial_cat;
26 #define serial_device(dev) ((struct serial_dev*)(dev)->underlay)
29 serial_accept_one(struct serial_dev* sdev, u8_t val)
31 return !!rbuffer_put(&sdev->rxbuf, val);
35 serial_accept_buffer(struct serial_dev* sdev, void* val, size_t len)
37 return !!rbuffer_puts(&sdev->rxbuf, val, len);
41 serial_end_recv(struct serial_dev* sdev)
43 mark_device_done_read(sdev->dev);
45 pwake_one(&sdev->wq_rxdone);
49 serial_end_xmit(struct serial_dev* sdev, size_t len)
51 mark_device_done_write(sdev->dev);
54 pwake_one(&sdev->wq_txdone);
58 serial_readone_nowait(struct serial_dev* sdev, u8_t* val)
62 int rd_len = rbuffer_get(&sdev->rxbuf, (char*)val);
70 serial_readone(struct serial_dev* sdev, u8_t* val)
74 mark_device_doing_read(sdev->dev);
76 while (!rbuffer_get(&sdev->rxbuf, (char*)val)) {
77 unlock_and_wait(sdev, wq_rxdone);
84 serial_readbuf(struct serial_dev* sdev, u8_t* buf, size_t len)
88 mark_device_doing_read(sdev->dev);
91 while (!(rdlen = rbuffer_gets(&sdev->rxbuf, (char*)buf, len))) {
92 unlock_and_wait(sdev, wq_rxdone);
101 serial_readbuf_nowait(struct serial_dev* sdev, u8_t* buf, size_t len)
105 mark_device_doing_read(sdev->dev);
107 int rdlen = rbuffer_gets(&sdev->rxbuf, (char*)buf, len);
115 serial_writebuf(struct serial_dev* sdev, u8_t* buf, size_t len)
119 mark_device_doing_write(sdev->dev);
121 if (sdev->write(sdev, buf, len) == RXTX_DONE) {
125 unlock_and_wait(sdev, wq_txdone);
128 int rdlen = sdev->wr_len;
135 serial_writebuf_nowait(struct serial_dev* sdev, u8_t* buf, size_t len)
139 mark_device_doing_write(sdev->dev);
141 sdev->write(sdev, buf, len);
142 int rdlen = sdev->wr_len;
150 __serial_read(struct device* dev, void* buf, off_t fpos, size_t len)
152 return serial_readbuf(serial_device(dev), (u8_t*)buf, len);
156 __serial_read_async(struct device* dev, void* buf, off_t fpos, size_t len)
158 return serial_readbuf_nowait(
159 serial_device(dev), (u8_t*)buf, len);
163 __serial_read_page(struct device* dev, void* buf, off_t fpos)
165 return serial_readbuf(serial_device(dev), (u8_t*)buf, MEM_PAGE);
169 __serial_write(struct device* dev, void* buf, off_t fpos, size_t len)
171 return serial_writebuf(serial_device(dev), (u8_t*)buf, len);
175 __serial_write_async(struct device* dev, void* buf, off_t fpos, size_t len)
177 return serial_writebuf_nowait(
178 serial_device(dev), (u8_t*)buf, len);
182 __serial_write_page(struct device* dev, void* buf, off_t fpos)
184 return serial_writebuf(serial_device(dev), (u8_t*)buf, MEM_PAGE);
188 __serial_exec_command(struct device* dev, u32_t req, va_list args)
190 struct serial_dev* sdev = serial_device(dev);
192 if (!sdev->exec_cmd) {
196 return sdev->exec_cmd(sdev, req, args);
200 __serial_poll_event(struct device* dev)
202 struct serial_dev* sdev = serial_device(dev);
204 return sdev->dev->poll_evflags;
207 static void sdev_execmd(struct serial_dev* sdev, u32_t req, ...)
212 sdev->exec_cmd(sdev, req, args);
218 __serial_set_speed(struct device* dev, speed_t speed)
220 struct serial_dev* sdev = serial_device(dev);
223 sdev_execmd(sdev, SERIO_SETBRDIV, speed);
229 __serial_set_cntrl_mode(struct device* dev, tcflag_t cflag)
231 struct serial_dev* sdev = serial_device(dev);
234 sdev_execmd(sdev, SERIO_SETCNTRLMODE, cflag);
239 #define RXBUF_SIZE 512
242 serial_create(struct devclass* class, char* if_ident)
244 struct serial_dev* sdev = valloc(sizeof(struct serial_dev));
245 struct device* dev = device_allocseq(dev_meta(serial_cat), sdev);
246 dev->ops.read = __serial_read;
247 dev->ops.read_page = __serial_read_page;
248 dev->ops.read_async = __serial_read_async;
249 dev->ops.write_async = __serial_write_async;
250 dev->ops.write = __serial_write;
251 dev->ops.write_page = __serial_write_page;
252 dev->ops.exec_cmd = __serial_exec_command;
253 dev->ops.poll = __serial_poll_event;
256 dev->underlay = sdev;
258 struct termport_capability* tp_cap =
259 new_capability(TERMPORT_CAP, struct termport_capability);
260 tp_cap->set_speed = __serial_set_speed;
261 tp_cap->set_cntrl_mode = __serial_set_cntrl_mode;
263 waitq_init(&sdev->wq_rxdone);
264 waitq_init(&sdev->wq_txdone);
265 rbuffer_init(&sdev->rxbuf, valloc(RXBUF_SIZE), RXBUF_SIZE);
266 llist_append(&serial_devs, &sdev->sdev_list);
268 device_grant_capability(dev, cap_meta(tp_cap));
270 register_device(dev, class, "s%d", class->variant);
272 term_create(dev, if_ident);
279 serial_get_avilable()
281 struct serial_dev *pos, *n;
282 llist_for_each(pos, n, &serial_devs, sdev_list)
284 if (!device_locked(pos->dev)) {
295 serial_cat = device_addcat(NULL, "serial");
299 owloysius_fetch_init(init_serial_dev, on_earlyboot)