Home | History | Annotate | Line # | Download | only in citrus
citrus_mmap.c revision 1.4.24.1
      1  1.4.24.1  pgoyette /*	$NetBSD: citrus_mmap.c,v 1.4.24.1 2017/03/20 06:56:57 pgoyette Exp $	*/
      2       1.1  tshiozak 
      3       1.1  tshiozak /*-
      4       1.1  tshiozak  * Copyright (c)2003 Citrus Project,
      5       1.1  tshiozak  * All rights reserved.
      6       1.1  tshiozak  *
      7       1.1  tshiozak  * Redistribution and use in source and binary forms, with or without
      8       1.1  tshiozak  * modification, are permitted provided that the following conditions
      9       1.1  tshiozak  * are met:
     10       1.1  tshiozak  * 1. Redistributions of source code must retain the above copyright
     11       1.1  tshiozak  *    notice, this list of conditions and the following disclaimer.
     12       1.1  tshiozak  * 2. Redistributions in binary form must reproduce the above copyright
     13       1.1  tshiozak  *    notice, this list of conditions and the following disclaimer in the
     14       1.1  tshiozak  *    documentation and/or other materials provided with the distribution.
     15       1.1  tshiozak  *
     16       1.1  tshiozak  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND
     17       1.1  tshiozak  * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
     18       1.1  tshiozak  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
     19       1.1  tshiozak  * ARE DISCLAIMED.  IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE
     20       1.1  tshiozak  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
     21       1.1  tshiozak  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
     22       1.1  tshiozak  * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
     23       1.1  tshiozak  * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
     24       1.1  tshiozak  * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
     25       1.1  tshiozak  * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
     26       1.1  tshiozak  * SUCH DAMAGE.
     27       1.1  tshiozak  */
     28       1.1  tshiozak 
     29       1.1  tshiozak #include <sys/cdefs.h>
     30       1.1  tshiozak #if defined(LIBC_SCCS) && !defined(lint)
     31  1.4.24.1  pgoyette __RCSID("$NetBSD: citrus_mmap.c,v 1.4.24.1 2017/03/20 06:56:57 pgoyette Exp $");
     32       1.1  tshiozak #endif /* LIBC_SCCS and not lint */
     33       1.1  tshiozak 
     34       1.1  tshiozak #include "namespace.h"
     35  1.4.24.1  pgoyette 
     36  1.4.24.1  pgoyette #include <sys/mman.h>
     37  1.4.24.1  pgoyette #include <sys/stat.h>
     38  1.4.24.1  pgoyette 
     39       1.1  tshiozak #include <assert.h>
     40       1.1  tshiozak #include <errno.h>
     41       1.1  tshiozak #include <stdio.h>
     42       1.1  tshiozak #include <stdlib.h>
     43       1.1  tshiozak #include <string.h>
     44       1.1  tshiozak #include <unistd.h>
     45       1.1  tshiozak #include <fcntl.h>
     46       1.1  tshiozak #include <limits.h>
     47       1.1  tshiozak 
     48       1.1  tshiozak #include "citrus_namespace.h"
     49       1.1  tshiozak #include "citrus_region.h"
     50       1.1  tshiozak #include "citrus_mmap.h"
     51       1.1  tshiozak 
     52       1.1  tshiozak int
     53       1.1  tshiozak _citrus_map_file(struct _citrus_region * __restrict r,
     54       1.2  christos     const char * __restrict path)
     55       1.1  tshiozak {
     56       1.2  christos 	int fd, ret = 0;
     57       1.1  tshiozak 	struct stat st;
     58       1.1  tshiozak 	void *head;
     59       1.1  tshiozak 
     60       1.1  tshiozak 	_DIAGASSERT(r != NULL);
     61       1.1  tshiozak 
     62       1.1  tshiozak 	_region_init(r, NULL, 0);
     63       1.2  christos 
     64       1.4  christos 	if ((fd = open(path, O_RDONLY | O_CLOEXEC)) == -1)
     65       1.2  christos 		return errno;
     66       1.2  christos 
     67       1.2  christos 	if (fstat(fd, &st)  == -1) {
     68       1.1  tshiozak 		ret = errno;
     69       1.2  christos 		goto error;
     70       1.1  tshiozak 	}
     71       1.2  christos 	if (!S_ISREG(st.st_mode)) {
     72       1.2  christos 		ret = EOPNOTSUPP;
     73       1.2  christos 		goto error;
     74       1.1  tshiozak 	}
     75       1.2  christos 
     76       1.1  tshiozak 	head = mmap(NULL, (size_t)st.st_size, PROT_READ, MAP_FILE|MAP_PRIVATE,
     77       1.2  christos 	    fd, (off_t)0);
     78       1.2  christos 	if (head == MAP_FAILED) {
     79       1.1  tshiozak 		ret = errno;
     80       1.2  christos 		goto error;
     81       1.1  tshiozak 	}
     82       1.1  tshiozak 	_region_init(r, head, (size_t)st.st_size);
     83       1.1  tshiozak 
     84       1.2  christos error:
     85       1.2  christos 	(void)close(fd);
     86       1.2  christos 	return ret;
     87       1.1  tshiozak }
     88       1.1  tshiozak 
     89       1.1  tshiozak void
     90       1.1  tshiozak _citrus_unmap_file(struct _citrus_region *r)
     91       1.1  tshiozak {
     92       1.1  tshiozak 
     93       1.1  tshiozak 	_DIAGASSERT(r != NULL);
     94       1.1  tshiozak 
     95       1.1  tshiozak 	if (_region_head(r) != NULL) {
     96       1.2  christos 		(void)munmap(_region_head(r), _region_size(r));
     97       1.1  tshiozak 		_region_init(r, NULL, 0);
     98       1.1  tshiozak 	}
     99       1.1  tshiozak }
    100