File: main.c

package info (click to toggle)
libexplain 1.4.D001-8
  • links: PTS, VCS
  • area: main
  • in suites: buster
  • size: 39,324 kB
  • sloc: ansic: 156,027; makefile: 47,893; sh: 16,303; yacc: 1,894; awk: 245
file content (103 lines) | stat: -rw-r--r-- 2,951 bytes parent folder | download | duplicates (5)
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
/*
 * libexplain - Explain errno values returned by libc functions
 * Copyright (C) 2009, 2012 Peter Miller
 * Written by Peter Miller <pmiller@opensource.org.au>
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 3 of the License, or (at
 * your option) any later version.
 *
 * 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. If not, see <http://www.gnu.org/licenses/>.
 */

#include <libexplain/ac/stdlib.h>
#include <libexplain/ac/string.h>
#include <libexplain/ac/sys/mman.h>
#include <libexplain/ac/unistd.h>


static int
mincore_says_is_in_core(void *data, size_t data_size)
{
    size_t          page_size;
    size_t          lo_pages;
    size_t          hi_pages;
    void            *lo;
    void            *hi;
    size_t          bsize;
    size_t          vec_bytes;
    unsigned char   vec[100];

    page_size = getpagesize();

    lo_pages = (size_t)data / page_size;
    hi_pages = ((size_t)data + data_size + page_size - 1) / page_size;
    lo = (void *)(lo_pages * page_size);
    hi = (void *)(hi_pages * page_size);
    bsize = (char *)hi - (char *)lo;
    vec_bytes = hi_pages - lo_pages;
    if (vec_bytes > sizeof(vec))
        return 0;
    return (mincore(lo, bsize, (void *)vec) >= 0);
}


static int
mincore_detects_stack_ok(void)
{
    char            dummy[1000];

    /* set memory to force copy-on-write, of necessary */
    memset(dummy, 0xAA, sizeof(dummy));
    return mincore_says_is_in_core(dummy, sizeof(dummy));
}


static int
mincore_detects_heap_ok(void)
{
    size_t          dummy_size;
    char            *dummy;
    int             result;

    dummy_size = 1000;
    dummy = malloc(dummy_size);
    /* set memory to force copy-on-write, of necessary */
    memset(dummy, 0xAA, dummy_size);
    result = mincore_says_is_in_core(dummy, dummy_size);
    free(dummy);
    return result;
}


int
main(int argc, char **argv)
{
    int             ok;

    (void)argc;
    (void)argv;

    /*
     * The mincore function is highly variable in implementation between
     * Unix implementations, and even from one release to another of a
     * single brand of Unix.  So we have to check.
     *
     *    "[The mincore function] is probably still more os-specific and, in
     *    fact, architecture specific, than you think.  This deep in the VM live
     *    daemons." -- Daniel Pittman <daniel@rimspace.net>
     */
    ok = mincore_detects_stack_ok() && mincore_detects_heap_ok();

    return (ok ? EXIT_SUCCESS : EXIT_FAILURE);
}


/* vim: set ts=8 sw=4 et : */