]> git.lyx.org Git - lyx.git/blob - src/support/lyxsum.cpp
a8ff0ec742ec9567adcbbfd44c41c67dd7bbeb48
[lyx.git] / src / support / lyxsum.cpp
1 /**
2  * \file lyxsum.cpp
3  * This file is part of LyX, the document processor.
4  * Licence details can be found in the file COPYING.
5  *
6  * \author Lars Gullik Bjønnes
7  *
8  * Full author contact details are available in file CREDITS.
9  */
10
11 #include <config.h>
12
13 #include "support/lyxlib.h"
14 #include "support/debug.h"
15 #include "support/FileName.h"
16
17 #include <boost/crc.hpp>
18
19 #include <algorithm>
20 #include <iomanip>
21
22 using namespace std;
23
24 // OK, this is ugly, but it is the only workaround I found to compile
25 // with gcc (any version) on a system which uses a non-GNU toolchain.
26 // The problem is that gcc uses a weak symbol for a particular
27 // instantiation and that the system linker usually does not
28 // understand those weak symbols (seen on HP-UX, tru64, AIX and
29 // others). Thus we force an explicit instanciation of this particular
30 // template (JMarc)
31 template struct boost::detail::crc_table_t<32, 0x04C11DB7, true>;
32
33 // Various implementations of lyx::sum(), depending on what methods
34 // are available. Order is faster to slowest.
35 #if defined(HAVE_MMAP) && defined(HAVE_MUNMAP)
36
37 #ifdef HAVE_SYS_TYPES_H
38 # include <sys/types.h>
39 #endif
40 #ifdef HAVE_SYS_STAT_H
41 # include <sys/stat.h>
42 #endif
43 #include <fcntl.h>
44 #ifdef HAVE_UNISTD_H
45 # include <unistd.h>
46 #endif
47
48 #include <sys/mman.h>
49
50
51 namespace lyx {
52 namespace support {
53
54 unsigned long sum(char const * file)
55 {
56         //LYXERR(Debug::FILES, "lyx::sum() using mmap (lightning fast)");
57
58         int fd = open(file, O_RDONLY);
59         if (!fd)
60                 return 0;
61
62         struct stat info;
63         fstat(fd, &info);
64
65         void * mm = mmap(0, info.st_size, PROT_READ,
66                          MAP_PRIVATE, fd, 0);
67         // Some platforms have the wrong type for MAP_FAILED (compaq cxx).
68         if (mm == reinterpret_cast<void*>(MAP_FAILED)) {
69                 close(fd);
70                 return 0;
71         }
72
73         char * beg = static_cast<char*>(mm);
74         char * end = beg + info.st_size;
75
76         boost::crc_32_type crc;
77         crc.process_block(beg, end);
78         unsigned long result = crc.checksum();
79
80         munmap(mm, info.st_size);
81         close(fd);
82
83         return result;
84 }
85
86 } // namespace support
87 } // namespace lyx
88
89 #else // No mmap
90
91 #include <fstream>
92 #include <iterator>
93
94 namespace {
95
96 template<typename InputIterator>
97 inline
98 unsigned long do_crc(InputIterator first, InputIterator last)
99 {
100         boost::crc_32_type crc;
101         crc = for_each(first, last, crc);
102         return crc.checksum();
103 }
104
105 } // namespace anon
106
107
108 namespace lyx {
109 namespace support {
110
111
112 unsigned long sum(char const * file)
113 {
114         ifstream ifs(file, ios_base::in | ios_base::binary);
115         if (!ifs)
116                 return 0;
117
118 #if HAVE_DECL_ISTREAMBUF_ITERATOR
119         //LYXERR(Debug::FILES, "lyx::sum() using istreambuf_iterator (fast)");
120         istreambuf_iterator<char> beg(ifs);
121         istreambuf_iterator<char> end;
122 #else
123         //LYXERR(Debug::FILES, "lyx::sum() using istream_iterator (slow as a snail)");
124         ifs.unsetf(ios::skipws);
125         istream_iterator<char> beg(ifs);
126         istream_iterator<char> end;
127 #endif
128
129         return do_crc(beg,end);
130 }
131
132 } // namespace support
133 } // namespace lyx
134
135 #endif // mmap