summaryrefslogtreecommitdiffstats
path: root/vm/mmap.c
blob: 9e7ae471af19c41fe07ea4c8e7587e4e9a987fb1 (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
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
#include <string.h>
#include "filesys/file.h"
#include "threads/thread.h"
#include "threads/vaddr.h"
#include "threads/malloc.h"
#include "threads/palloc.h"
#include "vm/mmap.h"
#include "vm/page.h"

static struct mmap_table_entry *mmap_table_get (struct mmap_table *table,
    mapid_t mapid);

/* initialize memory mapped files table */
bool
mmap_table_init (struct mmap_table *table)
{
  table->ids = palloc_get_page (PAL_ZERO);
  if (table->ids == NULL)
    return false;
  table->id_cap  = PGSIZE / sizeof (table->ids[0]);
  table->id_free = 0;
  table->id_max  = -1;
  return true;
}

/* frees the content of mmap table */
void
mmap_table_free (struct mmap_table *table)
{
  //TODO
  palloc_free_page (table->ids);
}

/* inserts a new mmap entry in the mmap table
   returns -1 if entry is invalid or already in the table */
mapid_t
mmap_table_insert (struct mmap_table *table, void *addr, struct file *file,
    off_t len)
{
  off_t ofs = 0;
  size_t page_read_bytes;
  struct mmap_table_entry *mme;
  mapid_t mapid;

  /* no more entries left */
  if (table->id_free >= table->id_cap)
    return -1;

  mme = malloc (sizeof *mme);
  if (mme == NULL)
    return -1;

  mapid = table->id_free++;
  mme->addr  = addr;
  mme->file  = file;
  table->ids[mapid] = mme;

  /* create needed pages in page table */
  while (len > 0)
    {
      page_read_bytes = (len < PGSIZE) ? len : PGSIZE;

      if (!page_table_insert_segment (&thread_current ()->page_table, file, ofs,
            addr, page_read_bytes, true))
        return -1;

      /* Advance. */
      len  -= PGSIZE;
      ofs  += page_read_bytes;
      addr += PGSIZE;
    }

  /* update index of free/max mapid index */
  if (mapid > table->id_max)
    table->id_max = mapid;
  while (table->ids[table->id_free] != NULL)
  {
    table->id_free++;
    if (table->id_free >= table->id_cap)
      break;
  }

  return mapid;
}

static struct mmap_table_entry *
mmap_table_get (struct mmap_table *table, mapid_t mapid)
{
  if (mapid < 0 || mapid >= table->id_cap || !table->ids[mapid])
    return NULL;
  return table->ids[mapid];
}

bool
mmap_table_remove (struct mmap_table *table, mapid_t mapid)
{
  struct mmap_table_entry *mme = mmap_table_get(table, mapid);
  if (mme == NULL)
    return false;

  table->ids[mapid] = NULL;

  //TODO

  /* update index of free/max file descriptor index */
  if (mapid < table->id_free)
    table->id_free = mapid;
  while (table->ids[table->id_max] == NULL)
  {
    table->id_max--;
    if (table->id_max < 0)
      break;
  }

  return true;
}