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>
6 #include <lunaix/syslog.h>
8 #include <sys/mm/pagetable.h>
10 #include <hal/serial.h>
15 #define lock_sdev(sdev) device_lock((sdev)->dev)
16 #define unlock_sdev(sdev) device_unlock((sdev)->dev)
17 #define unlock_and_wait(sdev, wq) \
24 static DEFINE_LLIST(serial_devs);
25 static int serial_idx = 0;
27 static struct device_cat* serial_cat;
29 #define serial_device(dev) ((struct serial_dev*)(dev)->underlay)
32 serial_accept_one(struct serial_dev* sdev, u8_t val)
34 return !!rbuffer_put(&sdev->rxbuf, val);
38 serial_accept_buffer(struct serial_dev* sdev, void* val, size_t len)
40 return !!rbuffer_puts(&sdev->rxbuf, val, len);
44 serial_end_recv(struct serial_dev* sdev)
46 mark_device_done_read(sdev->dev);
48 pwake_one(&sdev->wq_rxdone);
50 struct termport_capability* tpcap;
51 tpcap = get_capability(sdev->tp_cap, typeof(*tpcap));
52 term_notify_data_avaliable(tpcap);
56 serial_end_xmit(struct serial_dev* sdev, size_t len)
58 mark_device_done_write(sdev->dev);
61 pwake_one(&sdev->wq_txdone);
65 serial_readone_nowait(struct serial_dev* sdev, u8_t* val)
69 int rd_len = rbuffer_get(&sdev->rxbuf, (char*)val);
77 serial_readone(struct serial_dev* sdev, u8_t* val)
81 mark_device_doing_read(sdev->dev);
83 while (!rbuffer_get(&sdev->rxbuf, (char*)val)) {
84 unlock_and_wait(sdev, wq_rxdone);
91 serial_readbuf(struct serial_dev* sdev, u8_t* buf, size_t len)
95 mark_device_doing_read(sdev->dev);
98 while (!(rdlen = rbuffer_gets(&sdev->rxbuf, (char*)buf, len))) {
99 unlock_and_wait(sdev, wq_rxdone);
108 serial_readbuf_nowait(struct serial_dev* sdev, u8_t* buf, size_t len)
112 mark_device_doing_read(sdev->dev);
114 int rdlen = rbuffer_gets(&sdev->rxbuf, (char*)buf, len);
122 serial_writebuf(struct serial_dev* sdev, u8_t* buf, size_t len)
126 mark_device_doing_write(sdev->dev);
128 if (sdev->write(sdev, buf, len) == RXTX_DONE) {
132 unlock_and_wait(sdev, wq_txdone);
135 int rdlen = sdev->wr_len;
142 serial_writebuf_nowait(struct serial_dev* sdev, u8_t* buf, size_t len)
146 mark_device_doing_write(sdev->dev);
148 sdev->write(sdev, buf, len);
149 int rdlen = sdev->wr_len;
157 __serial_read(struct device* dev, void* buf, off_t fpos, size_t len)
159 return serial_readbuf(serial_device(dev), (u8_t*)buf, len);
163 __serial_read_async(struct device* dev, void* buf, off_t fpos, size_t len)
165 return serial_readbuf_nowait(
166 serial_device(dev), (u8_t*)buf, len);
170 __serial_read_page(struct device* dev, void* buf, off_t fpos)
172 return serial_readbuf(serial_device(dev), (u8_t*)buf, PAGE_SIZE);
176 __serial_write(struct device* dev, void* buf, off_t fpos, size_t len)
178 return serial_writebuf(serial_device(dev), (u8_t*)buf, len);
182 __serial_write_async(struct device* dev, void* buf, off_t fpos, size_t len)
184 return serial_writebuf_nowait(
185 serial_device(dev), (u8_t*)buf, len);
189 __serial_write_page(struct device* dev, void* buf, off_t fpos)
191 return serial_writebuf(serial_device(dev), (u8_t*)buf, PAGE_SIZE);
195 __serial_exec_command(struct device* dev, u32_t req, va_list args)
197 struct serial_dev* sdev = serial_device(dev);
199 if (!sdev->exec_cmd) {
203 return sdev->exec_cmd(sdev, req, args);
207 __serial_poll_event(struct device* dev)
209 struct serial_dev* sdev = serial_device(dev);
211 return sdev->dev->poll_evflags;
214 static void sdev_execmd(struct serial_dev* sdev, u32_t req, ...)
219 sdev->exec_cmd(sdev, req, args);
225 __serial_set_speed(struct device* dev, speed_t speed)
227 struct serial_dev* sdev = serial_device(dev);
230 sdev_execmd(sdev, SERIO_SETBRDRATE, speed);
236 __serial_set_baseclk(struct device* dev, unsigned int base)
238 struct serial_dev* sdev = serial_device(dev);
241 sdev_execmd(sdev, SERIO_SETBRDBASE, base);
247 __serial_set_cntrl_mode(struct device* dev, tcflag_t cflag)
249 struct serial_dev* sdev = serial_device(dev);
252 sdev_execmd(sdev, SERIO_SETCNTRLMODE, cflag);
257 #define RXBUF_SIZE 512
259 static struct termport_cap_ops tpcap_ops = {
260 .set_cntrl_mode = __serial_set_cntrl_mode,
261 .set_clkbase = __serial_set_baseclk,
262 .set_speed = __serial_set_speed
266 serial_create(struct devclass* class, char* if_ident)
268 struct serial_dev* sdev = vzalloc(sizeof(struct serial_dev));
269 struct device* dev = device_allocseq(dev_meta(serial_cat), sdev);
270 dev->ops.read = __serial_read;
271 dev->ops.read_page = __serial_read_page;
272 dev->ops.read_async = __serial_read_async;
273 dev->ops.write_async = __serial_write_async;
274 dev->ops.write = __serial_write;
275 dev->ops.write_page = __serial_write_page;
276 dev->ops.exec_cmd = __serial_exec_command;
277 dev->ops.poll = __serial_poll_event;
280 dev->underlay = sdev;
282 struct termport_capability* tp_cap =
283 new_capability(TERMPORT_CAP, struct termport_capability);
285 term_cap_set_operations(tp_cap, &tpcap_ops);
286 sdev->tp_cap = cap_meta(tp_cap);
288 waitq_init(&sdev->wq_rxdone);
289 waitq_init(&sdev->wq_txdone);
290 rbuffer_init(&sdev->rxbuf, valloc(RXBUF_SIZE), RXBUF_SIZE);
291 llist_append(&serial_devs, &sdev->sdev_list);
293 device_grant_capability(dev, cap_meta(tp_cap));
295 register_device(dev, class, "%s%d", if_ident, class->variant);
297 term_create(dev, if_ident);
299 INFO("interface: %s, %xh:%xh.%d", dev->name_val,
300 class->fn_grp, class->device, class->variant);
307 serial_get_avilable()
309 struct serial_dev *pos, *n;
310 llist_for_each(pos, n, &serial_devs, sdev_list)
312 if (!device_locked(pos->dev)) {
323 serial_cat = device_addcat(NULL, "serial");
327 owloysius_fetch_init(init_serial_dev, on_earlyboot)