os_fid.c
上传用户:tsgydb
上传日期:2007-04-14
资源大小:10674k
文件大小:5k
源码类别:

MySQL数据库

开发平台:

Visual C++

  1. /*-
  2.  * See the file LICENSE for redistribution information.
  3.  *
  4.  * Copyright (c) 1996, 1997, 1998, 1999, 2000
  5.  * Sleepycat Software.  All rights reserved.
  6.  */
  7. #include "db_config.h"
  8. #ifndef lint
  9. static const char revid[] = "$Id: os_fid.c,v 11.7 2000/10/26 14:18:08 bostic Exp $";
  10. #endif /* not lint */
  11. #include "db_int.h"
  12. #define SERIAL_INIT 0
  13. static u_int32_t fid_serial = SERIAL_INIT;
  14. /*
  15.  * __os_fileid --
  16.  * Return a unique identifier for a file.
  17.  */
  18. int
  19. __os_fileid(dbenv, fname, unique_okay, fidp)
  20. DB_ENV *dbenv;
  21. const char *fname;
  22. int unique_okay;
  23. u_int8_t *fidp;
  24. {
  25. size_t i;
  26. u_int32_t tmp;
  27. u_int8_t *p;
  28. int ret;
  29. /*
  30.  * The documentation for GetFileInformationByHandle() states that the
  31.  * inode-type numbers are not constant between processes.  Actually,
  32.  * they are, they're the NTFS MFT indexes.  So, this works on NTFS,
  33.  * but perhaps not on other platforms, and perhaps not over a network.
  34.  * Can't think of a better solution right now.
  35.  */
  36. DB_FH fh;
  37. HANDLE handle;
  38. BY_HANDLE_FILE_INFORMATION fi;
  39. BOOL retval = FALSE;
  40. /* Clear the buffer. */
  41. memset(fidp, 0, DB_FILE_ID_LEN);
  42. /*
  43.  * Initialize/increment the serial number we use to help avoid
  44.  * fileid collisions.  Note that we don't bother with locking;
  45.  * it's unpleasant to do from down in here, and if we race on
  46.  * this no real harm will be done, since the finished fileid
  47.  * has so many other components.
  48.  *
  49.  * We increment by 100000 on each call as a simple way of
  50.  * randomizing;  simply incrementing seems potentially less useful
  51.  * if pids are also simply incremented, since this is process-local
  52.  * and we may be one of a set of processes starting up.  100000
  53.  * pushes us out of pid space on most platforms, and has few
  54.  * interesting properties in base 2.
  55.  */
  56. if (fid_serial == SERIAL_INIT)
  57. fid_serial = (u_int32_t)getpid();
  58. else
  59. fid_serial += 100000;
  60. /*
  61.  * First we open the file, because we're not given a handle to it.
  62.  * If we can't open it, we're in trouble.
  63.  */
  64. if ((ret = __os_open(dbenv, fname, DB_OSO_RDONLY, _S_IREAD, &fh)) != 0)
  65. return (ret);
  66. /* File open, get its info */
  67. handle = (HANDLE)_get_osfhandle(fh.fd);
  68. if (handle == INVALID_HANDLE_VALUE)
  69. ret = __os_win32_errno();
  70. else
  71. if ((retval = GetFileInformationByHandle(handle, &fi)) == FALSE)
  72. ret = __os_win32_errno();
  73. __os_closehandle(&fh);
  74. if (handle == INVALID_HANDLE_VALUE || retval == FALSE)
  75. return (ret);
  76. /*
  77.  * We want the three 32-bit words which tell us the volume ID and
  78.  * the file ID.  We make a crude attempt to copy the bytes over to
  79.  * the callers buffer.
  80.  *
  81.  * We don't worry about byte sexing or the actual variable sizes.
  82.  *
  83.  * When this routine is called from the DB access methods, it's only
  84.  * called once -- whatever ID is generated when a database is created
  85.  * is stored in the database file's metadata, and that is what is
  86.  * saved in the mpool region's information to uniquely identify the
  87.  * file.
  88.  *
  89.  * When called from the mpool layer this routine will be called each
  90.  * time a new thread of control wants to share the file, which makes
  91.  * things tougher.  As far as byte sexing goes, since the mpool region
  92.  * lives on a single host, there's no issue of that -- the entire
  93.  * region is byte sex dependent.  As far as variable sizes go, we make
  94.  * the simplifying assumption that 32-bit and 64-bit processes will
  95.  * get the same 32-bit values if we truncate any returned 64-bit value
  96.  * to a 32-bit value.
  97.  */
  98. tmp = (u_int32_t)fi.nFileIndexLow;
  99. for (p = (u_int8_t *)&tmp, i = sizeof(u_int32_t); i > 0; --i)
  100. *fidp++ = *p++;
  101. tmp = (u_int32_t)fi.nFileIndexHigh;
  102. for (p = (u_int8_t *)&tmp, i = sizeof(u_int32_t); i > 0; --i)
  103. *fidp++ = *p++;
  104. if (unique_okay) {
  105. /*
  106.  * Use the system time to try to get a unique value
  107.  * within this process.  A millisecond counter
  108.  * overflows 32 bits in about 49 days.  So we use 8
  109.  * bytes, and don't bother with the volume ID, which
  110.  * is not very useful for our purposes.
  111.  */
  112. SYSTEMTIME st;
  113. GetSystemTime(&st);
  114. tmp = (st.wYear - 1900) * 12 + (st.wMonth - 1);
  115. for (p = (u_int8_t *)&tmp, i = sizeof(u_int32_t); i > 0; --i)
  116. *fidp++ = *p++;
  117. tmp = ((((st.wDay - 1) * 24 + st.wHour) * 60 +
  118. st.wMinute) * 60 + st.wSecond) * 1000 +
  119. st.wMilliseconds;
  120. for (p = (u_int8_t *)&tmp, i = sizeof(u_int32_t); i > 0; --i)
  121. *fidp++ = *p++;
  122. for (p = (u_int8_t *)&fid_serial, i = sizeof(u_int32_t);
  123.     i > 0; --i)
  124. *fidp++ = *p++;
  125. } else {
  126. tmp = (u_int32_t)fi.dwVolumeSerialNumber;
  127. for (p = (u_int8_t *)&tmp, i = sizeof(u_int32_t); i > 0; --i)
  128. *fidp++ = *p++;
  129. }
  130. return (0);
  131. }