aboutsummaryrefslogtreecommitdiff
path: root/toys/other/devmem.c
blob: 920d85636d2ef060dab4a65e0dc8033cda8a5a13 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
/* devmem.c - Access physical addresses
 *
 * Copyright 2019 The Android Open Source Project

USE_DEVMEM(NEWTOY(devmem, "<1>3", TOYFLAG_USR|TOYFLAG_BIN))

config DEVMEM
  bool "devmem"
  default y
  help
    usage: devmem ADDR [WIDTH [DATA]]

    Read/write physical address. WIDTH is 1, 2, 4, or 8 bytes (default 4).
    Prefix ADDR with 0x for hexadecimal, output is in same base as address.
*/

#define FOR_devmem
#include "toys.h"

void devmem_main(void)
{
  int writing = toys.optc == 3, page_size = sysconf(_SC_PAGESIZE), bytes = 4,fd;
  unsigned long long data = 0, map_off, map_len;
  unsigned long addr = atolx(*toys.optargs);
  void *map, *p;

  // WIDTH?
  if (toys.optc>1) {
    int i;

    if ((i=stridx("1248", *toys.optargs[1]))==-1 || toys.optargs[1][1])
      error_exit("bad width: %s", toys.optargs[1]);
    bytes = 1<<i;
  }

  // DATA? Report out of range values as errors rather than truncating.
  if (writing) data = atolx_range(toys.optargs[2], 0, (1ULL<<(8*bytes))-1);

  // Map in just enough.
  if (CFG_TOYBOX_FORK) {
    fd = xopen("/dev/mem", (writing ? O_RDWR : O_RDONLY) | O_SYNC);

    map_off = addr & ~(page_size - 1);
    map_len = (addr+bytes-map_off);
    map = xmmap(0, map_len, writing ? PROT_WRITE : PROT_READ, MAP_SHARED, fd,
        map_off);
    p = map + (addr & (page_size - 1));
    close(fd);
  } else p = (void *)addr;

  // Not using peek()/poke() because registers care about size of read/write
  if (writing) {
    if (bytes == 1) *(unsigned char *)p = data;
    else if (bytes == 2) *(unsigned short *)p = data;
    else if (bytes == 4) *(unsigned int *)p = data;
    else if (bytes == 8) *(unsigned long long *)p = data;
  } else {
    if (bytes == 1) data = *(unsigned char *)p;
    else if (bytes == 2) data = *(unsigned short *)p;
    else if (bytes == 4) data = *(unsigned int *)p;
    else if (bytes == 8) data = *(unsigned long long *)p;
    printf((!strchr(*toys.optargs, 'x')) ? "%0*lld\n" : "0x%0*llx\n",
      bytes*2, data);
  }

  if (CFG_TOYBOX_FORK) munmap(map, map_len);
}