aboutsummaryrefslogtreecommitdiff
path: root/riscv/devices.h
blob: 83781884b9910aa7fd8934efee25b538194a85ae (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
#ifndef _RISCV_DEVICES_H
#define _RISCV_DEVICES_H

#include "decode.h"
#include <map>
#include <vector>

class processor_t;

class abstract_device_t {
 public:
  virtual bool load(reg_t addr, size_t len, uint8_t* bytes) = 0;
  virtual bool store(reg_t addr, size_t len, const uint8_t* bytes) = 0;
  // Return a pointer to the start of the page that addr falls in, or NULL if
  // there is no IO device at that address.
  virtual char* page(reg_t addr) { return NULL; }
  virtual ~abstract_device_t() {}
};

class bus_t : public abstract_device_t {
 public:
  bool load(reg_t addr, size_t len, uint8_t* bytes);
  bool store(reg_t addr, size_t len, const uint8_t* bytes);
  // Return a pointer to the start of the page that addr falls in, or NULL if
  // there is no IO device at that address.
  char* page(reg_t paddr);
  void add_device(reg_t addr, abstract_device_t* dev);

 private:
  std::map<reg_t, abstract_device_t*> devices;
};

class rom_device_t : public abstract_device_t {
 public:
  rom_device_t(std::vector<char> data);
  bool load(reg_t addr, size_t len, uint8_t* bytes);
  bool store(reg_t addr, size_t len, const uint8_t* bytes);
  const std::vector<char>& contents() { return data; }
 private:
  std::vector<char> data;
};

class rtc_t : public abstract_device_t {
 public:
  rtc_t(std::vector<processor_t*>&);
  bool load(reg_t addr, size_t len, uint8_t* bytes);
  bool store(reg_t addr, size_t len, const uint8_t* bytes);
  size_t size() { return regs.size() * sizeof(regs[0]); }
  void increment(reg_t inc);
 private:
  std::vector<processor_t*>& procs;
  std::vector<uint64_t> regs;
  uint64_t time() { return regs[0]; }
};

#endif