123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198 |
- /*
- * Copyright (c) 2013 Google, Inc
- *
- * SPDX-License-Identifier: GPL-2.0+
- */
- #include <common.h>
- #include <dm.h>
- #include <errno.h>
- #include <fdtdec.h>
- #include <malloc.h>
- #include <dm-demo.h>
- #include <asm/io.h>
- #include <asm/gpio.h>
- DECLARE_GLOBAL_DATA_PTR;
- /* Shape size */
- #define WIDTH 8
- #define HEIGHT 6
- struct shape_data {
- int num_chars; /* Number of non-space characters output so far */
- struct gpio_desc gpio_desc[8];
- int gpio_count;
- };
- /* Crazy little function to draw shapes on the console */
- static int shape_hello(struct udevice *dev, int ch)
- {
- const struct dm_demo_pdata *pdata = dev_get_platdata(dev);
- struct shape_data *data = dev_get_priv(dev);
- static const struct shape {
- int start;
- int end;
- int dstart;
- int dend;
- } shapes[3] = {
- { 0, 1, 0, 1 },
- { 0, WIDTH, 0, 0 },
- { HEIGHT / 2 - 1, WIDTH - HEIGHT / 2 + 1, -1, 1},
- };
- struct shape shape;
- unsigned int index;
- int line, pos, inside;
- const char *colour = pdata->colour;
- int first = 0;
- if (!ch)
- ch = pdata->default_char;
- if (!ch)
- ch = '@';
- index = (pdata->sides / 2) - 1;
- if (index >= ARRAY_SIZE(shapes))
- return -EIO;
- shape = shapes[index];
- for (line = 0; line < HEIGHT; line++) {
- first = 1;
- for (pos = 0; pos < WIDTH; pos++) {
- inside = pos >= shape.start && pos < shape.end;
- if (inside) {
- putc(first ? *colour++ : ch);
- data->num_chars++;
- first = 0;
- if (!*colour)
- colour = pdata->colour;
- } else {
- putc(' ');
- }
- }
- putc('\n');
- shape.start += shape.dstart;
- shape.end += shape.dend;
- if (shape.start < 0) {
- shape.dstart = -shape.dstart;
- shape.dend = -shape.dend;
- shape.start += shape.dstart;
- shape.end += shape.dend;
- }
- }
- return 0;
- }
- static int shape_status(struct udevice *dev, int *status)
- {
- struct shape_data *data = dev_get_priv(dev);
- *status = data->num_chars;
- return 0;
- }
- static int set_light(struct udevice *dev, int light)
- {
- struct shape_data *priv = dev_get_priv(dev);
- struct gpio_desc *desc;
- int ret;
- int i;
- desc = priv->gpio_desc;
- for (i = 0; i < priv->gpio_count; i++, desc++) {
- uint mask = 1 << i;
- ret = dm_gpio_set_value(desc, light & mask);
- if (ret < 0)
- return ret;
- }
- return 0;
- }
- static int get_light(struct udevice *dev)
- {
- struct shape_data *priv = dev_get_priv(dev);
- struct gpio_desc *desc;
- uint value = 0;
- int ret;
- int i;
- desc = priv->gpio_desc;
- for (i = 0; i < priv->gpio_count; i++, desc++) {
- uint mask = 1 << i;
- ret = dm_gpio_get_value(desc);
- if (ret < 0)
- return ret;
- if (ret)
- value |= mask;
- }
- return value;
- }
- static const struct demo_ops shape_ops = {
- .hello = shape_hello,
- .status = shape_status,
- .get_light = get_light,
- .set_light = set_light,
- };
- static int shape_ofdata_to_platdata(struct udevice *dev)
- {
- struct dm_demo_pdata *pdata = dev_get_platdata(dev);
- int ret;
- /* Parse the data that is common with all demo devices */
- ret = demo_parse_dt(dev);
- if (ret)
- return ret;
- /* Parse the data that only we need */
- pdata->default_char = fdtdec_get_int(gd->fdt_blob, dev->of_offset,
- "character", '@');
- return 0;
- }
- static int dm_shape_probe(struct udevice *dev)
- {
- struct shape_data *priv = dev_get_priv(dev);
- int ret;
- ret = gpio_request_list_by_name(dev, "light-gpios", priv->gpio_desc,
- ARRAY_SIZE(priv->gpio_desc),
- GPIOD_IS_OUT | GPIOD_IS_OUT_ACTIVE);
- if (ret < 0)
- return ret;
- priv->gpio_count = ret;
- debug("%s: %d GPIOs\n", __func__, priv->gpio_count);
- return 0;
- }
- static int dm_shape_remove(struct udevice *dev)
- {
- struct shape_data *priv = dev_get_priv(dev);
- return gpio_free_list(dev, priv->gpio_desc, priv->gpio_count);
- }
- static const struct udevice_id demo_shape_id[] = {
- { "demo-shape", 0 },
- { },
- };
- U_BOOT_DRIVER(demo_shape_drv) = {
- .name = "demo_shape_drv",
- .of_match = demo_shape_id,
- .id = UCLASS_DEMO,
- .ofdata_to_platdata = shape_ofdata_to_platdata,
- .ops = &shape_ops,
- .probe = dm_shape_probe,
- .remove = dm_shape_remove,
- .priv_auto_alloc_size = sizeof(struct shape_data),
- .platdata_auto_alloc_size = sizeof(struct dm_demo_pdata),
- };
|