annotate C++/Converter.cpp @ 217:1aceace80f61

Driver: remove useless line
author David Demelier <markand@malikania.fr>
date Mon, 05 May 2014 22:36:57 +0200
parents 5bb5712d400f
children 3c12f0e8bbb9
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
215
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
1 /*
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
2 * Converter.cpp -- iconv based converter
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
3 *
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
4 * Copyright (c) 2013, 2014 David Demelier <markand@malikania.fr>
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
5 *
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
6 * Permission to use, copy, modify, and/or distribute this software for any
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
7 * purpose with or without fee is hereby granted, provided that the above
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
8 * copyright notice and this permission notice appear in all copies.
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
9 *
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
10 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
11 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
12 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
13 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
14 * WHATSOEVER RESULTING FROM LOSS OF USE, DATA OR PROFITS, WHETHER IN AN
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
15 * ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING OUT OF
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
16 * OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
17 */
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
18
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
19 #include <cerrno>
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
20 #include <string>
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
21 #include <iterator>
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
22 #include <memory>
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
23 #include <string>
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
24 #include <stdexcept>
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
25 #include <vector>
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
26
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
27 #include <iconv.h>
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
28
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
29 #include "Converter.h"
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
30
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
31 struct Deleter {
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
32 void operator()(iconv_t desc)
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
33 {
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
34 iconv_close(desc);
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
35 }
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
36 };
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
37
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
38 using Iconv = std::unique_ptr<std::remove_pointer<iconv_t>::type, Deleter>;
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
39
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
40 std::string Converter::convert(const char *from,
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
41 const char *to,
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
42 const std::string &input)
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
43 {
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
44 // No conversion if from and to are identical
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
45 if (std::strcmp(from, to) == 0)
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
46 return input;
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
47
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
48 // Try to open the conversion descriptor
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
49 auto cd = iconv_open(to, from);
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
50
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
51 if (cd == (iconv_t)-1)
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
52 throw std::invalid_argument(std::strerror(errno));
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
53
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
54 Iconv cv(cd);
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
55 std::size_t insize(input.size());
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
56 std::size_t outsize(insize);
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
57 std::vector<char> result(insize + 1);
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
58
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
59 auto *b = &input[0];
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
60 auto *p = &result[0];
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
61
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
62 while (insize > 0) {
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
63 /* Convert */
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
64 auto r = iconv(cv.get(), &b, &insize, &p, &outsize);
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
65
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
66 if (r == (size_t)-1) {
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
67 switch (errno) {
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
68 case EBADF:
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
69 case EILSEQ:
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
70 case EINVAL:
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
71 throw std::invalid_argument(std::strerror(errno));
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
72 case E2BIG:
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
73 /*
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
74 * Here, we need to reallocate more data because the output
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
75 * string may need more space.
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
76 *
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
77 * We use 16 as an optimistic value.
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
78 */
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
79
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
80 result.reserve(result.size() + 16 + 1);
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
81 p = &result[result.size()];
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
82 outsize += 16;
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
83 default:
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
84 break;
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
85 }
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
86 }
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
87
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
88 }
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
89
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
90 return std::string(&result[0], (p - &result[0]));
5bb5712d400f Converter: iconv based converter
David Demelier <markand@malikania.fr>
parents:
diff changeset
91 }