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
|
/*
* OpenVPN -- An application to securely tunnel IP networks
* over a single TCP/UDP port, with support for SSL/TLS-based
* session authentication and key exchange,
* packet encryption, packet authentication, and
* packet compression.
*
* Copyright (C) 2002-2011 OpenVPN Technologies, Inc. <sales@openvpn.net>
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License version 2
* as published by the Free Software Foundation.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program (see the file COPYING included with this
* distribution); if not, write to the Free Software Foundation, Inc.,
* 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
*/
/*
* Maintain usage stats in a memory-mapped file
*/
#include "syshead.h"
#if defined(ENABLE_MEMSTATS)
#include <sys/mman.h>
#include "error.h"
#include "misc.h"
#include "mstats.h"
#include "memdbg.h"
volatile struct mmap_stats *mmap_stats = NULL; /* GLOBAL */
static char mmap_fn[128];
void
mstats_open(const char *fn)
{
void *data;
ssize_t stat;
int fd;
struct mmap_stats ms;
if (mmap_stats) /* already called? */
return;
/* verify that filename is not too long */
if (strlen(fn) >= sizeof(mmap_fn))
msg (M_FATAL, "mstats_open: filename too long");
/* create file that will be memory mapped */
fd = open (fn, O_CREAT | O_TRUNC | O_RDWR, S_IRUSR | S_IWUSR);
if (fd < 0)
{
msg (M_ERR, "mstats_open: cannot open: %s", fn);
return;
}
/* set the file to the correct size to contain a
struct mmap_stats, and zero it */
CLEAR(ms);
ms.state = MSTATS_ACTIVE;
stat = write(fd, &ms, sizeof(ms));
if (stat != sizeof(ms))
{
msg (M_ERR, "mstats_open: write error: %s", fn);
close(fd);
return;
}
/* mmap the file */
data = mmap(NULL, sizeof(struct mmap_stats), PROT_READ|PROT_WRITE, MAP_SHARED, fd, 0);
if (data == MAP_FAILED)
{
msg (M_ERR, "mstats_open: write error: %s", fn);
close(fd);
return;
}
/* close the fd (mmap now controls the file) */
if (close(fd))
{
msg (M_ERR, "mstats_open: close error: %s", fn);
}
/* save filename so we can delete it later */
strcpy(mmap_fn, fn);
/* save a global pointer to memory-mapped region */
mmap_stats = (struct mmap_stats *)data;
msg (M_INFO, "memstats data will be written to %s", fn);
}
void
mstats_close(void)
{
if (mmap_stats)
{
mmap_stats->state = MSTATS_EXPIRED;
if (munmap((void *)mmap_stats, sizeof(struct mmap_stats)))
msg (M_WARN | M_ERRNO, "mstats_close: munmap error");
delete_file(mmap_fn);
mmap_stats = NULL;
}
}
#endif
|