source: mainline/uspace/drv/char/pl011/main.c@ 7348c4b

lfn serial ticket/834-toolchain-update topic/msim-upgrade topic/simplify-dev-export
Last change on this file since 7348c4b was 7348c4b, checked in by Jakub Jermář <jakub@…>, 6 years ago

arm64: Add PL011 uspace driver

  • Property mode set to 100644
File size: 4.0 KB
Line 
1/*
2 * Copyright (c) 2018 Petr Pavlu
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions
7 * are met:
8 *
9 * - Redistributions of source code must retain the above copyright
10 * notice, this list of conditions and the following disclaimer.
11 * - Redistributions in binary form must reproduce the above copyright
12 * notice, this list of conditions and the following disclaimer in the
13 * documentation and/or other materials provided with the distribution.
14 * - The name of the author may not be used to endorse or promote products
15 * derived from this software without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
18 * IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
19 * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
20 * IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
21 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
22 * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24 * THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
26 * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 */
28
29/** @addtogroup uspace_drv_pl011
30 * @{
31 */
32/** @file ARM PrimeCell PL011 UART driver.
33 */
34
35#include <ddf/driver.h>
36#include <ddf/log.h>
37#include <device/hw_res_parsed.h>
38#include <errno.h>
39#include <stdio.h>
40
41#include "pl011.h"
42
43#define NAME "pl011"
44
45static errno_t pl011_dev_add(ddf_dev_t *dev);
46static errno_t pl011_dev_remove(ddf_dev_t *dev);
47static errno_t pl011_dev_gone(ddf_dev_t *dev);
48static errno_t pl011_fun_online(ddf_fun_t *fun);
49static errno_t pl011_fun_offline(ddf_fun_t *fun);
50
51static driver_ops_t driver_ops = {
52 .dev_add = pl011_dev_add,
53 .dev_remove = pl011_dev_remove,
54 .dev_gone = pl011_dev_gone,
55 .fun_online = pl011_fun_online,
56 .fun_offline = pl011_fun_offline
57};
58
59static driver_t pl011_driver = {
60 .name = NAME,
61 .driver_ops = &driver_ops
62};
63
64static errno_t pl011_get_res(ddf_dev_t *dev, pl011_res_t *res)
65{
66 async_sess_t *parent_sess;
67 hw_res_list_parsed_t hw_res;
68 errno_t rc;
69
70 parent_sess = ddf_dev_parent_sess_get(dev);
71 if (parent_sess == NULL)
72 return ENOMEM;
73
74 hw_res_list_parsed_init(&hw_res);
75 rc = hw_res_get_list_parsed(parent_sess, &hw_res, 0);
76 if (rc != EOK)
77 return rc;
78
79 if (hw_res.mem_ranges.count != 1) {
80 rc = EINVAL;
81 goto error;
82 }
83
84 res->base = RNGABS(hw_res.mem_ranges.ranges[0]);
85
86 if (hw_res.irqs.count != 1) {
87 rc = EINVAL;
88 goto error;
89 }
90
91 res->irq = hw_res.irqs.irqs[0];
92
93 return EOK;
94error:
95 hw_res_list_parsed_clean(&hw_res);
96 return rc;
97}
98
99static errno_t pl011_dev_add(ddf_dev_t *dev)
100{
101 pl011_t *pl011;
102 pl011_res_t res;
103 errno_t rc;
104
105 ddf_msg(LVL_DEBUG, "pl011_dev_add(%p)", dev);
106
107 pl011 = ddf_dev_data_alloc(dev, sizeof(pl011_t));
108 if (pl011 == NULL) {
109 ddf_msg(LVL_ERROR, "Failed allocating soft state.");
110 return ENOMEM;
111 }
112
113 pl011->dev = dev;
114
115 rc = pl011_get_res(dev, &res);
116 if (rc != EOK) {
117 ddf_msg(LVL_ERROR, "Failed getting hardware resource list.");
118 return EIO;
119 }
120
121 return pl011_add(pl011, &res);
122}
123
124static errno_t pl011_dev_remove(ddf_dev_t *dev)
125{
126 pl011_t *pl011 = (pl011_t *)ddf_dev_data_get(dev);
127
128 ddf_msg(LVL_DEBUG, "pl011_dev_remove(%p)", dev);
129
130 return pl011_remove(pl011);
131}
132
133static errno_t pl011_dev_gone(ddf_dev_t *dev)
134{
135 pl011_t *pl011 = (pl011_t *)ddf_dev_data_get(dev);
136
137 ddf_msg(LVL_DEBUG, "pl011_dev_gone(%p)", dev);
138
139 return pl011_gone(pl011);
140}
141
142static errno_t pl011_fun_online(ddf_fun_t *fun)
143{
144 ddf_msg(LVL_DEBUG, "pl011_fun_online()");
145 return ddf_fun_online(fun);
146}
147
148static errno_t pl011_fun_offline(ddf_fun_t *fun)
149{
150 ddf_msg(LVL_DEBUG, "pl011_fun_offline()");
151 return ddf_fun_offline(fun);
152}
153
154int main(int argc, char *argv[])
155{
156 printf(NAME ": PL011 serial device driver\n");
157 ddf_log_init(NAME);
158 return ddf_driver_main(&pl011_driver);
159}
160
161/** @}
162 */
Note: See TracBrowser for help on using the repository browser.