Annotation of embedaddon/ntp/sntp/libopts/text_mmap.c, revision 1.1

1.1     ! misho       1: /**
        !             2:  * \file text_mmap.c
        !             3:  *
        !             4:  * Time-stamp:      "2010-07-17 10:15:32 bkorb"
        !             5:  *
        !             6:  *  This file is part of AutoOpts, a companion to AutoGen.
        !             7:  *  AutoOpts is free software.
        !             8:  *  AutoOpts is Copyright (c) 1992-2011 by Bruce Korb - all rights reserved
        !             9:  *
        !            10:  *  AutoOpts is available under any one of two licenses.  The license
        !            11:  *  in use must be one of these two and the choice is under the control
        !            12:  *  of the user of the license.
        !            13:  *
        !            14:  *   The GNU Lesser General Public License, version 3 or later
        !            15:  *      See the files "COPYING.lgplv3" and "COPYING.gplv3"
        !            16:  *
        !            17:  *   The Modified Berkeley Software Distribution License
        !            18:  *      See the file "COPYING.mbsd"
        !            19:  *
        !            20:  *  These files have the following md5sums:
        !            21:  *
        !            22:  *  43b91e8ca915626ed3818ffb1b71248b pkg/libopts/COPYING.gplv3
        !            23:  *  06a1a2e4760c90ea5e1dad8dfaac4d39 pkg/libopts/COPYING.lgplv3
        !            24:  *  66a5cedaf62c4b2637025f049f9b826f pkg/libopts/COPYING.mbsd
        !            25:  */
        !            26: 
        !            27: #ifndef MAP_ANONYMOUS
        !            28: #  ifdef   MAP_ANON
        !            29: #  define  MAP_ANONYMOUS   MAP_ANON
        !            30: #  endif
        !            31: #endif
        !            32: 
        !            33: /*
        !            34:  *  Some weird systems require that a specifically invalid FD number
        !            35:  *  get passed in as an argument value.  Which value is that?  Well,
        !            36:  *  as everybody knows, if open(2) fails, it returns -1, so that must
        !            37:  *  be the value.  :)
        !            38:  */
        !            39: #define AO_INVALID_FD  -1
        !            40: 
        !            41: #define FILE_WRITABLE(_prt,_flg) \
        !            42:         (   (_prt & PROT_WRITE) \
        !            43:          && ((_flg & (MAP_SHARED|MAP_PRIVATE)) == MAP_SHARED))
        !            44: #define MAP_FAILED_PTR ((void*)MAP_FAILED)
        !            45: 
        !            46: /*=export_func  text_mmap
        !            47:  * private:
        !            48:  *
        !            49:  * what:  map a text file with terminating NUL
        !            50:  *
        !            51:  * arg:   char const*,  pzFile,  name of the file to map
        !            52:  * arg:   int,          prot,    mmap protections (see mmap(2))
        !            53:  * arg:   int,          flags,   mmap flags (see mmap(2))
        !            54:  * arg:   tmap_info_t*, mapinfo, returned info about the mapping
        !            55:  *
        !            56:  * ret-type:   void*
        !            57:  * ret-desc:   The mmaped data address
        !            58:  *
        !            59:  * doc:
        !            60:  *
        !            61:  * This routine will mmap a file into memory ensuring that there is at least
        !            62:  * one @file{NUL} character following the file data.  It will return the
        !            63:  * address where the file contents have been mapped into memory.  If there is a
        !            64:  * problem, then it will return @code{MAP_FAILED} and set @file{errno}
        !            65:  * appropriately.
        !            66:  *
        !            67:  * The named file does not exist, @code{stat(2)} will set @file{errno} as it
        !            68:  * will.  If the file is not a regular file, @file{errno} will be
        !            69:  * @code{EINVAL}.  At that point, @code{open(2)} is attempted with the access
        !            70:  * bits set appropriately for the requested @code{mmap(2)} protections and flag
        !            71:  * bits.  On failure, @file{errno} will be set according to the documentation
        !            72:  * for @code{open(2)}.  If @code{mmap(2)} fails, @file{errno} will be set as
        !            73:  * that routine sets it.  If @code{text_mmap} works to this point, a valid
        !            74:  * address will be returned, but there may still be ``issues''.
        !            75:  *
        !            76:  * If the file size is not an even multiple of the system page size, then
        !            77:  * @code{text_map} will return at this point and @file{errno} will be zero.
        !            78:  * Otherwise, an anonymous map is attempted.  If not available, then an attempt
        !            79:  * is made to @code{mmap(2)} @file{/dev/zero}.  If any of these fail, the
        !            80:  * address of the file's data is returned, bug @code{no} @file{NUL} characters
        !            81:  * are mapped after the end of the data.
        !            82:  *
        !            83:  * see: mmap(2), open(2), stat(2)
        !            84:  *
        !            85:  * err: Any error code issued by mmap(2), open(2), stat(2) is possible.
        !            86:  *      Additionally, if the specified file is not a regular file, then
        !            87:  *      errno will be set to @code{EINVAL}.
        !            88:  *
        !            89:  * example:
        !            90:  * #include <mylib.h>
        !            91:  * tmap_info_t mi;
        !            92:  * int no_nul;
        !            93:  * void* data = text_mmap("file", PROT_WRITE, MAP_PRIVATE, &mi);
        !            94:  * if (data == MAP_FAILED) return;
        !            95:  * no_nul = (mi.txt_size == mi.txt_full_size);
        !            96:  * << use the data >>
        !            97:  * text_munmap(&mi);
        !            98: =*/
        !            99: void*
        !           100: text_mmap(char const* pzFile, int prot, int flags, tmap_info_t* pMI)
        !           101: {
        !           102:     memset(pMI, 0, sizeof(*pMI));
        !           103: #ifdef HAVE_MMAP
        !           104:     pMI->txt_zero_fd = -1;
        !           105: #endif
        !           106:     pMI->txt_fd = -1;
        !           107: 
        !           108:     /*
        !           109:      *  Make sure we can stat the regular file.  Save the file size.
        !           110:      */
        !           111:     {
        !           112:         struct stat sb;
        !           113:         if (stat(pzFile, &sb) != 0) {
        !           114:             pMI->txt_errno = errno;
        !           115:             return MAP_FAILED_PTR;
        !           116:         }
        !           117: 
        !           118:         if (! S_ISREG(sb.st_mode)) {
        !           119:             pMI->txt_errno = errno = EINVAL;
        !           120:             return MAP_FAILED_PTR;
        !           121:         }
        !           122: 
        !           123:         pMI->txt_size = sb.st_size;
        !           124:     }
        !           125: 
        !           126:     /*
        !           127:      *  Map mmap flags and protections into open flags and do the open.
        !           128:      */
        !           129:     {
        !           130:         int o_flag;
        !           131:         /*
        !           132:          *  See if we will be updating the file.  If we can alter the memory
        !           133:          *  and if we share the data and we are *not* copy-on-writing the data,
        !           134:          *  then our updates will show in the file, so we must open with
        !           135:          *  write access.
        !           136:          */
        !           137:         if (FILE_WRITABLE(prot,flags))
        !           138:             o_flag = O_RDWR;
        !           139:         else
        !           140:             o_flag = O_RDONLY;
        !           141: 
        !           142:         /*
        !           143:          *  If you're not sharing the file and you are writing to it,
        !           144:          *  then don't let anyone else have access to the file.
        !           145:          */
        !           146:         if (((flags & MAP_SHARED) == 0) && (prot & PROT_WRITE))
        !           147:             o_flag |= O_EXCL;
        !           148: 
        !           149:         pMI->txt_fd = open(pzFile, o_flag);
        !           150:     }
        !           151: 
        !           152:     if (pMI->txt_fd == AO_INVALID_FD) {
        !           153:         pMI->txt_errno = errno;
        !           154:         return MAP_FAILED_PTR;
        !           155:     }
        !           156: 
        !           157: #ifdef HAVE_MMAP /* * * * * WITH MMAP * * * * * */
        !           158:     /*
        !           159:      *  do the mmap.  If we fail, then preserve errno, close the file and
        !           160:      *  return the failure.
        !           161:      */
        !           162:     pMI->txt_data =
        !           163:         mmap(NULL, pMI->txt_size+1, prot, flags, pMI->txt_fd, (size_t)0);
        !           164:     if (pMI->txt_data == MAP_FAILED_PTR) {
        !           165:         pMI->txt_errno = errno;
        !           166:         goto fail_return;
        !           167:     }
        !           168: 
        !           169:     /*
        !           170:      *  Most likely, everything will turn out fine now.  The only difficult
        !           171:      *  part at this point is coping with files with sizes that are a multiple
        !           172:      *  of the page size.  Handling that is what this whole thing is about.
        !           173:      */
        !           174:     pMI->txt_zero_fd = -1;
        !           175:     pMI->txt_errno   = 0;
        !           176: 
        !           177:     {
        !           178:         void* pNuls;
        !           179: #ifdef _SC_PAGESIZE
        !           180:         size_t pgsz = sysconf(_SC_PAGESIZE);
        !           181: #else
        !           182:         size_t pgsz = getpagesize();
        !           183: #endif
        !           184:         /*
        !           185:          *  Compute the pagesize rounded mapped memory size.
        !           186:          *  IF this is not the same as the file size, then there are NUL's
        !           187:          *  at the end of the file mapping and all is okay.
        !           188:          */
        !           189:         pMI->txt_full_size = (pMI->txt_size + (pgsz - 1)) & ~(pgsz - 1);
        !           190:         if (pMI->txt_size != pMI->txt_full_size)
        !           191:             return pMI->txt_data;
        !           192: 
        !           193:         /*
        !           194:          *  Still here?  We have to remap the trailing inaccessible page
        !           195:          *  either anonymously or to /dev/zero.
        !           196:          */
        !           197:         pMI->txt_full_size += pgsz;
        !           198: #if defined(MAP_ANONYMOUS)
        !           199:         pNuls = mmap(
        !           200:                 (void*)(((char*)pMI->txt_data) + pMI->txt_size),
        !           201:                 pgsz, PROT_READ|PROT_WRITE,
        !           202:                 MAP_ANONYMOUS|MAP_FIXED|MAP_PRIVATE, AO_INVALID_FD, (size_t)0);
        !           203: 
        !           204:         if (pNuls != MAP_FAILED_PTR)
        !           205:             return pMI->txt_data;
        !           206: 
        !           207:         pMI->txt_errno = errno;
        !           208: 
        !           209: #elif defined(HAVE_DEV_ZERO)
        !           210:         pMI->txt_zero_fd = open("/dev/zero", O_RDONLY);
        !           211: 
        !           212:         if (pMI->txt_zero_fd == AO_INVALID_FD) {
        !           213:             pMI->txt_errno = errno;
        !           214: 
        !           215:         } else {
        !           216:             pNuls = mmap(
        !           217:                     (void*)(((char*)pMI->txt_data) + pMI->txt_size), pgsz,
        !           218:                     PROT_READ|PROT_WRITE, MAP_PRIVATE|MAP_FIXED,
        !           219:                     pMI->txt_zero_fd, 0 );
        !           220: 
        !           221:             if (pNuls != MAP_FAILED_PTR)
        !           222:                 return pMI->txt_data;
        !           223: 
        !           224:             pMI->txt_errno = errno;
        !           225:             close(pMI->txt_zero_fd);
        !           226:             pMI->txt_zero_fd = -1;
        !           227:         }
        !           228: #endif
        !           229: 
        !           230:         pMI->txt_full_size = pMI->txt_size;
        !           231:     }
        !           232: 
        !           233:     {
        !           234:         void* p = AGALOC(pMI->txt_size+1, "file text");
        !           235:         memcpy(p, pMI->txt_data, pMI->txt_size);
        !           236:         ((char*)p)[pMI->txt_size] = NUL;
        !           237:         munmap(pMI->txt_data, pMI->txt_size );
        !           238:         pMI->txt_data = p;
        !           239:     }
        !           240:     pMI->txt_alloc = 1;
        !           241:     return pMI->txt_data;
        !           242: 
        !           243: #else /* * * * * * no HAVE_MMAP * * * * * */
        !           244: 
        !           245:     pMI->txt_data = AGALOC(pMI->txt_size+1, "file text");
        !           246:     if (pMI->txt_data == NULL) {
        !           247:         pMI->txt_errno = ENOMEM;
        !           248:         goto fail_return;
        !           249:     }
        !           250: 
        !           251:     {
        !           252:         size_t sz = pMI->txt_size;
        !           253:         char*  pz = pMI->txt_data;
        !           254: 
        !           255:         while (sz > 0) {
        !           256:             ssize_t rdct = read(pMI->txt_fd, pz, sz);
        !           257:             if (rdct <= 0) {
        !           258:                 pMI->txt_errno = errno;
        !           259:                 fprintf(stderr, zFSErrReadFile,
        !           260:                         errno, strerror(errno), pzFile);
        !           261:                 free(pMI->txt_data);
        !           262:                 goto fail_return;
        !           263:             }
        !           264: 
        !           265:             pz += rdct;
        !           266:             sz -= rdct;
        !           267:         }
        !           268: 
        !           269:         *pz = NUL;
        !           270:     }
        !           271: 
        !           272:     /*
        !           273:      *  We never need a dummy page mapped in
        !           274:      */
        !           275:     pMI->txt_zero_fd = -1;
        !           276:     pMI->txt_errno   = 0;
        !           277: 
        !           278:     return pMI->txt_data;
        !           279: 
        !           280: #endif /* * * * * * no HAVE_MMAP * * * * * */
        !           281: 
        !           282:  fail_return:
        !           283:     if (pMI->txt_fd >= 0) {
        !           284:         close(pMI->txt_fd);
        !           285:         pMI->txt_fd = -1;
        !           286:     }
        !           287:     errno = pMI->txt_errno;
        !           288:     pMI->txt_data = MAP_FAILED_PTR;
        !           289:     return pMI->txt_data;
        !           290: }
        !           291: 
        !           292: 
        !           293: /*=export_func  text_munmap
        !           294:  * private:
        !           295:  *
        !           296:  * what:  unmap the data mapped in by text_mmap
        !           297:  *
        !           298:  * arg:   tmap_info_t*, mapinfo, info about the mapping
        !           299:  *
        !           300:  * ret-type:   int
        !           301:  * ret-desc:   -1 or 0.  @file{errno} will have the error code.
        !           302:  *
        !           303:  * doc:
        !           304:  *
        !           305:  * This routine will unmap the data mapped in with @code{text_mmap} and close
        !           306:  * the associated file descriptors opened by that function.
        !           307:  *
        !           308:  * see: munmap(2), close(2)
        !           309:  *
        !           310:  * err: Any error code issued by munmap(2) or close(2) is possible.
        !           311: =*/
        !           312: int
        !           313: text_munmap(tmap_info_t* pMI)
        !           314: {
        !           315: #ifdef HAVE_MMAP
        !           316:     int res = 0;
        !           317:     if (pMI->txt_alloc) {
        !           318:         /*
        !           319:          *  IF the user has write permission and the text is not mapped private,
        !           320:          *  then write back any changes.  Hopefully, nobody else has modified
        !           321:          *  the file in the mean time.
        !           322:          */
        !           323:         if (   ((pMI->txt_prot & PROT_WRITE) != 0)
        !           324:             && ((pMI->txt_flags & MAP_PRIVATE) == 0))  {
        !           325: 
        !           326:             if (lseek(pMI->txt_fd, (size_t)0, SEEK_SET) != 0)
        !           327:                 goto error_return;
        !           328: 
        !           329:             res = (write(pMI->txt_fd, pMI->txt_data, pMI->txt_size) < 0)
        !           330:                 ? errno : 0;
        !           331:         }
        !           332: 
        !           333:         AGFREE(pMI->txt_data);
        !           334:         errno = res;
        !           335:     } else {
        !           336:         res = munmap(pMI->txt_data, pMI->txt_full_size);
        !           337:     }
        !           338:     if (res != 0)
        !           339:         goto error_return;
        !           340: 
        !           341:     res = close(pMI->txt_fd);
        !           342:     if (res != 0)
        !           343:         goto error_return;
        !           344: 
        !           345:     pMI->txt_fd = -1;
        !           346:     errno = 0;
        !           347:     if (pMI->txt_zero_fd != -1) {
        !           348:         res = close(pMI->txt_zero_fd);
        !           349:         pMI->txt_zero_fd = -1;
        !           350:     }
        !           351: 
        !           352:  error_return:
        !           353:     pMI->txt_errno = errno;
        !           354:     return res;
        !           355: #else  /* HAVE_MMAP */
        !           356: 
        !           357:     errno = 0;
        !           358:     /*
        !           359:      *  IF the memory is writable *AND* it is not private (copy-on-write)
        !           360:      *     *AND* the memory is "sharable" (seen by other processes)
        !           361:      *  THEN rewrite the data.
        !           362:      */
        !           363:     if (   FILE_WRITABLE(pMI->txt_prot, pMI->txt_flags)
        !           364:         && (lseek(pMI->txt_fd, 0, SEEK_SET) >= 0) ) {
        !           365:         write(pMI->txt_fd, pMI->txt_data, pMI->txt_size);
        !           366:     }
        !           367: 
        !           368:     close(pMI->txt_fd);
        !           369:     pMI->txt_fd = -1;
        !           370:     pMI->txt_errno = errno;
        !           371:     free(pMI->txt_data);
        !           372: 
        !           373:     return pMI->txt_errno;
        !           374: #endif /* HAVE_MMAP */
        !           375: }
        !           376: 
        !           377: /*
        !           378:  * Local Variables:
        !           379:  * mode: C
        !           380:  * c-file-style: "stroustrup"
        !           381:  * indent-tabs-mode: nil
        !           382:  * End:
        !           383:  * end of autoopts/text_mmap.c */

FreeBSD-CVSweb <freebsd-cvsweb@FreeBSD.org>