Landfall

Annotated pkg/landfall-examples/letux400_keypad/letux400_keypad.c

229:810ca800993b
11 months ago Paul Boddie Employ a picture representation compatible with 16-bit SPI transfers over DMA. cpm-library-improvements
paul@0 1
/*
paul@0 2
 * Access the keypad GPIOs on the Letux 400 notebook computer.
paul@0 3
 *
paul@31 4
 * Copyright (C) 2018 Paul Boddie <paul@boddie.org.uk>
paul@0 5
 *
paul@0 6
 * This program is free software; you can redistribute it and/or
paul@0 7
 * modify it under the terms of the GNU General Public License as
paul@0 8
 * published by the Free Software Foundation; either version 2 of
paul@0 9
 * the License, or (at your option) any later version.
paul@0 10
 *
paul@0 11
 * This program is distributed in the hope that it will be useful,
paul@0 12
 * but WITHOUT ANY WARRANTY; without even the implied warranty of
paul@0 13
 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
paul@0 14
 * GNU General Public License for more details.
paul@0 15
 *
paul@0 16
 * You should have received a copy of the GNU General Public License
paul@0 17
 * along with this program; if not, write to the Free Software
paul@0 18
 * Foundation, Inc., 51 Franklin Street, Fifth Floor,
paul@0 19
 * Boston, MA  02110-1301, USA
paul@0 20
 */
