]> git.lyx.org Git - lyx.git/blob - src/support/lyxsum.cpp
use FileName::isDirectory()
[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 "debug.h"
15 #include "support/FileName.h"
16
17 #include <boost/crc.hpp>
18
19 #include <algorithm>
20
21 using std::endl;
22 using std::string;
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(FileName const & file)
55 {
56         LYXERR(Debug::FILES) << "lyx::sum() using mmap (lightning fast)"
57                              << endl;
58
59         int fd = open(file.toFilesystemEncoding().c_str(), O_RDONLY);
60         if (!fd)
61                 return 0;
62
63         struct stat info;
64         fstat(fd, &info);
65
66         void * mm = mmap(0, info.st_size, PROT_READ,
67                          MAP_PRIVATE, fd, 0);
68         // Some platforms have the wrong type for MAP_FAILED (compaq cxx).
69         if (mm == reinterpret_cast<void*>(MAP_FAILED)) {
70                 close(fd);
71                 return 0;
72         }
73
74         char * beg = static_cast<char*>(mm);
75         char * end = beg + info.st_size;
76
77         boost::crc_32_type crc;
78         crc.process_block(beg, end);
79         unsigned long result = crc.checksum();
80
81         munmap(mm, info.st_size);
82         close(fd);
83
84         return result;
85 }
86
87 } // namespace support
88 } // namespace lyx
89
90 #else // No mmap
91
92 #include <fstream>
93 #include <iterator>
94
95
96 namespace {
97
98 template<typename InputIterator>
99 inline
100 unsigned long do_crc(InputIterator first, InputIterator last)
101 {
102         boost::crc_32_type crc;
103         crc = std::for_each(first, last, crc);
104         return crc.checksum();
105 }
106
107 } // namespace anon
108
109
110 namespace lyx {
111 namespace support {
112
113 using std::ifstream;
114 #if HAVE_DECL_ISTREAMBUF_ITERATOR
115 using std::istreambuf_iterator;
116
117 unsigned long sum(FileName const & file)
118 {
119         LYXERR(Debug::FILES) << "lyx::sum() using istreambuf_iterator (fast)"
120                              << endl;
121
122         // a directory may be passed here so we need to test it. (bug 3622)
123         if (file.isDirectory())
124                 return 0;
125         string filename = file.toFilesystemEncoding();
126         ifstream ifs(filename.c_str());
127         if (!ifs)
128                 return 0;
129
130         istreambuf_iterator<char> beg(ifs);
131         istreambuf_iterator<char> end;
132
133         return do_crc(beg,end);
134 }
135 #else
136
137 using std::istream_iterator;
138 using std::ios;
139
140 unsigned long sum(FileName const & file)
141 {
142         LYXERR(Debug::FILES)
143                 << "lyx::sum() using istream_iterator (slow as a snail)"
144                 << endl;
145
146         // a directory may be passed here so we need to test it. (bug 3622)
147         if (file.isDirectory())
148                 return 0;
149
150         string filename = file.toFilesystemEncoding();
151         ifstream ifs(filename.c_str());
152         if (!ifs)
153                 return 0;
154
155         ifs.unsetf(ios::skipws);
156         istream_iterator<char> beg(ifs);
157         istream_iterator<char> end;
158
159         return do_crc(beg,end);
160 }
161 #endif
162
163 } // namespace support
164 } // namespace lyx
165
166 #endif // mmap