1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
|
// file : build/utility.cxx -*- C++ -*-
// copyright : Copyright (c) 2014-2015 Code Synthesis Ltd
// license : MIT; see accompanying LICENSE file
#include <build/utility>
#include <cstdlib> // strtol()
using namespace std;
namespace build
{
const string empty_string;
const path empty_path;
const dir_path empty_dir_path;
unsigned int
to_version (const string& s)
{
// See tests/version.
//
auto parse = [&s] (size_t& p, const char* m, long min = 0, long max = 99)
-> unsigned int
{
if (s[p] == '-' || s[p] == '+') // stoi() allows these.
throw invalid_argument (m);
const char* b (s.c_str () + p);
char* e;
long r (strtol (b, &e, 10));
if (b == e || r < min || r > max)
throw invalid_argument (m);
p = e - s.c_str ();
return static_cast<unsigned int> (r);
};
auto bail = [] (const char* m) {throw invalid_argument (m);};
unsigned int ma, mi, bf, ab (0);
size_t p (0), n (s.size ());
ma = parse (p, "invalid major version");
if (p >= n || s[p] != '.')
bail ("'.' expected after major version");
mi = parse (++p, "invalid minor version");
if (p >= n || s[p] != '.')
bail ("'.' expected after minor version");
bf = parse (++p, "invalid bugfix version");
if (p < n)
{
if (s[p] != '-')
bail ("'-' expected after bugfix version");
char k (s[++p]);
if (k != 'a' && k != 'b')
bail ("'a' or 'b' expected in release component");
ab = parse (++p, "invalid release component", 1, 49);
if (p != n)
bail ("junk after release component");
if (k == 'b')
ab += 50;
}
// AABBCCDD
unsigned int r (ma * 1000000U +
mi * 10000U +
bf * 100U);
if (ab != 0)
{
if (r == 0)
bail ("0.0.0 version with release component");
r -= 100;
r += ab;
}
return r;
}
bool exception_unwinding_dtor = false;
}
|