paul@0 21
paul@0 22
#include <l4/devices/gpio-jz4730.h>
paul@0 23
#include <l4/re/c/util/video/goos_fb.h>
paul@0 24
#include <l4/re/c/video/view.h>
paul@0 25
#include <l4/sys/cache.h>
paul@0 26
#include <l4/util/util.h>
paul@0 27
paul@0 28
#include <stdio.h>
paul@0 29
#include <stdint.h>
paul@0 30
#include <unistd.h>
paul@0 31
paul@0 32
#include "memory.h"
paul@0 33
paul@0 34
enum Jz4730_keypad_gpio
paul@0 35
{
paul@0 36
  Jz4730_keypad_gpio_inputs_count = 8,
paul@0 37
  Jz4730_keypad_gpio_outputs_count = 17,
paul@0 38
};
paul@0 39
paul@0 40
/* Port A input pins. */
paul@0 41
paul@0 42
const uint8_t Jz4730_keypad_inputs[Jz4730_keypad_gpio_inputs_count] = {
paul@0 43
  0, 1, 2, 3, 4, 5, 6, 7
paul@0 44
};
paul@0 45
paul@0 46
const Pin_slice Jz4730_keypad_inputs_mask = {.mask=0x000000ff, .offset=0};
paul@0 47
paul@0 48
/* Port D output pins. */
paul@0 49
paul@0 50
const uint8_t Jz4730_keypad_outputs[Jz4730_keypad_gpio_outputs_count] = {
paul@0 51
  0, 1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12, 13, 14, 15, 29
paul@0 52
};
paul@0 53
paul@0 54
const Pin_slice Jz4730_keypad_outputs_mask = {.mask=0x2000ffff, .offset=0};
paul@0 55
paul@0 56
/* Video abstractions. */
paul@0 57
paul@0 58
static l4re_util_video_goos_fb_t gfb;
paul@0 59
static l4re_video_view_info_t fbi;
paul@0 60
static void *fb;
paul@0 61
paul@0 62
/* Peripheral memory regions. */
paul@0 63
paul@0 64
static l4_addr_t gpio_virt_base = 0, gpio_virt_base_end = 0;
paul@0 65
paul@0 66
/* GPIO abstractions. */
paul@0 67
paul@0 68
static void *gpio_port_a, *gpio_port_d;
paul@0 69
paul@0 70
/* Keypad status. */
paul@0 71
paul@0 72
uint32_t keypad[Jz4730_keypad_gpio_outputs_count];
paul@0 73
paul@0 74
paul@0 75
paul@0 76
/* Initialise the pins for scanning the keypad. */
paul@0 77
paul@0 78
static void init_keyscan(void)
paul@0 79
{
paul@0 80
  jz4730_gpio_multi_setup(gpio_port_a, &Jz4730_keypad_inputs_mask, Fix_input, 0);
paul@0 81
  jz4730_gpio_multi_config_pull(gpio_port_a, &Jz4730_keypad_inputs_mask, Pull_up);
paul@0 82
  jz4730_gpio_multi_setup(gpio_port_d, &Jz4730_keypad_outputs_mask, Fix_input, 0);
paul@0 83
}
paul@0 84
paul@0 85
/*
paul@0 86
Scan the keypad by enabling each output column and inspecting each input row.
paul@0 87
Store each column bitmap in the keypad array.
paul@0 88
*/
paul@0 89
paul@0 90
static void scan_keypad(void)
paul@0 91
{
paul@0 92
  uint8_t column, row;
paul@0 93
  uint32_t value;
paul@0 94
paul@0 95
  for (column = 0; column < Jz4730_keypad_gpio_outputs_count; column++)
paul@0 96
  {
paul@0 97
    jz4730_gpio_setup(gpio_port_d, Jz4730_keypad_outputs[column], Fix_output, 0);
paul@0 98
    l4_sleep(1);
paul@0 99
paul@0 100
    value = 0;
paul@0 101
paul@0 102
    for (row = 0; row < Jz4730_keypad_gpio_inputs_count; row++)
paul@0 103
      value = (value << 1) | (jz4730_gpio_get(gpio_port_a, Jz4730_keypad_inputs[row]) ? 0 : 1);
paul@0 104
paul@0 105
    keypad[column] = value;
paul@0 106
paul@0 107
    jz4730_gpio_setup(gpio_port_d, Jz4730_keypad_outputs[column], Fix_input, 0);
paul@0 108
  }
paul@0 109
}
paul@0 110
paul@0 111
static uint32_t bitmask(uint32_t size)
paul@0 112
{
paul@0 113
  return (1 << size) - 1;
paul@0 114
}
paul@0 115
paul@0 116
static uint32_t truncate_channel(uint32_t value, uint32_t size)
paul@0 117
{
paul@0 118
  return (value >> (8 - size)) & bitmask(size);
paul@0 119
}
paul@0 120
paul@0 121
static void set_pixel(uint32_t pos, uint8_t bpp, uint32_t value)
paul@0 122
{
paul@0 123
  if (bpp <= 8) *(uint8_t *) pos = value;
paul@0 124
  else if (bpp <= 16) *(uint16_t *) pos = value;
paul@0 125
  else *(uint32_t *) pos = value;
paul@0 126
}
paul@0 127
paul@0 128
/* Show the state of a key on the display. */
paul@0 129
paul@0 130
static void show_keystate(uint32_t x, uint32_t y, uint32_t w, uint32_t h, uint32_t rgb)
paul@0 131
{
paul@0 132
  uint8_t bpp = l4re_video_bits_per_pixel(&fbi.pixel_info);
paul@0 133
  uint8_t bytes_per_pixel = fbi.pixel_info.bytes_per_pixel;
paul@0 134
  uint32_t bytes_per_line = fbi.bytes_per_line;
paul@0 135
  uint32_t pos;
paul@0 136
  uint32_t col, row;
paul@0 137
paul@0 138
  rgb = (truncate_channel(rgb >> 16, fbi.pixel_info.r.size) << fbi.pixel_info.r.shift) |
paul@0 139
        (truncate_channel(rgb >> 8, fbi.pixel_info.g.size) << fbi.pixel_info.g.shift) |
paul@0 140
        (truncate_channel(rgb, fbi.pixel_info.b.size) << fbi.pixel_info.b.shift);
paul@0 141
paul@0 142
  for (row = 0; row < h; row++)
paul@0 143
  {
paul@0 144
    pos = (uint32_t) fb + (y + row) * bytes_per_line + x * bytes_per_pixel;
paul@0 145
paul@0 146
    for (col = 0; col < w; col++)
paul@0 147
    {
paul@0 148
      set_pixel(pos, bpp, rgb);
paul@0 149
      pos += bytes_per_pixel;
paul@0 150
    }
paul@0 151
  }
paul@0 152
}
paul@0 153
paul@0 154
/* Show the keypad status on the display. */
paul@0 155
paul@0 156
static void show_keypad(void)
paul@0 157
{
paul@0 158
  uint32_t colsize = fbi.width / Jz4730_keypad_gpio_outputs_count,
paul@0 159
           rowsize = fbi.height / Jz4730_keypad_gpio_inputs_count;
paul@0 160
  uint8_t column, row;
paul@0 161
  uint32_t mask;
paul@0 162
paul@0 163
  for (column = 0; column < Jz4730_keypad_gpio_outputs_count; column++)
paul@0 164
paul@0 165
    for (row = 0, mask = 1 << (Jz4730_keypad_gpio_inputs_count - 1);
paul@0 166
         row < Jz4730_keypad_gpio_inputs_count;
paul@0 167
         row++, mask >>= 1)
paul@0 168
paul@0 169
      show_keystate(column * colsize, row * rowsize, colsize, rowsize,
paul@0 170
                    keypad[column] & mask ? 0xffffff : 0);
paul@0 171
paul@0 172
  /* Refresh the display. */
paul@0 173
paul@0 174
  l4re_util_video_goos_fb_refresh(&gfb, 0, 0, fbi.width, fbi.height);
paul@0 175
}
paul@0 176
paul@0 177
int main(void)
paul@0 178
{
paul@0 179
  int result;
paul@0 180
paul@0 181
  if ((result = get_memory("jz4730-gpio", &gpio_virt_base, &gpio_virt_base_end)) < 0)
paul@0 182
    return 1;
paul@0 183
paul@0 184
  if (l4re_util_video_goos_fb_setup_name(&gfb, "fb"))
paul@0 185
    return 1;
paul@0 186
paul@0 187
  if (l4re_util_video_goos_fb_view_info(&gfb, &fbi))
paul@0 188
    return 1;
paul@0 189
paul@0 190
  if (!(fb = l4re_util_video_goos_fb_attach_buffer(&gfb)))
paul@0 191
    return 1;
paul@0 192
paul@0 193
  gpio_port_a = jz4730_gpio_init(gpio_virt_base, gpio_virt_base + 0x30, 32);
paul@0 194
  gpio_port_d = jz4730_gpio_init(gpio_virt_base + 0x90, gpio_virt_base + 0xc0, 32);
paul@0 195
paul@0 196
  init_keyscan();
paul@0 197
paul@0 198
  while (1)
paul@0 199
  {
paul@0 200
    scan_keypad();
paul@0 201
    show_keypad();
paul@0 202
  }
paul@0 203
paul@0 204
  return 0;
paul@0 205
}