1 //===- VersionTuple.cpp - Version Number Handling ---------------*- C++ -*-===//
2 //
3 //                     The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 //
10 // This file implements the VersionTuple class, which represents a version in
11 // the form major[.minor[.subminor]].
12 //
13 //===----------------------------------------------------------------------===//
14 #include "clang/Basic/VersionTuple.h"
15 #include "llvm/Support/raw_ostream.h"
16 
17 using namespace clang;
18 
19 std::string VersionTuple::getAsString() const {
20   std::string Result;
21   {
22     llvm::raw_string_ostream Out(Result);
23     Out << *this;
24   }
25   return Result;
26 }
27 
28 raw_ostream& clang::operator<<(raw_ostream &Out,
29                                      const VersionTuple &V) {
30   Out << V.getMajor();
31   if (Optional<unsigned> Minor = V.getMinor())
32     Out << (V.usesUnderscores() ? '_' : '.') << *Minor;
33   if (Optional<unsigned> Subminor = V.getSubminor())
34     Out << (V.usesUnderscores() ? '_' : '.') << *Subminor;
35   if (Optional<unsigned> Build = V.getBuild())
36     Out << (V.usesUnderscores() ? '_' : '.') << *Build;
37   return Out;
38 }
39 
40 static bool parseInt(StringRef &input, unsigned &value) {
41   assert(value == 0);
42   if (input.empty()) return true;
43 
44   char next = input[0];
45   input = input.substr(1);
46   if (next < '0' || next > '9') return true;
47   value = (unsigned) (next - '0');
48 
49   while (!input.empty()) {
50     next = input[0];
51     if (next < '0' || next > '9') return false;
52     input = input.substr(1);
53     value = value * 10 + (unsigned) (next - '0');
54   }
55 
56   return false;
57 }
58 
59 bool VersionTuple::tryParse(StringRef input) {
60   unsigned major = 0, minor = 0, micro = 0, build = 0;
61 
62   // Parse the major version, [0-9]+
63   if (parseInt(input, major)) return true;
64 
65   if (input.empty()) {
66     *this = VersionTuple(major);
67     return false;
68   }
69 
70   // If we're not done, parse the minor version, \.[0-9]+
71   if (input[0] != '.') return true;
72   input = input.substr(1);
73   if (parseInt(input, minor)) return true;
74 
75   if (input.empty()) {
76     *this = VersionTuple(major, minor);
77     return false;
78   }
79 
80   // If we're not done, parse the micro version, \.[0-9]+
81   if (input[0] != '.') return true;
82   input = input.substr(1);
83   if (parseInt(input, micro)) return true;
84 
85   if (input.empty()) {
86     *this = VersionTuple(major, minor, micro);
87     return false;
88   }
89 
90   // If we're not done, parse the micro version, \.[0-9]+
91   if (input[0] != '.') return true;
92   input = input.substr(1);
93   if (parseInt(input, build)) return true;
94 
95   // If we have characters left over, it's an error.
96   if (!input.empty()) return true;
97 
98   *this = VersionTuple(major, minor, micro, build);
99   return false;
100 }
101