Merge branch 'DTSC'
Conflicts: server.html
This commit is contained in:
		
						commit
						f91d7fa317
					
				
					 36 changed files with 2138 additions and 6770 deletions
				
			
		
							
								
								
									
										53
									
								
								util/amf.cpp
									
										
									
									
									
								
							
							
						
						
									
										53
									
								
								util/amf.cpp
									
										
									
									
									
								
							| 
						 | 
				
			
			@ -2,6 +2,7 @@
 | 
			
		|||
/// Holds all code for the AMF namespace.
 | 
			
		||||
 | 
			
		||||
#include "amf.h"
 | 
			
		||||
#include <sstream>
 | 
			
		||||
#include <cstdio> //needed for stderr only
 | 
			
		||||
 | 
			
		||||
/// Returns the std::string Indice for the current object, if available.
 | 
			
		||||
| 
						 | 
				
			
			@ -105,43 +106,45 @@ AMF::Object::Object(std::string indice, AMF::obj0type setType){//object type ini
 | 
			
		|||
/// Prints the contents of this object to std::cerr.
 | 
			
		||||
/// If this object contains other objects, it will call itself recursively
 | 
			
		||||
/// and print all nested content in a nice human-readable format.
 | 
			
		||||
void AMF::Object::Print(std::string indent){
 | 
			
		||||
  std::cerr << indent;
 | 
			
		||||
std::string AMF::Object::Print(std::string indent){
 | 
			
		||||
  std::stringstream st;
 | 
			
		||||
  st << indent;
 | 
			
		||||
  // print my type
 | 
			
		||||
  switch (myType){
 | 
			
		||||
    case AMF::AMF0_NUMBER: std::cerr << "Number"; break;
 | 
			
		||||
    case AMF::AMF0_BOOL: std::cerr << "Bool"; break;
 | 
			
		||||
    case AMF::AMF0_NUMBER: st << "Number"; break;
 | 
			
		||||
    case AMF::AMF0_BOOL: st << "Bool"; break;
 | 
			
		||||
    case AMF::AMF0_STRING://short string
 | 
			
		||||
    case AMF::AMF0_LONGSTRING: std::cerr << "String"; break;
 | 
			
		||||
    case AMF::AMF0_OBJECT: std::cerr << "Object"; break;
 | 
			
		||||
    case AMF::AMF0_MOVIECLIP: std::cerr << "MovieClip"; break;
 | 
			
		||||
    case AMF::AMF0_NULL: std::cerr << "Null"; break;
 | 
			
		||||
    case AMF::AMF0_UNDEFINED: std::cerr << "Undefined"; break;
 | 
			
		||||
    case AMF::AMF0_REFERENCE: std::cerr << "Reference"; break;
 | 
			
		||||
    case AMF::AMF0_ECMA_ARRAY: std::cerr << "ECMA Array"; break;
 | 
			
		||||
    case AMF::AMF0_OBJ_END: std::cerr << "Object end"; break;
 | 
			
		||||
    case AMF::AMF0_STRICT_ARRAY: std::cerr << "Strict Array"; break;
 | 
			
		||||
    case AMF::AMF0_DATE: std::cerr << "Date"; break;
 | 
			
		||||
    case AMF::AMF0_UNSUPPORTED: std::cerr << "Unsupported"; break;
 | 
			
		||||
    case AMF::AMF0_RECORDSET: std::cerr << "Recordset"; break;
 | 
			
		||||
    case AMF::AMF0_XMLDOC: std::cerr << "XML Document"; break;
 | 
			
		||||
    case AMF::AMF0_TYPED_OBJ: std::cerr << "Typed Object"; break;
 | 
			
		||||
    case AMF::AMF0_UPGRADE: std::cerr << "Upgrade to AMF3"; break;
 | 
			
		||||
    case AMF::AMF0_DDV_CONTAINER: std::cerr << "DDVTech Container"; break;
 | 
			
		||||
    case AMF::AMF0_LONGSTRING: st << "String"; break;
 | 
			
		||||
    case AMF::AMF0_OBJECT: st << "Object"; break;
 | 
			
		||||
    case AMF::AMF0_MOVIECLIP: st << "MovieClip"; break;
 | 
			
		||||
    case AMF::AMF0_NULL: st << "Null"; break;
 | 
			
		||||
    case AMF::AMF0_UNDEFINED: st << "Undefined"; break;
 | 
			
		||||
    case AMF::AMF0_REFERENCE: st << "Reference"; break;
 | 
			
		||||
    case AMF::AMF0_ECMA_ARRAY: st << "ECMA Array"; break;
 | 
			
		||||
    case AMF::AMF0_OBJ_END: st << "Object end"; break;
 | 
			
		||||
    case AMF::AMF0_STRICT_ARRAY: st << "Strict Array"; break;
 | 
			
		||||
    case AMF::AMF0_DATE: st << "Date"; break;
 | 
			
		||||
    case AMF::AMF0_UNSUPPORTED: st << "Unsupported"; break;
 | 
			
		||||
    case AMF::AMF0_RECORDSET: st << "Recordset"; break;
 | 
			
		||||
    case AMF::AMF0_XMLDOC: st << "XML Document"; break;
 | 
			
		||||
    case AMF::AMF0_TYPED_OBJ: st << "Typed Object"; break;
 | 
			
		||||
    case AMF::AMF0_UPGRADE: st << "Upgrade to AMF3"; break;
 | 
			
		||||
    case AMF::AMF0_DDV_CONTAINER: st << "DDVTech Container"; break;
 | 
			
		||||
  }
 | 
			
		||||
  // print my string indice, if available
 | 
			
		||||
  std::cerr << " " << myIndice << " ";
 | 
			
		||||
  st << " " << myIndice << " ";
 | 
			
		||||
  // print my numeric or string contents
 | 
			
		||||
  switch (myType){
 | 
			
		||||
    case AMF::AMF0_NUMBER: case AMF::AMF0_BOOL: case AMF::AMF0_REFERENCE: case AMF::AMF0_DATE: std::cerr << numval; break;
 | 
			
		||||
    case AMF::AMF0_STRING: case AMF::AMF0_LONGSTRING: case AMF::AMF0_XMLDOC: case AMF::AMF0_TYPED_OBJ: std::cerr << strval; break;
 | 
			
		||||
    case AMF::AMF0_NUMBER: case AMF::AMF0_BOOL: case AMF::AMF0_REFERENCE: case AMF::AMF0_DATE: st << numval; break;
 | 
			
		||||
    case AMF::AMF0_STRING: case AMF::AMF0_LONGSTRING: case AMF::AMF0_XMLDOC: case AMF::AMF0_TYPED_OBJ: st << strval; break;
 | 
			
		||||
    default: break;//we don't care about the rest, and don't want a compiler warning...
 | 
			
		||||
  }
 | 
			
		||||
  std::cerr << std::endl;
 | 
			
		||||
  st << std::endl;
 | 
			
		||||
  // if I hold other objects, print those too, recursively.
 | 
			
		||||
  if (contents.size() > 0){
 | 
			
		||||
    for (std::vector<AMF::Object>::iterator it = contents.begin(); it != contents.end(); it++){it->Print(indent+"  ");}
 | 
			
		||||
    for (std::vector<AMF::Object>::iterator it = contents.begin(); it != contents.end(); it++){st << it->Print(indent+"  ");}
 | 
			
		||||
  }
 | 
			
		||||
  return st.str();
 | 
			
		||||
};//print
 | 
			
		||||
 | 
			
		||||
/// Packs the AMF object to a std::string for transfer over the network.
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -70,7 +70,7 @@ namespace AMF{
 | 
			
		|||
      Object(std::string indice, double val, obj0type setType = AMF0_NUMBER);
 | 
			
		||||
      Object(std::string indice, std::string val, obj0type setType = AMF0_STRING);
 | 
			
		||||
      Object(std::string indice, obj0type setType = AMF0_OBJECT);
 | 
			
		||||
      void Print(std::string indent = "");
 | 
			
		||||
      std::string Print(std::string indent = "");
 | 
			
		||||
      std::string Pack();
 | 
			
		||||
    protected:
 | 
			
		||||
      std::string myIndice; ///< Holds this objects indice, if any.
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										45
									
								
								util/auth.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										45
									
								
								util/auth.cpp
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,45 @@
 | 
			
		|||
#include "auth.h"
 | 
			
		||||
#include "base64.h"
 | 
			
		||||
 | 
			
		||||
static unsigned char __gbv2keypub_der[] = {
 | 
			
		||||
  0x30, 0x82, 0x01, 0x22, 0x30, 0x0d, 0x06, 0x09, 0x2a, 0x86, 0x48, 0x86,
 | 
			
		||||
  0xf7, 0x0d, 0x01, 0x01, 0x01, 0x05, 0x00, 0x03, 0x82, 0x01, 0x0f, 0x00,
 | 
			
		||||
  0x30, 0x82, 0x01, 0x0a, 0x02, 0x82, 0x01, 0x01, 0x00, 0xe5, 0xd7, 0x9c,
 | 
			
		||||
  0x7d, 0x73, 0xc6, 0xe6, 0xfb, 0x35, 0x7e, 0xd7, 0x57, 0x99, 0x07, 0xdb,
 | 
			
		||||
  0x99, 0x70, 0xc9, 0xd0, 0x3e, 0x53, 0x57, 0x3c, 0x1e, 0x55, 0xda, 0x0f,
 | 
			
		||||
  0x69, 0xbf, 0x26, 0x79, 0xc7, 0xb6, 0xdd, 0x8e, 0x83, 0x32, 0x65, 0x74,
 | 
			
		||||
  0x0d, 0x74, 0x48, 0x42, 0x49, 0x22, 0x52, 0x58, 0x56, 0xc3, 0xe4, 0x49,
 | 
			
		||||
  0x5d, 0xac, 0x6a, 0x94, 0xb1, 0x64, 0x14, 0xbf, 0x4d, 0xd5, 0xd7, 0x3a,
 | 
			
		||||
  0xca, 0x5c, 0x1e, 0x6f, 0x42, 0x30, 0xac, 0x29, 0xaa, 0xa0, 0x85, 0xd2,
 | 
			
		||||
  0x16, 0xa2, 0x8e, 0x89, 0x12, 0xc4, 0x92, 0x06, 0xea, 0xed, 0x48, 0xf6,
 | 
			
		||||
  0xdb, 0xed, 0x4f, 0x62, 0x6c, 0xfa, 0xcf, 0xc2, 0xb9, 0x8d, 0x04, 0xb2,
 | 
			
		||||
  0xba, 0x63, 0xc9, 0xcc, 0xee, 0x23, 0x64, 0x46, 0x14, 0x12, 0xc8, 0x38,
 | 
			
		||||
  0x67, 0x69, 0x6b, 0xaf, 0xd1, 0x7c, 0xb1, 0xb5, 0x79, 0xe4, 0x4e, 0x3a,
 | 
			
		||||
  0xa7, 0xe8, 0x28, 0x89, 0x25, 0xc0, 0xd0, 0xd8, 0xc7, 0xd2, 0x26, 0xaa,
 | 
			
		||||
  0xf5, 0xbf, 0x36, 0x55, 0x01, 0x89, 0x58, 0x1f, 0x1e, 0xf5, 0xa5, 0x42,
 | 
			
		||||
  0x8f, 0x60, 0x2e, 0xc2, 0xd8, 0x21, 0x0b, 0x6c, 0x8d, 0xbb, 0x72, 0xf2,
 | 
			
		||||
  0x19, 0x30, 0xe3, 0x4c, 0x3e, 0x80, 0xe7, 0xf2, 0xe3, 0x89, 0x4f, 0xd4,
 | 
			
		||||
  0xee, 0x96, 0x3e, 0x4a, 0x9b, 0xe5, 0x16, 0x01, 0xf1, 0x98, 0xc9, 0x0b,
 | 
			
		||||
  0xd6, 0xdf, 0x8a, 0x64, 0x47, 0xc4, 0x44, 0xcc, 0x92, 0x69, 0x28, 0xee,
 | 
			
		||||
  0x7d, 0xac, 0xdc, 0x30, 0x56, 0x3a, 0xe7, 0xbc, 0xba, 0x45, 0x16, 0x2c,
 | 
			
		||||
  0x4c, 0x46, 0x6b, 0x2b, 0x20, 0xfb, 0x3d, 0x20, 0x35, 0xbb, 0x48, 0x49,
 | 
			
		||||
  0x13, 0x65, 0xc9, 0x9a, 0x38, 0x10, 0x84, 0x1a, 0x8c, 0xc9, 0xd7, 0xde,
 | 
			
		||||
  0x07, 0x10, 0x5a, 0xfb, 0xb4, 0x95, 0xae, 0x18, 0xf2, 0xe3, 0x15, 0xe8,
 | 
			
		||||
  0xad, 0x7e, 0xe5, 0x3c, 0xa8, 0x47, 0x85, 0xd6, 0x1f, 0x54, 0xb5, 0xa3,
 | 
			
		||||
  0x79, 0x02, 0x03, 0x01, 0x00, 0x01
 | 
			
		||||
}; ///< The GBv2 public key file.
 | 
			
		||||
static unsigned int __gbv2keypub_der_len = 294; ///< Length of GBv2 public key data
 | 
			
		||||
 | 
			
		||||
/// Attempts to load the GBv2 public key.
 | 
			
		||||
Auth::Auth(){
 | 
			
		||||
  const unsigned char * key = __gbv2keypub_der;
 | 
			
		||||
  pubkey = d2i_RSAPublicKey(0, &key, __gbv2keypub_der_len);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Attempts to verify RSA signature using the public key.
 | 
			
		||||
/// Assumes basesign argument is base64 encoded RSA signature for data.
 | 
			
		||||
/// Returns true if the data could be verified, false otherwise.
 | 
			
		||||
bool Auth::PubKey_Check(std::string & data, std::string basesign){
 | 
			
		||||
  std::string sign = Base64::decode(basesign);
 | 
			
		||||
  return (RSA_verify(NID_md5, (unsigned char*)data.c_str(), data.size(), (unsigned char*)sign.c_str(), sign.size(), pubkey) == 1);
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										11
									
								
								util/auth.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								util/auth.h
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,11 @@
 | 
			
		|||
#include <string>
 | 
			
		||||
#include <openssl/rsa.h>
 | 
			
		||||
#include <openssl/x509.h>
 | 
			
		||||
 | 
			
		||||
class Auth{
 | 
			
		||||
  private:
 | 
			
		||||
    RSA * pubkey; ///< Holds the public key.
 | 
			
		||||
  public:
 | 
			
		||||
    Auth(); ///< Attempts to load the GBv2 public key.
 | 
			
		||||
    bool PubKey_Check(std::string & data, std::string basesign); ///< Attempts to verify RSA signature using the public key.
 | 
			
		||||
};
 | 
			
		||||
							
								
								
									
										64
									
								
								util/base64.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										64
									
								
								util/base64.cpp
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,64 @@
 | 
			
		|||
#include "base64.h"
 | 
			
		||||
 | 
			
		||||
/// Needed for base64_encode function
 | 
			
		||||
const std::string Base64::chars = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/";
 | 
			
		||||
 | 
			
		||||
  /// Helper for base64_decode function
 | 
			
		||||
inline bool Base64::is_base64(unsigned char c) {
 | 
			
		||||
  return (isalnum(c) || (c == '+') || (c == '/'));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Used to base64 encode data. Input is the plaintext as std::string, output is the encoded data as std::string.
 | 
			
		||||
/// \param input Plaintext data to encode.
 | 
			
		||||
/// \returns Base64 encoded data.
 | 
			
		||||
std::string Base64::encode(std::string const input) {
 | 
			
		||||
  std::string ret;
 | 
			
		||||
  unsigned int in_len = input.size();
 | 
			
		||||
  char quad[4], triple[3];
 | 
			
		||||
  unsigned int i, x, n = 3;
 | 
			
		||||
  for (x = 0; x < in_len; x = x + 3){
 | 
			
		||||
    if ((in_len - x) / 3 == 0){n = (in_len - x) % 3;}
 | 
			
		||||
    for (i=0; i < 3; i++){triple[i] = '0';}
 | 
			
		||||
    for (i=0; i < n; i++){triple[i] = input[x + i];}
 | 
			
		||||
    quad[0] = chars[(triple[0] & 0xFC) >> 2]; // FC = 11111100
 | 
			
		||||
    quad[1] = chars[((triple[0] & 0x03) << 4) | ((triple[1] & 0xF0) >> 4)]; // 03 = 11
 | 
			
		||||
    quad[2] = chars[((triple[1] & 0x0F) << 2) | ((triple[2] & 0xC0) >> 6)]; // 0F = 1111, C0=11110
 | 
			
		||||
    quad[3] = chars[triple[2] & 0x3F]; // 3F = 111111
 | 
			
		||||
    if (n < 3){quad[3] = '=';}
 | 
			
		||||
    if (n < 2){quad[2] = '=';}
 | 
			
		||||
    for(i=0; i < 4; i++){ret += quad[i];}
 | 
			
		||||
  }
 | 
			
		||||
  return ret;
 | 
			
		||||
}//base64_encode
 | 
			
		||||
 | 
			
		||||
/// Used to base64 decode data. Input is the encoded data as std::string, output is the plaintext data as std::string.
 | 
			
		||||
/// \param input Base64 encoded data to decode.
 | 
			
		||||
/// \returns Plaintext decoded data.
 | 
			
		||||
std::string Base64::decode(std::string const& encoded_string) {
 | 
			
		||||
  int in_len = encoded_string.size();
 | 
			
		||||
  int i = 0;
 | 
			
		||||
  int j = 0;
 | 
			
		||||
  int in_ = 0;
 | 
			
		||||
  unsigned char char_array_4[4], char_array_3[3];
 | 
			
		||||
  std::string ret;
 | 
			
		||||
  while (in_len-- && ( encoded_string[in_] != '=') && is_base64(encoded_string[in_])) {
 | 
			
		||||
    char_array_4[i++] = encoded_string[in_]; in_++;
 | 
			
		||||
    if (i ==4) {
 | 
			
		||||
      for (i = 0; i <4; i++){char_array_4[i] = chars.find(char_array_4[i]);}
 | 
			
		||||
      char_array_3[0] = (char_array_4[0] << 2) + ((char_array_4[1] & 0x30) >> 4);
 | 
			
		||||
      char_array_3[1] = ((char_array_4[1] & 0xf) << 4) + ((char_array_4[2] & 0x3c) >> 2);
 | 
			
		||||
      char_array_3[2] = ((char_array_4[2] & 0x3) << 6) + char_array_4[3];
 | 
			
		||||
      for (i = 0; (i < 3); i++){ret += char_array_3[i];}
 | 
			
		||||
      i = 0;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  if (i) {
 | 
			
		||||
    for (j = i; j <4; j++){char_array_4[j] = 0;}
 | 
			
		||||
    for (j = 0; j <4; j++){char_array_4[j] = chars.find(char_array_4[j]);}
 | 
			
		||||
    char_array_3[0] = (char_array_4[0] << 2) + ((char_array_4[1] & 0x30) >> 4);
 | 
			
		||||
    char_array_3[1] = ((char_array_4[1] & 0xf) << 4) + ((char_array_4[2] & 0x3c) >> 2);
 | 
			
		||||
    char_array_3[2] = ((char_array_4[2] & 0x3) << 6) + char_array_4[3];
 | 
			
		||||
    for (j = 0; (j < i - 1); j++) ret += char_array_3[j];
 | 
			
		||||
  }
 | 
			
		||||
  return ret;
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										11
									
								
								util/base64.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										11
									
								
								util/base64.h
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,11 @@
 | 
			
		|||
#include <string>
 | 
			
		||||
 | 
			
		||||
/// Holds base64 decoding and encoding functions.
 | 
			
		||||
class Base64{
 | 
			
		||||
  private:
 | 
			
		||||
    static const std::string chars;
 | 
			
		||||
    static inline bool is_base64(unsigned char c);
 | 
			
		||||
  public:
 | 
			
		||||
    static std::string encode(std::string const input);
 | 
			
		||||
    static std::string decode(std::string const& encoded_string);
 | 
			
		||||
};
 | 
			
		||||
							
								
								
									
										105
									
								
								util/config.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										105
									
								
								util/config.cpp
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,105 @@
 | 
			
		|||
/// \file config.cpp
 | 
			
		||||
/// Contains generic functions for managing configuration.
 | 
			
		||||
 | 
			
		||||
#include "config.h"
 | 
			
		||||
#include <string.h>
 | 
			
		||||
#include <signal.h>
 | 
			
		||||
 | 
			
		||||
#ifdef __FreeBSD__
 | 
			
		||||
#include <sys/wait.h>
 | 
			
		||||
#else
 | 
			
		||||
#include <wait.h>
 | 
			
		||||
#endif
 | 
			
		||||
#include <errno.h>
 | 
			
		||||
#include <iostream>
 | 
			
		||||
#include <sys/types.h>
 | 
			
		||||
#include <unistd.h>
 | 
			
		||||
#include <fcntl.h>
 | 
			
		||||
#include <pwd.h>
 | 
			
		||||
#include <getopt.h>
 | 
			
		||||
#include <stdlib.h>
 | 
			
		||||
#include <stdio.h>
 | 
			
		||||
#include <fstream>
 | 
			
		||||
 | 
			
		||||
/// Creates a new configuration manager.
 | 
			
		||||
Util::Config::Config(){
 | 
			
		||||
  listen_port = 4242;
 | 
			
		||||
  daemon_mode = true;
 | 
			
		||||
  interface = "0.0.0.0";
 | 
			
		||||
  username = "root";
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Parses commandline arguments.
 | 
			
		||||
/// Calls exit if an unknown option is encountered, printing a help message.
 | 
			
		||||
/// confsection must be either already set or never be set at all when this function is called.
 | 
			
		||||
/// In other words: do not change confsection after calling this function.
 | 
			
		||||
void Util::Config::parseArgs(int argc, char ** argv){
 | 
			
		||||
  int opt = 0;
 | 
			
		||||
  static const char *optString = "ndvp:i:u:c:h?";
 | 
			
		||||
  static const struct option longOpts[] = {
 | 
			
		||||
    {"help",0,0,'h'},
 | 
			
		||||
    {"port",1,0,'p'},
 | 
			
		||||
    {"interface",1,0,'i'},
 | 
			
		||||
    {"username",1,0,'u'},
 | 
			
		||||
    {"no-daemon",0,0,'n'},
 | 
			
		||||
    {"daemon",0,0,'d'},
 | 
			
		||||
    {"version",0,0,'v'}
 | 
			
		||||
  };
 | 
			
		||||
  while ((opt = getopt_long(argc, argv, optString, longOpts, 0)) != -1){
 | 
			
		||||
    switch (opt){
 | 
			
		||||
      case 'p': listen_port = atoi(optarg); break;
 | 
			
		||||
      case 'i': interface = optarg; break;
 | 
			
		||||
      case 'n': daemon_mode = false; break;
 | 
			
		||||
      case 'd': daemon_mode = true; break;
 | 
			
		||||
      case 'u': username = optarg; break;
 | 
			
		||||
      case 'v':
 | 
			
		||||
        printf("%s\n", TOSTRING(VERSION));
 | 
			
		||||
        exit(1);
 | 
			
		||||
        break;
 | 
			
		||||
      case 'h':
 | 
			
		||||
      case '?':
 | 
			
		||||
        std::string doingdaemon = "true";
 | 
			
		||||
        if (!daemon_mode){doingdaemon = "false";}
 | 
			
		||||
        printf("Options: -h[elp], -?, -v[ersion], -n[odaemon], -d[aemon], -p[ort] VAL, -i[nterface] VAL, -u[sername] VAL\n");
 | 
			
		||||
        printf("Defaults:\n  interface: %s\n  port: %i\n  daemon mode: %s\n  username: %s\n", interface.c_str(), listen_port, doingdaemon.c_str(), username.c_str());
 | 
			
		||||
        printf("Username root means no change to UID, no matter what the UID is.\n");
 | 
			
		||||
        printf("This is %s version %s\n", argv[0], TOSTRING(VERSION));
 | 
			
		||||
        exit(1);
 | 
			
		||||
        break;
 | 
			
		||||
    }
 | 
			
		||||
  }//commandline options parser
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Sets the current process' running user
 | 
			
		||||
void Util::setUser(std::string username){
 | 
			
		||||
  if (username != "root"){
 | 
			
		||||
    struct passwd * user_info = getpwnam(username.c_str());
 | 
			
		||||
    if (!user_info){
 | 
			
		||||
      #if DEBUG >= 1
 | 
			
		||||
      fprintf(stderr, "Error: could not setuid %s: could not get PID\n", username.c_str());
 | 
			
		||||
      #endif
 | 
			
		||||
      return;
 | 
			
		||||
    }else{
 | 
			
		||||
      if (setuid(user_info->pw_uid) != 0){
 | 
			
		||||
        #if DEBUG >= 1
 | 
			
		||||
        fprintf(stderr, "Error: could not setuid %s: not allowed\n", username.c_str());
 | 
			
		||||
        #endif
 | 
			
		||||
      }else{
 | 
			
		||||
        #if DEBUG >= 3
 | 
			
		||||
        fprintf(stderr, "Changed user to %s\n", username.c_str());
 | 
			
		||||
        #endif
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Will turn the current process into a daemon.
 | 
			
		||||
/// Works by calling daemon(1,0):
 | 
			
		||||
/// Does not change directory to root.
 | 
			
		||||
/// Does redirect output to /dev/null
 | 
			
		||||
void Util::Daemonize(){
 | 
			
		||||
  #if DEBUG >= 3
 | 
			
		||||
  fprintf(stderr, "Going into background mode...\n");
 | 
			
		||||
  #endif
 | 
			
		||||
  daemon(1, 0);
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										29
									
								
								util/config.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										29
									
								
								util/config.h
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,29 @@
 | 
			
		|||
/// \file config.h
 | 
			
		||||
/// Contains generic function headers for managing configuration.
 | 
			
		||||
 | 
			
		||||
#include <string>
 | 
			
		||||
 | 
			
		||||
#define STRINGIFY(x) #x
 | 
			
		||||
#define TOSTRING(x) STRINGIFY(x)
 | 
			
		||||
 | 
			
		||||
/// Contains utility code, not directly related to streaming media
 | 
			
		||||
namespace Util{
 | 
			
		||||
 | 
			
		||||
  /// Deals with parsing configuration from commandline options.
 | 
			
		||||
  class Config{
 | 
			
		||||
    public:
 | 
			
		||||
      bool daemon_mode;
 | 
			
		||||
      std::string interface;
 | 
			
		||||
      int listen_port;
 | 
			
		||||
      std::string username;
 | 
			
		||||
      Config();
 | 
			
		||||
      void parseArgs(int argc, char ** argv);
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  /// Will set the active user to the named username.
 | 
			
		||||
  void setUser(std::string user);
 | 
			
		||||
 | 
			
		||||
  /// Will turn the current process into a daemon.
 | 
			
		||||
  void Daemonize();
 | 
			
		||||
 | 
			
		||||
};
 | 
			
		||||
| 
						 | 
				
			
			@ -23,6 +23,12 @@ DTSC::Stream::Stream(unsigned int rbuffers){
 | 
			
		|||
  buffercount = rbuffers;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Returns the time in milliseconds of the last received packet.
 | 
			
		||||
/// This is _not_ the time this packet was received, only the stored time.
 | 
			
		||||
unsigned int DTSC::Stream::getTime(){
 | 
			
		||||
  return buffers.front().getContentP("time")->NumValue();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Attempts to parse a packet from the given std::string buffer.
 | 
			
		||||
/// Returns true if successful, removing the parsed part from the buffer string.
 | 
			
		||||
/// Returns false if invalid or not enough data is in the buffer.
 | 
			
		||||
| 
						 | 
				
			
			@ -60,8 +66,9 @@ bool DTSC::Stream::parsePacket(std::string & buffer){
 | 
			
		|||
      return true;
 | 
			
		||||
    }
 | 
			
		||||
    #if DEBUG >= 2
 | 
			
		||||
    std::cerr << "Error: Invalid DTMI data! I *will* get stuck!" << std::endl;
 | 
			
		||||
    std::cerr << "Error: Invalid DTMI data: " << buffer.substr(0, 4) << std::endl;
 | 
			
		||||
    #endif
 | 
			
		||||
    buffer.erase(0, 1);
 | 
			
		||||
  }
 | 
			
		||||
  return false;
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			@ -191,6 +198,13 @@ const char * DTSC::DTMI::Str(){return strval.c_str();};
 | 
			
		|||
/// If this object is not a container type, this function will always return 0.
 | 
			
		||||
int DTSC::DTMI::hasContent(){return contents.size();};
 | 
			
		||||
 | 
			
		||||
/// Returns true if this DTSC::DTMI value is non-default.
 | 
			
		||||
/// Non-default means it is either not a root element or has content.
 | 
			
		||||
bool DTSC::DTMI::isEmpty(){
 | 
			
		||||
  if (myType != DTMI_ROOT){return false;}
 | 
			
		||||
  return (hasContent() == 0);
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/// Adds an DTSC::DTMI to this object. Works for all types, but only makes sense for container types.
 | 
			
		||||
/// This function resets DTMI::packed to an empty string, forcing a repack on the next call to DTMI::Pack.
 | 
			
		||||
/// If the indice name already exists, replaces the indice.
 | 
			
		||||
| 
						 | 
				
			
			@ -206,9 +220,12 @@ void DTSC::DTMI::addContent(DTSC::DTMI c){
 | 
			
		|||
};
 | 
			
		||||
 | 
			
		||||
/// Returns a pointer to the object held at indice i.
 | 
			
		||||
/// Returns AMF::AMF0_DDV_CONTAINER of indice "error" if no object is held at this indice.
 | 
			
		||||
/// Returns null pointer if no object is held at this indice.
 | 
			
		||||
/// \param i The indice of the object in this container.
 | 
			
		||||
DTSC::DTMI* DTSC::DTMI::getContentP(int i){return &contents.at(i);};
 | 
			
		||||
DTSC::DTMI* DTSC::DTMI::getContentP(int i){
 | 
			
		||||
  if (contents.size() <= (unsigned int)i){return 0;}
 | 
			
		||||
  return &contents.at(i);
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/// Returns a copy of the object held at indice i.
 | 
			
		||||
/// Returns a AMF::AMF0_DDV_CONTAINER of indice "error" if no object is held at this indice.
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -62,6 +62,7 @@ namespace DTSC{
 | 
			
		|||
    std::string & StrValue();
 | 
			
		||||
    const char * Str();
 | 
			
		||||
    int hasContent();
 | 
			
		||||
    bool isEmpty();
 | 
			
		||||
    void addContent(DTMI c);
 | 
			
		||||
    DTMI* getContentP(int i);
 | 
			
		||||
    DTMI getContent(int i);
 | 
			
		||||
| 
						 | 
				
			
			@ -106,9 +107,9 @@ namespace DTSC{
 | 
			
		|||
  class Ring {
 | 
			
		||||
    public:
 | 
			
		||||
      Ring(unsigned int v);
 | 
			
		||||
      unsigned int b; ///< Holds current number of buffer. May and is intended to change unexpectedly!
 | 
			
		||||
      bool waiting; ///< If true, this Ring is currently waiting for a buffer fill.
 | 
			
		||||
      bool starved; ///< If true, this Ring can no longer receive valid data.
 | 
			
		||||
      volatile unsigned int b; ///< Holds current number of buffer. May and is intended to change unexpectedly!
 | 
			
		||||
      volatile bool waiting; ///< If true, this Ring is currently waiting for a buffer fill.
 | 
			
		||||
      volatile bool starved; ///< If true, this Ring can no longer receive valid data.
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  /// Holds temporary data for a DTSC stream and provides functions to utilize it.
 | 
			
		||||
| 
						 | 
				
			
			@ -129,6 +130,7 @@ namespace DTSC{
 | 
			
		|||
      std::string & outPacket(unsigned int num);
 | 
			
		||||
      std::string & outHeader();
 | 
			
		||||
      Ring * getRing();
 | 
			
		||||
      unsigned int getTime();
 | 
			
		||||
      void dropRing(Ring * ptr);
 | 
			
		||||
  private:
 | 
			
		||||
      std::deque<DTSC::DTMI> buffers;
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										319
									
								
								util/flv_tag.cpp
									
										
									
									
									
								
							
							
						
						
									
										319
									
								
								util/flv_tag.cpp
									
										
									
									
									
								
							| 
						 | 
				
			
			@ -9,6 +9,7 @@
 | 
			
		|||
#include <fcntl.h> //for Tag::FileLoader
 | 
			
		||||
#include <stdlib.h> //malloc
 | 
			
		||||
#include <string.h> //memcpy
 | 
			
		||||
#include <sstream>
 | 
			
		||||
 | 
			
		||||
/// Holds the last FLV header parsed.
 | 
			
		||||
/// Defaults to a audio+video header on FLV version 0x01 if no header received yet.
 | 
			
		||||
| 
						 | 
				
			
			@ -100,80 +101,84 @@ bool FLV::Tag::isInitData(){
 | 
			
		|||
/// audio, video or metadata, what encoding is used, and the details
 | 
			
		||||
/// of the encoding itself.
 | 
			
		||||
std::string FLV::Tag::tagType(){
 | 
			
		||||
  std::string R = "";
 | 
			
		||||
  std::stringstream R;
 | 
			
		||||
  R << len << " bytes of ";
 | 
			
		||||
  switch (data[0]){
 | 
			
		||||
    case 0x09:
 | 
			
		||||
      switch (data[11] & 0x0F){
 | 
			
		||||
        case 1: R += "JPEG"; break;
 | 
			
		||||
        case 2: R += "H263"; break;
 | 
			
		||||
        case 3: R += "ScreenVideo1"; break;
 | 
			
		||||
        case 4: R += "VP6"; break;
 | 
			
		||||
        case 5: R += "VP6Alpha"; break;
 | 
			
		||||
        case 6: R += "ScreenVideo2"; break;
 | 
			
		||||
        case 7: R += "H264"; break;
 | 
			
		||||
        default: R += "unknown"; break;
 | 
			
		||||
        case 1: R << "JPEG"; break;
 | 
			
		||||
        case 2: R << "H263"; break;
 | 
			
		||||
        case 3: R << "ScreenVideo1"; break;
 | 
			
		||||
        case 4: R << "VP6"; break;
 | 
			
		||||
        case 5: R << "VP6Alpha"; break;
 | 
			
		||||
        case 6: R << "ScreenVideo2"; break;
 | 
			
		||||
        case 7: R << "H264"; break;
 | 
			
		||||
        default: R << "unknown"; break;
 | 
			
		||||
      }
 | 
			
		||||
    R += " video ";
 | 
			
		||||
    R << " video ";
 | 
			
		||||
    switch (data[11] & 0xF0){
 | 
			
		||||
        case 0x10: R += "keyframe"; break;
 | 
			
		||||
        case 0x20: R += "iframe"; break;
 | 
			
		||||
        case 0x30: R += "disposableiframe"; break;
 | 
			
		||||
        case 0x40: R += "generatedkeyframe"; break;
 | 
			
		||||
        case 0x50: R += "videoinfo"; break;
 | 
			
		||||
        case 0x10: R << "keyframe"; break;
 | 
			
		||||
        case 0x20: R << "iframe"; break;
 | 
			
		||||
        case 0x30: R << "disposableiframe"; break;
 | 
			
		||||
        case 0x40: R << "generatedkeyframe"; break;
 | 
			
		||||
        case 0x50: R << "videoinfo"; break;
 | 
			
		||||
      }
 | 
			
		||||
      if ((data[11] & 0x0F) == 7){
 | 
			
		||||
        switch (data[12]){
 | 
			
		||||
          case 0: R += " header"; break;
 | 
			
		||||
          case 1: R += " NALU"; break;
 | 
			
		||||
          case 2: R += " endofsequence"; break;
 | 
			
		||||
          case 0: R << " header"; break;
 | 
			
		||||
          case 1: R << " NALU"; break;
 | 
			
		||||
          case 2: R << " endofsequence"; break;
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
      break;
 | 
			
		||||
    case 0x08:
 | 
			
		||||
      switch (data[11] & 0xF0){
 | 
			
		||||
        case 0x00: R += "linear PCM PE"; break;
 | 
			
		||||
        case 0x10: R += "ADPCM"; break;
 | 
			
		||||
        case 0x20: R += "MP3"; break;
 | 
			
		||||
        case 0x30: R += "linear PCM LE"; break;
 | 
			
		||||
        case 0x40: R += "Nelly16kHz"; break;
 | 
			
		||||
        case 0x50: R += "Nelly8kHz"; break;
 | 
			
		||||
        case 0x60: R += "Nelly"; break;
 | 
			
		||||
        case 0x70: R += "G711A-law"; break;
 | 
			
		||||
        case 0x80: R += "G711mu-law"; break;
 | 
			
		||||
        case 0x90: R += "reserved"; break;
 | 
			
		||||
        case 0xA0: R += "AAC"; break;
 | 
			
		||||
        case 0xB0: R += "Speex"; break;
 | 
			
		||||
        case 0xE0: R += "MP38kHz"; break;
 | 
			
		||||
        case 0xF0: R += "DeviceSpecific"; break;
 | 
			
		||||
        default: R += "unknown"; break;
 | 
			
		||||
        case 0x00: R << "linear PCM PE"; break;
 | 
			
		||||
        case 0x10: R << "ADPCM"; break;
 | 
			
		||||
        case 0x20: R << "MP3"; break;
 | 
			
		||||
        case 0x30: R << "linear PCM LE"; break;
 | 
			
		||||
        case 0x40: R << "Nelly16kHz"; break;
 | 
			
		||||
        case 0x50: R << "Nelly8kHz"; break;
 | 
			
		||||
        case 0x60: R << "Nelly"; break;
 | 
			
		||||
        case 0x70: R << "G711A-law"; break;
 | 
			
		||||
        case 0x80: R << "G711mu-law"; break;
 | 
			
		||||
        case 0x90: R << "reserved"; break;
 | 
			
		||||
        case 0xA0: R << "AAC"; break;
 | 
			
		||||
        case 0xB0: R << "Speex"; break;
 | 
			
		||||
        case 0xE0: R << "MP38kHz"; break;
 | 
			
		||||
        case 0xF0: R << "DeviceSpecific"; break;
 | 
			
		||||
        default: R << "unknown"; break;
 | 
			
		||||
      }
 | 
			
		||||
      switch (data[11] & 0x0C){
 | 
			
		||||
        case 0x0: R += " 5.5kHz"; break;
 | 
			
		||||
        case 0x4: R += " 11kHz"; break;
 | 
			
		||||
        case 0x8: R += " 22kHz"; break;
 | 
			
		||||
        case 0xC: R += " 44kHz"; break;
 | 
			
		||||
        case 0x0: R << " 5.5kHz"; break;
 | 
			
		||||
        case 0x4: R << " 11kHz"; break;
 | 
			
		||||
        case 0x8: R << " 22kHz"; break;
 | 
			
		||||
        case 0xC: R << " 44kHz"; break;
 | 
			
		||||
      }
 | 
			
		||||
      switch (data[11] & 0x02){
 | 
			
		||||
        case 0: R += " 8bit"; break;
 | 
			
		||||
        case 2: R += " 16bit"; break;
 | 
			
		||||
        case 0: R << " 8bit"; break;
 | 
			
		||||
        case 2: R << " 16bit"; break;
 | 
			
		||||
      }
 | 
			
		||||
      switch (data[11] & 0x01){
 | 
			
		||||
        case 0: R += " mono"; break;
 | 
			
		||||
        case 1: R += " stereo"; break;
 | 
			
		||||
        case 0: R << " mono"; break;
 | 
			
		||||
        case 1: R << " stereo"; break;
 | 
			
		||||
      }
 | 
			
		||||
      R += " audio";
 | 
			
		||||
      R << " audio";
 | 
			
		||||
      if ((data[12] == 0) && ((data[11] & 0xF0) == 0xA0)){
 | 
			
		||||
        R += " initdata";
 | 
			
		||||
        R << " initdata";
 | 
			
		||||
      }
 | 
			
		||||
      break;
 | 
			
		||||
    case 0x12:
 | 
			
		||||
      R += "(meta)data";
 | 
			
		||||
    case 0x12:{
 | 
			
		||||
      R << "(meta)data: ";
 | 
			
		||||
      AMF::Object metadata = AMF::parse((unsigned char*)data+11, len-15);
 | 
			
		||||
      R << metadata.Print();
 | 
			
		||||
      break;
 | 
			
		||||
    }
 | 
			
		||||
    default:
 | 
			
		||||
      R += "unknown";
 | 
			
		||||
      R << "unknown";
 | 
			
		||||
      break;
 | 
			
		||||
  }
 | 
			
		||||
  return R;
 | 
			
		||||
  return R.str();
 | 
			
		||||
}//FLV::Tag::tagtype
 | 
			
		||||
 | 
			
		||||
/// Returns the 32-bit timestamp of this tag.
 | 
			
		||||
| 
						 | 
				
			
			@ -297,7 +302,7 @@ bool FLV::Tag::DTSCLoader(DTSC::Stream & S){
 | 
			
		|||
        if (S.getPacket().getContentP("interframe")){data[11] += 0x20;}
 | 
			
		||||
        if (S.getPacket().getContentP("disposableframe")){data[11] += 0x30;}
 | 
			
		||||
        break;
 | 
			
		||||
      case DTSC::AUDIO:
 | 
			
		||||
      case DTSC::AUDIO:{
 | 
			
		||||
        if ((unsigned int)len == S.lastData().length() + 16){
 | 
			
		||||
          memcpy(data+12, S.lastData().c_str(), S.lastData().length());
 | 
			
		||||
        }else{
 | 
			
		||||
| 
						 | 
				
			
			@ -307,12 +312,18 @@ bool FLV::Tag::DTSCLoader(DTSC::Stream & S){
 | 
			
		|||
        data[11] = 0;
 | 
			
		||||
        if (S.metadata.getContentP("audio")->getContentP("codec")->StrValue() == "AAC"){data[11] += 0xA0;}
 | 
			
		||||
        if (S.metadata.getContentP("audio")->getContentP("codec")->StrValue() == "MP3"){data[11] += 0x20;}
 | 
			
		||||
        if (S.metadata.getContentP("audio")->getContentP("rate")->NumValue() == 11025){data[11] += 0x04;}
 | 
			
		||||
        if (S.metadata.getContentP("audio")->getContentP("rate")->NumValue() == 22050){data[11] += 0x08;}
 | 
			
		||||
        if (S.metadata.getContentP("audio")->getContentP("rate")->NumValue() == 44100){data[11] += 0x0C;}
 | 
			
		||||
        unsigned int datarate = S.metadata.getContentP("audio")->getContentP("rate")->NumValue();
 | 
			
		||||
        if (datarate >= 44100){
 | 
			
		||||
          data[11] += 0x0C;
 | 
			
		||||
        }else if(datarate >= 22050){
 | 
			
		||||
          data[11] += 0x08;
 | 
			
		||||
        }else if(datarate >= 11025){
 | 
			
		||||
          data[11] += 0x04;
 | 
			
		||||
        }
 | 
			
		||||
        if (S.metadata.getContentP("audio")->getContentP("size")->NumValue() == 16){data[11] += 0x02;}
 | 
			
		||||
        if (S.metadata.getContentP("audio")->getContentP("channels")->NumValue() > 1){data[11] += 0x01;}
 | 
			
		||||
        break;
 | 
			
		||||
      }
 | 
			
		||||
      case DTSC::META:
 | 
			
		||||
        memcpy(data+11, S.lastData().c_str(), S.lastData().length());
 | 
			
		||||
        break;
 | 
			
		||||
| 
						 | 
				
			
			@ -329,6 +340,9 @@ bool FLV::Tag::DTSCLoader(DTSC::Stream & S){
 | 
			
		|||
  data[1] = ((len-15) >> 16) & 0xFF;
 | 
			
		||||
  data[2] = ((len-15) >> 8) & 0xFF;
 | 
			
		||||
  data[3] = (len-15) & 0xFF;
 | 
			
		||||
  data[8] = 0;
 | 
			
		||||
  data[9] = 0;
 | 
			
		||||
  data[10] = 0;
 | 
			
		||||
  tagTime(S.getPacket().getContentP("time")->NumValue());
 | 
			
		||||
  return true;
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			@ -336,7 +350,7 @@ bool FLV::Tag::DTSCLoader(DTSC::Stream & S){
 | 
			
		|||
/// Helper function that properly sets the tag length from the internal len variable.
 | 
			
		||||
void FLV::Tag::setLen(){
 | 
			
		||||
  int len4 = len - 4;
 | 
			
		||||
  int i = len-1;
 | 
			
		||||
  int i = len;
 | 
			
		||||
  data[--i] = (len4) & 0xFF;
 | 
			
		||||
  len4 >>= 8;
 | 
			
		||||
  data[--i] = (len4) & 0xFF;
 | 
			
		||||
| 
						 | 
				
			
			@ -375,6 +389,9 @@ bool FLV::Tag::DTSCVideoInit(DTSC::Stream & S){
 | 
			
		|||
  data[1] = ((len-15) >> 16) & 0xFF;
 | 
			
		||||
  data[2] = ((len-15) >> 8) & 0xFF;
 | 
			
		||||
  data[3] = (len-15) & 0xFF;
 | 
			
		||||
  data[8] = 0;
 | 
			
		||||
  data[9] = 0;
 | 
			
		||||
  data[10] = 0;
 | 
			
		||||
  tagTime(0);
 | 
			
		||||
  return true;
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			@ -402,23 +419,25 @@ bool FLV::Tag::DTSCAudioInit(DTSC::Stream & S){
 | 
			
		|||
    data[11] = 0;
 | 
			
		||||
    if (S.metadata.getContentP("audio")->getContentP("codec")->StrValue() == "AAC"){data[11] += 0xA0;}
 | 
			
		||||
    if (S.metadata.getContentP("audio")->getContentP("codec")->StrValue() == "MP3"){data[11] += 0x20;}
 | 
			
		||||
    if (S.metadata.getContentP("audio")->getContentP("rate")->NumValue() == 11000){data[11] += 0x04;}
 | 
			
		||||
    if (S.metadata.getContentP("audio")->getContentP("rate")->NumValue() == 22000){data[11] += 0x08;}
 | 
			
		||||
    if (S.metadata.getContentP("audio")->getContentP("rate")->NumValue() == 44000){data[11] += 0x0C;}
 | 
			
		||||
    unsigned int datarate = S.metadata.getContentP("audio")->getContentP("rate")->NumValue();
 | 
			
		||||
    if (datarate >= 44100){
 | 
			
		||||
      data[11] += 0x0C;
 | 
			
		||||
    }else if(datarate >= 22050){
 | 
			
		||||
      data[11] += 0x08;
 | 
			
		||||
    }else if(datarate >= 11025){
 | 
			
		||||
      data[11] += 0x04;
 | 
			
		||||
    }
 | 
			
		||||
    if (S.metadata.getContentP("audio")->getContentP("size")->NumValue() == 16){data[11] += 0x02;}
 | 
			
		||||
    if (S.metadata.getContentP("audio")->getContentP("channels")->NumValue() > 1){data[11] += 0x01;}
 | 
			
		||||
  }
 | 
			
		||||
  setLen();
 | 
			
		||||
  switch (S.lastType()){
 | 
			
		||||
    case DTSC::VIDEO: data[0] = 0x09; break;
 | 
			
		||||
    case DTSC::AUDIO: data[0] = 0x08; break;
 | 
			
		||||
    case DTSC::META: data[0] = 0x12; break;
 | 
			
		||||
    default: break;
 | 
			
		||||
  }
 | 
			
		||||
  data[0] = 0x08;
 | 
			
		||||
  data[1] = ((len-15) >> 16) & 0xFF;
 | 
			
		||||
  data[2] = ((len-15) >> 8) & 0xFF;
 | 
			
		||||
  data[3] = (len-15) & 0xFF;
 | 
			
		||||
  data[8] = 0;
 | 
			
		||||
  data[9] = 0;
 | 
			
		||||
  data[10] = 0;
 | 
			
		||||
  tagTime(0);
 | 
			
		||||
  return true;
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			@ -501,6 +520,9 @@ bool FLV::Tag::DTSCMetaInit(DTSC::Stream & S){
 | 
			
		|||
  data[1] = ((len-15) >> 16) & 0xFF;
 | 
			
		||||
  data[2] = ((len-15) >> 8) & 0xFF;
 | 
			
		||||
  data[3] = (len-15) & 0xFF;
 | 
			
		||||
  data[8] = 0;
 | 
			
		||||
  data[9] = 0;
 | 
			
		||||
  data[10] = 0;
 | 
			
		||||
  tagTime(0);
 | 
			
		||||
  return true;
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			@ -749,3 +771,178 @@ bool FLV::Tag::FileLoader(FILE * f){
 | 
			
		|||
  fcntl(fileno(f), F_SETFL, preflags);
 | 
			
		||||
  return false;
 | 
			
		||||
}//FLV_GetPacket
 | 
			
		||||
 | 
			
		||||
DTSC::DTMI FLV::Tag::toDTSC(DTSC::DTMI & metadata){
 | 
			
		||||
  DTSC::DTMI pack_out; // Storage for outgoing DTMI data.
 | 
			
		||||
 | 
			
		||||
  if (data[0] == 0x12){
 | 
			
		||||
    AMF::Object meta_in = AMF::parse((unsigned char*)data+11, len-15);
 | 
			
		||||
    if (meta_in.getContentP(0) && (meta_in.getContentP(0)->StrValue() == "onMetaData") && meta_in.getContentP(1)){
 | 
			
		||||
      AMF::Object * tmp = meta_in.getContentP(1);
 | 
			
		||||
      if (tmp->getContentP("videocodecid")){
 | 
			
		||||
        switch ((unsigned int)tmp->getContentP("videocodecid")->NumValue()){
 | 
			
		||||
          case 2: Meta_Put(metadata, "video", "codec", "H263"); break;
 | 
			
		||||
          case 4: Meta_Put(metadata, "video", "codec", "VP6"); break;
 | 
			
		||||
          case 7: Meta_Put(metadata, "video", "codec", "H264"); break;
 | 
			
		||||
          default: Meta_Put(metadata, "video", "codec", "?"); break;
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
      if (tmp->getContentP("audiocodecid")){
 | 
			
		||||
        switch ((unsigned int)tmp->getContentP("audiocodecid")->NumValue()){
 | 
			
		||||
          case 2: Meta_Put(metadata, "audio", "codec", "MP3"); break;
 | 
			
		||||
          case 10: Meta_Put(metadata, "audio", "codec", "AAC"); break;
 | 
			
		||||
          default: Meta_Put(metadata, "audio", "codec", "?"); break;
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
      if (tmp->getContentP("width")){
 | 
			
		||||
        Meta_Put(metadata, "video", "width", tmp->getContentP("width")->NumValue());
 | 
			
		||||
      }
 | 
			
		||||
      if (tmp->getContentP("height")){
 | 
			
		||||
        Meta_Put(metadata, "video", "height", tmp->getContentP("height")->NumValue());
 | 
			
		||||
      }
 | 
			
		||||
      if (tmp->getContentP("framerate")){
 | 
			
		||||
        Meta_Put(metadata, "video", "fpks", tmp->getContentP("framerate")->NumValue()*1000);
 | 
			
		||||
      }
 | 
			
		||||
      if (tmp->getContentP("videodatarate")){
 | 
			
		||||
        Meta_Put(metadata, "video", "bps", (tmp->getContentP("videodatarate")->NumValue()*1024)/8);
 | 
			
		||||
      }
 | 
			
		||||
      if (tmp->getContentP("audiodatarate")){
 | 
			
		||||
        Meta_Put(metadata, "audio", "bps", (tmp->getContentP("audiodatarate")->NumValue()*1024)/8);
 | 
			
		||||
      }
 | 
			
		||||
      if (tmp->getContentP("audiosamplerate")){
 | 
			
		||||
        Meta_Put(metadata, "audio", "rate", tmp->getContentP("audiosamplerate")->NumValue());
 | 
			
		||||
      }
 | 
			
		||||
      if (tmp->getContentP("audiosamplesize")){
 | 
			
		||||
        Meta_Put(metadata, "audio", "size", tmp->getContentP("audiosamplesize")->NumValue());
 | 
			
		||||
      }
 | 
			
		||||
      if (tmp->getContentP("stereo")){
 | 
			
		||||
        if (tmp->getContentP("stereo")->NumValue() == 1){
 | 
			
		||||
          Meta_Put(metadata, "audio", "channels", 2);
 | 
			
		||||
        }else{
 | 
			
		||||
          Meta_Put(metadata, "audio", "channels", 1);
 | 
			
		||||
        }
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    return pack_out;//empty
 | 
			
		||||
  }
 | 
			
		||||
  if (data[0] == 0x08){
 | 
			
		||||
    char audiodata = data[11];
 | 
			
		||||
    if (needsInitData() && isInitData()){
 | 
			
		||||
      if ((audiodata & 0xF0) == 0xA0){
 | 
			
		||||
        Meta_Put(metadata, "audio", "init", std::string((char*)data+13, (size_t)len-17));
 | 
			
		||||
      }else{
 | 
			
		||||
        Meta_Put(metadata, "audio", "init", std::string((char*)data+12, (size_t)len-16));
 | 
			
		||||
      }
 | 
			
		||||
      return pack_out;//skip rest of parsing, get next tag.
 | 
			
		||||
    }
 | 
			
		||||
    pack_out = DTSC::DTMI("audio", DTSC::DTMI_ROOT);
 | 
			
		||||
    pack_out.addContent(DTSC::DTMI("datatype", "audio"));
 | 
			
		||||
    pack_out.addContent(DTSC::DTMI("time", tagTime()));
 | 
			
		||||
    if (!Meta_Has(metadata, "audio", "codec")){
 | 
			
		||||
      switch (audiodata & 0xF0){
 | 
			
		||||
        case 0x20: Meta_Put(metadata, "audio", "codec", "MP3"); break;
 | 
			
		||||
        case 0xA0: Meta_Put(metadata, "audio", "codec", "AAC"); break;
 | 
			
		||||
        default: Meta_Put(metadata, "audio", "codec", "?"); break;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    if (!Meta_Has(metadata, "audio", "rate")){
 | 
			
		||||
      switch (audiodata & 0x0C){
 | 
			
		||||
        case 0x0: Meta_Put(metadata, "audio", "rate", 5512); break;
 | 
			
		||||
        case 0x4: Meta_Put(metadata, "audio", "rate", 11025); break;
 | 
			
		||||
        case 0x8: Meta_Put(metadata, "audio", "rate", 22050); break;
 | 
			
		||||
        case 0xC: Meta_Put(metadata, "audio", "rate", 44100); break;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    if (!Meta_Has(metadata, "audio", "size")){
 | 
			
		||||
      switch (audiodata & 0x02){
 | 
			
		||||
        case 0x0: Meta_Put(metadata, "audio", "size", 8); break;
 | 
			
		||||
        case 0x2: Meta_Put(metadata, "audio", "size", 16); break;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    if (!Meta_Has(metadata, "audio", "channels")){
 | 
			
		||||
      switch (audiodata & 0x01){
 | 
			
		||||
        case 0x0: Meta_Put(metadata, "audio", "channels", 1); break;
 | 
			
		||||
        case 0x1: Meta_Put(metadata, "audio", "channels", 2); break;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    if ((audiodata & 0xF0) == 0xA0){
 | 
			
		||||
      pack_out.addContent(DTSC::DTMI("data", std::string((char*)data+13, (size_t)len-17)));
 | 
			
		||||
    }else{
 | 
			
		||||
      pack_out.addContent(DTSC::DTMI("data", std::string((char*)data+12, (size_t)len-16)));
 | 
			
		||||
    }
 | 
			
		||||
    return pack_out;
 | 
			
		||||
  }
 | 
			
		||||
  if (data[0] == 0x09){
 | 
			
		||||
    char videodata = data[11];
 | 
			
		||||
    if (needsInitData() && isInitData()){
 | 
			
		||||
      if ((videodata & 0x0F) == 7){
 | 
			
		||||
        Meta_Put(metadata, "video", "init", std::string((char*)data+16, (size_t)len-20));
 | 
			
		||||
      }else{
 | 
			
		||||
        Meta_Put(metadata, "video", "init", std::string((char*)data+12, (size_t)len-16));
 | 
			
		||||
      }
 | 
			
		||||
      return pack_out;//skip rest of parsing, get next tag.
 | 
			
		||||
    }
 | 
			
		||||
    if (!Meta_Has(metadata, "video", "codec")){
 | 
			
		||||
      switch (videodata & 0x0F){
 | 
			
		||||
        case 2: Meta_Put(metadata, "video", "codec", "H263"); break;
 | 
			
		||||
        case 4: Meta_Put(metadata, "video", "codec", "VP6"); break;
 | 
			
		||||
        case 7: Meta_Put(metadata, "video", "codec", "H264"); break;
 | 
			
		||||
        default: Meta_Put(metadata, "video", "codec", "?"); break;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
    pack_out = DTSC::DTMI("video", DTSC::DTMI_ROOT);
 | 
			
		||||
    pack_out.addContent(DTSC::DTMI("datatype", "video"));
 | 
			
		||||
    switch (videodata & 0xF0){
 | 
			
		||||
      case 0x10: pack_out.addContent(DTSC::DTMI("keyframe", 1)); break;
 | 
			
		||||
      case 0x20: pack_out.addContent(DTSC::DTMI("interframe", 1)); break;
 | 
			
		||||
      case 0x30: pack_out.addContent(DTSC::DTMI("disposableframe", 1)); break;
 | 
			
		||||
      case 0x40: pack_out.addContent(DTSC::DTMI("keyframe", 1)); break;
 | 
			
		||||
      case 0x50: return DTSC::DTMI(); break;//the video info byte we just throw away - useless to us...
 | 
			
		||||
    }
 | 
			
		||||
    pack_out.addContent(DTSC::DTMI("time", tagTime()));
 | 
			
		||||
    if ((videodata & 0x0F) == 7){
 | 
			
		||||
      switch (data[12]){
 | 
			
		||||
        case 1: pack_out.addContent(DTSC::DTMI("nalu", 1)); break;
 | 
			
		||||
        case 2: pack_out.addContent(DTSC::DTMI("nalu_end", 1)); break;
 | 
			
		||||
      }
 | 
			
		||||
      int offset = (data[13] << 16) + (data[14] << 8) + data[15];
 | 
			
		||||
      offset = (offset << 8) >> 8;
 | 
			
		||||
      pack_out.addContent(DTSC::DTMI("offset", offset));
 | 
			
		||||
      pack_out.addContent(DTSC::DTMI("data", std::string((char*)data+16, (size_t)len-20)));
 | 
			
		||||
    }else{
 | 
			
		||||
      pack_out.addContent(DTSC::DTMI("data", std::string((char*)data+12, (size_t)len-16)));
 | 
			
		||||
    }
 | 
			
		||||
    return pack_out;
 | 
			
		||||
  }
 | 
			
		||||
  return pack_out;//should never get here
 | 
			
		||||
}//FLV::Tag::toDTSC
 | 
			
		||||
 | 
			
		||||
/// Inserts std::string type metadata into the passed DTMI object.
 | 
			
		||||
/// \arg meta The DTMI object to put the metadata into.
 | 
			
		||||
/// \arg cat Metadata category to insert into.
 | 
			
		||||
/// \arg elem Element name to put into the category.
 | 
			
		||||
/// \arg val Value to put into the element name.
 | 
			
		||||
void FLV::Tag::Meta_Put(DTSC::DTMI & meta, std::string cat, std::string elem, std::string val){
 | 
			
		||||
  if (meta.getContentP(cat) == 0){meta.addContent(DTSC::DTMI(cat));}
 | 
			
		||||
  meta.getContentP(cat)->addContent(DTSC::DTMI(elem, val));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Inserts uint64_t type metadata into the passed DTMI object.
 | 
			
		||||
/// \arg meta The DTMI object to put the metadata into.
 | 
			
		||||
/// \arg cat Metadata category to insert into.
 | 
			
		||||
/// \arg elem Element name to put into the category.
 | 
			
		||||
/// \arg val Value to put into the element name.
 | 
			
		||||
void FLV::Tag::Meta_Put(DTSC::DTMI & meta, std::string cat, std::string elem, uint64_t val){
 | 
			
		||||
  if (meta.getContentP(cat) == 0){meta.addContent(DTSC::DTMI(cat));}
 | 
			
		||||
  meta.getContentP(cat)->addContent(DTSC::DTMI(elem, val));
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Returns true if the named category and elementname are available in the metadata.
 | 
			
		||||
/// \arg meta The DTMI object to check.
 | 
			
		||||
/// \arg cat Metadata category to check.
 | 
			
		||||
/// \arg elem Element name to check.
 | 
			
		||||
bool FLV::Tag::Meta_Has(DTSC::DTMI & meta, std::string cat, std::string elem){
 | 
			
		||||
  if (meta.getContentP(cat) == 0){return false;}
 | 
			
		||||
  if (meta.getContentP(cat)->getContentP(elem) == 0){return false;}
 | 
			
		||||
  return true;
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -43,6 +43,7 @@ namespace FLV {
 | 
			
		|||
      bool DTSCVideoInit(DTSC::Stream & S);
 | 
			
		||||
      bool DTSCAudioInit(DTSC::Stream & S);
 | 
			
		||||
      bool DTSCMetaInit(DTSC::Stream & S);
 | 
			
		||||
      DTSC::DTMI toDTSC(DTSC::DTMI & metadata);
 | 
			
		||||
      bool MemLoader(char * D, unsigned int S, unsigned int & P);
 | 
			
		||||
      bool SockLoader(int sock);
 | 
			
		||||
      bool SockLoader(Socket::Connection sock);
 | 
			
		||||
| 
						 | 
				
			
			@ -56,6 +57,10 @@ namespace FLV {
 | 
			
		|||
      bool MemReadUntil(char * buffer, unsigned int count, unsigned int & sofar, char * D, unsigned int S, unsigned int & P);
 | 
			
		||||
      bool SockReadUntil(char * buffer, unsigned int count, unsigned int & sofar, Socket::Connection & sock);
 | 
			
		||||
      bool FileReadUntil(char * buffer, unsigned int count, unsigned int & sofar, FILE * f);
 | 
			
		||||
      //DTSC writer helpers
 | 
			
		||||
      void Meta_Put(DTSC::DTMI & meta, std::string cat, std::string elem, std::string val);
 | 
			
		||||
      void Meta_Put(DTSC::DTMI & meta, std::string cat, std::string elem, uint64_t val);
 | 
			
		||||
      bool Meta_Has(DTSC::DTMI & meta, std::string cat, std::string elem);
 | 
			
		||||
  };//Tag
 | 
			
		||||
 | 
			
		||||
};//FLV namespace
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										436
									
								
								util/json.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										436
									
								
								util/json.cpp
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,436 @@
 | 
			
		|||
/// \file json.cpp Holds all JSON-related code.
 | 
			
		||||
 | 
			
		||||
#include "json.h"
 | 
			
		||||
#include <sstream>
 | 
			
		||||
#include <fstream>
 | 
			
		||||
 | 
			
		||||
int JSON::Value::c2hex(int c){
 | 
			
		||||
  if (c >= '0' && c <= '9') return c - '0';
 | 
			
		||||
  if (c >= 'a' && c <= 'f') return c - 'a' + 10;
 | 
			
		||||
  if (c >= 'A' && c <= 'F') return c - 'A' + 10;
 | 
			
		||||
  return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
std::string JSON::Value::read_string(int separator, std::istream & fromstream){
 | 
			
		||||
  std::string out;
 | 
			
		||||
  bool escaped = false;
 | 
			
		||||
  while (fromstream.good()){
 | 
			
		||||
    int c = fromstream.get();
 | 
			
		||||
    if (c == '\\'){
 | 
			
		||||
      escaped = true;
 | 
			
		||||
      continue;
 | 
			
		||||
    }
 | 
			
		||||
    if (escaped){
 | 
			
		||||
      switch (c){
 | 
			
		||||
        case 'b': out += '\b'; break;
 | 
			
		||||
        case 'f': out += '\f'; break;
 | 
			
		||||
        case 'n': out += '\n'; break;
 | 
			
		||||
        case 'r': out += '\r'; break;
 | 
			
		||||
        case 't': out += '\t'; break;
 | 
			
		||||
        case 'u':{
 | 
			
		||||
           int d1 = fromstream.get();
 | 
			
		||||
           int d2 = fromstream.get();
 | 
			
		||||
           int d3 = fromstream.get();
 | 
			
		||||
           int d4 = fromstream.get();
 | 
			
		||||
           c = c2hex(d4) + (c2hex(d3) << 4) + (c2hex(d2) << 8) + (c2hex(d1) << 16);
 | 
			
		||||
        }
 | 
			
		||||
        default:
 | 
			
		||||
          out += (char)c;
 | 
			
		||||
          break;
 | 
			
		||||
      }
 | 
			
		||||
    }else{
 | 
			
		||||
      if (c == separator){
 | 
			
		||||
        return out;
 | 
			
		||||
      }else{
 | 
			
		||||
        out += (char)c;
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  return out;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
std::string JSON::Value::string_escape(std::string val){
 | 
			
		||||
  std::string out = "\"";
 | 
			
		||||
  for (unsigned int i = 0; i < val.size(); ++i){
 | 
			
		||||
    switch (val[i]){
 | 
			
		||||
      case '"': out += "\\\""; break;
 | 
			
		||||
      case '\\': out += "\\\\"; break;
 | 
			
		||||
      case '\n': out += "\\n"; break;
 | 
			
		||||
      case '\b': out += "\\b"; break;
 | 
			
		||||
      case '\f': out += "\\f"; break;
 | 
			
		||||
      case '\r': out += "\\r"; break;
 | 
			
		||||
      case '\t': out += "\\t"; break;
 | 
			
		||||
      default: out += val[i];
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  out += "\"";
 | 
			
		||||
  return out;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/// Sets this JSON::Value to null;
 | 
			
		||||
JSON::Value::Value(){
 | 
			
		||||
  null();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Sets this JSON::Value to read from this position in the std::istream
 | 
			
		||||
JSON::Value::Value(std::istream & fromstream){
 | 
			
		||||
  null();
 | 
			
		||||
  bool reading_object = false;
 | 
			
		||||
  bool reading_obj_name = false;
 | 
			
		||||
  bool reading_array = false;
 | 
			
		||||
  while (fromstream.good()){
 | 
			
		||||
    int c = fromstream.peek();
 | 
			
		||||
    switch (c){
 | 
			
		||||
      case '{':
 | 
			
		||||
        reading_object = true;
 | 
			
		||||
        reading_obj_name = true;
 | 
			
		||||
        c = fromstream.get();
 | 
			
		||||
        myType = OBJECT;
 | 
			
		||||
        break;
 | 
			
		||||
      case '[':
 | 
			
		||||
        reading_array = true;
 | 
			
		||||
        c = fromstream.get();
 | 
			
		||||
        myType = ARRAY;
 | 
			
		||||
        append(JSON::Value(fromstream));
 | 
			
		||||
        break;
 | 
			
		||||
      case '\'':
 | 
			
		||||
      case '"':
 | 
			
		||||
        c = fromstream.get();
 | 
			
		||||
        if (!reading_object || !reading_obj_name){
 | 
			
		||||
          myType = STRING;
 | 
			
		||||
          strVal = read_string(c, fromstream);
 | 
			
		||||
          return;
 | 
			
		||||
        }else{
 | 
			
		||||
          std::string tmpstr = read_string(c, fromstream);
 | 
			
		||||
          (*this)[tmpstr] = JSON::Value(fromstream);
 | 
			
		||||
        }
 | 
			
		||||
        break;
 | 
			
		||||
      case '0':
 | 
			
		||||
      case '1':
 | 
			
		||||
      case '2':
 | 
			
		||||
      case '3':
 | 
			
		||||
      case '4':
 | 
			
		||||
      case '5':
 | 
			
		||||
      case '6':
 | 
			
		||||
      case '7':
 | 
			
		||||
      case '8':
 | 
			
		||||
      case '9':
 | 
			
		||||
        c = fromstream.get();
 | 
			
		||||
        myType = INTEGER;
 | 
			
		||||
        intVal *= 10;
 | 
			
		||||
        intVal += c - '0';
 | 
			
		||||
        break;
 | 
			
		||||
      case ',':
 | 
			
		||||
        if (!reading_object && !reading_array) return;
 | 
			
		||||
        c = fromstream.get();
 | 
			
		||||
        if (reading_object){
 | 
			
		||||
          reading_obj_name = true;
 | 
			
		||||
        }else{
 | 
			
		||||
          append(JSON::Value(fromstream));
 | 
			
		||||
        }
 | 
			
		||||
        break;
 | 
			
		||||
      case '}':
 | 
			
		||||
        if (reading_object){c = fromstream.get();}
 | 
			
		||||
        return;
 | 
			
		||||
        break;
 | 
			
		||||
      case ']':
 | 
			
		||||
        if (reading_array){c = fromstream.get();}
 | 
			
		||||
        return;
 | 
			
		||||
        break;
 | 
			
		||||
      case 't':
 | 
			
		||||
      case 'T':
 | 
			
		||||
        myType = BOOL;
 | 
			
		||||
        intVal = 1;
 | 
			
		||||
        return;
 | 
			
		||||
        break;
 | 
			
		||||
      case 'f':
 | 
			
		||||
      case 'F':
 | 
			
		||||
        myType = BOOL;
 | 
			
		||||
        intVal = 0;
 | 
			
		||||
        return;
 | 
			
		||||
        break;
 | 
			
		||||
      case 'n':
 | 
			
		||||
      case 'N':
 | 
			
		||||
        myType = EMPTY;
 | 
			
		||||
        return;
 | 
			
		||||
        break;
 | 
			
		||||
      default:
 | 
			
		||||
        c = fromstream.get();//ignore this character
 | 
			
		||||
        continue;
 | 
			
		||||
        break;
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Sets this JSON::Value to the given string.
 | 
			
		||||
JSON::Value::Value(const std::string & val){
 | 
			
		||||
  myType = STRING;
 | 
			
		||||
  strVal = val;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Sets this JSON::Value to the given string.
 | 
			
		||||
JSON::Value::Value(const char * val){
 | 
			
		||||
  myType = STRING;
 | 
			
		||||
  strVal = val;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Sets this JSON::Value to the given integer.
 | 
			
		||||
JSON::Value::Value(long long int val){
 | 
			
		||||
  myType = INTEGER;
 | 
			
		||||
  intVal = val;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Compares a JSON::Value to another for equality.
 | 
			
		||||
bool JSON::Value::operator==(const JSON::Value & rhs) const{
 | 
			
		||||
  if (myType != rhs.myType) return false;
 | 
			
		||||
  if (myType == INTEGER || myType == BOOL){return intVal == rhs.intVal;}
 | 
			
		||||
  if (myType == STRING){return strVal == rhs.strVal;}
 | 
			
		||||
  if (myType == EMPTY){return true;}
 | 
			
		||||
  if (myType == OBJECT){
 | 
			
		||||
    if (objVal.size() != rhs.objVal.size()) return false;
 | 
			
		||||
    for (std::map<std::string, Value>::const_iterator it = objVal.begin(); it != objVal.end(); ++it){
 | 
			
		||||
      if (!rhs.isMember(it->first)){return false;}
 | 
			
		||||
      if (it->second != rhs.objVal.find(it->first)->second){return false;}
 | 
			
		||||
    }
 | 
			
		||||
    return true;
 | 
			
		||||
  }
 | 
			
		||||
  return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Compares a JSON::Value to another for equality.
 | 
			
		||||
bool JSON::Value::operator!=(const JSON::Value & rhs) const{
 | 
			
		||||
  return !((*this) == rhs);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Sets this JSON::Value to the given boolean.
 | 
			
		||||
JSON::Value & JSON::Value::operator=(const bool &rhs){
 | 
			
		||||
  null();
 | 
			
		||||
  myType = BOOL;
 | 
			
		||||
  if (rhs) intVal = 1;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Sets this JSON::Value to the given string.
 | 
			
		||||
JSON::Value & JSON::Value::operator=(const std::string &rhs){
 | 
			
		||||
  null();
 | 
			
		||||
  myType = STRING;
 | 
			
		||||
  strVal = rhs;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Sets this JSON::Value to the given string.
 | 
			
		||||
JSON::Value & JSON::Value::operator=(const char * rhs){
 | 
			
		||||
  return ((*this) = (std::string)rhs);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Sets this JSON::Value to the given integer.
 | 
			
		||||
JSON::Value & JSON::Value::operator=(const long long int &rhs){
 | 
			
		||||
  null();
 | 
			
		||||
  myType = INTEGER;
 | 
			
		||||
  intVal = rhs;
 | 
			
		||||
  return *this;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Sets this JSON::Value to the given integer.
 | 
			
		||||
JSON::Value & JSON::Value::operator=(const int &rhs){
 | 
			
		||||
  return ((*this) = (long long int)rhs);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Sets this JSON::Value to the given integer.
 | 
			
		||||
JSON::Value & JSON::Value::operator=(const unsigned int &rhs){
 | 
			
		||||
  return ((*this) = (long long int)rhs);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Automatic conversion to long long int - returns 0 if not an integer type.
 | 
			
		||||
JSON::Value::operator long long int(){
 | 
			
		||||
  return intVal;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/// Automatic conversion to std::string.
 | 
			
		||||
/// Returns the raw string value if available, otherwise calls toString().
 | 
			
		||||
JSON::Value::operator std::string(){
 | 
			
		||||
  if (myType == STRING){
 | 
			
		||||
    return strVal;
 | 
			
		||||
  }else{
 | 
			
		||||
    if (myType == EMPTY){
 | 
			
		||||
      return "";
 | 
			
		||||
    }else{
 | 
			
		||||
      return toString();
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Retrieves or sets the JSON::Value at this position in the object.
 | 
			
		||||
/// Converts destructively to object if not already an object.
 | 
			
		||||
JSON::Value & JSON::Value::operator[](const std::string i){
 | 
			
		||||
  if (myType != OBJECT){
 | 
			
		||||
    null();
 | 
			
		||||
    myType = OBJECT;
 | 
			
		||||
  }
 | 
			
		||||
  return objVal[i];
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Retrieves or sets the JSON::Value at this position in the object.
 | 
			
		||||
/// Converts destructively to object if not already an object.
 | 
			
		||||
JSON::Value & JSON::Value::operator[](const char * i){
 | 
			
		||||
  if (myType != OBJECT){
 | 
			
		||||
    null();
 | 
			
		||||
    myType = OBJECT;
 | 
			
		||||
  }
 | 
			
		||||
  return objVal[i];
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Retrieves or sets the JSON::Value at this position in the array.
 | 
			
		||||
/// Converts destructively to array if not already an array.
 | 
			
		||||
JSON::Value & JSON::Value::operator[](unsigned int i){
 | 
			
		||||
  if (myType != ARRAY){
 | 
			
		||||
    null();
 | 
			
		||||
    myType = ARRAY;
 | 
			
		||||
  }
 | 
			
		||||
  while (i >= arrVal.size()){
 | 
			
		||||
    append(JSON::Value());
 | 
			
		||||
  }
 | 
			
		||||
  return arrVal[i];
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Converts this JSON::Value to valid JSON notation and returns it.
 | 
			
		||||
/// Makes absolutely no attempts to pretty-print anything. :-)
 | 
			
		||||
std::string JSON::Value::toString(){
 | 
			
		||||
  switch (myType){
 | 
			
		||||
    case INTEGER: {
 | 
			
		||||
      std::stringstream st;
 | 
			
		||||
      st << intVal;
 | 
			
		||||
      return st.str();
 | 
			
		||||
      break;
 | 
			
		||||
    }
 | 
			
		||||
    case STRING: {
 | 
			
		||||
      return string_escape(strVal);
 | 
			
		||||
      break;
 | 
			
		||||
    }
 | 
			
		||||
    case ARRAY: {
 | 
			
		||||
      std::string tmp = "[";
 | 
			
		||||
      for (ArrIter it = ArrBegin(); it != ArrEnd(); it++){
 | 
			
		||||
        tmp += it->toString();
 | 
			
		||||
        if (it + 1 != ArrEnd()){tmp += ",";}
 | 
			
		||||
      }
 | 
			
		||||
      tmp += "]";
 | 
			
		||||
      return tmp;
 | 
			
		||||
      break;
 | 
			
		||||
    }
 | 
			
		||||
    case OBJECT: {
 | 
			
		||||
      std::string tmp2 = "{";
 | 
			
		||||
      ObjIter it3 = ObjEnd();
 | 
			
		||||
      --it3;
 | 
			
		||||
      for (ObjIter it2 = ObjBegin(); it2 != ObjEnd(); it2++){
 | 
			
		||||
        tmp2 += "\"" + it2->first + "\":";
 | 
			
		||||
        tmp2 += it2->second.toString();
 | 
			
		||||
        if (it2 != it3){tmp2 += ",";}
 | 
			
		||||
      }
 | 
			
		||||
      tmp2 += "}";
 | 
			
		||||
      return tmp2;
 | 
			
		||||
      break;
 | 
			
		||||
    }
 | 
			
		||||
    case EMPTY:
 | 
			
		||||
    default:
 | 
			
		||||
      return "null";
 | 
			
		||||
  }
 | 
			
		||||
  return "null";//should never get here...
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Appends the given value to the end of this JSON::Value array.
 | 
			
		||||
/// Turns this value into an array if it is not already one.
 | 
			
		||||
void JSON::Value::append(const JSON::Value & rhs){
 | 
			
		||||
  if (myType != ARRAY){
 | 
			
		||||
    null();
 | 
			
		||||
    myType = ARRAY;
 | 
			
		||||
  }
 | 
			
		||||
  arrVal.push_back(rhs);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Prepends the given value to the beginning of this JSON::Value array.
 | 
			
		||||
/// Turns this value into an array if it is not already one.
 | 
			
		||||
void JSON::Value::prepend(const JSON::Value & rhs){
 | 
			
		||||
  if (myType != ARRAY){
 | 
			
		||||
    null();
 | 
			
		||||
    myType = ARRAY;
 | 
			
		||||
  }
 | 
			
		||||
  arrVal.push_front(rhs);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// For array and object JSON::Value objects, reduces them
 | 
			
		||||
/// so they contain at most size elements, throwing away
 | 
			
		||||
/// the first elements and keeping the last ones.
 | 
			
		||||
/// Does nothing for other JSON::Value types, nor does it
 | 
			
		||||
/// do anything if the size is already lower or equal to the
 | 
			
		||||
/// given size.
 | 
			
		||||
void JSON::Value::shrink(unsigned int size){
 | 
			
		||||
  if (myType == ARRAY){
 | 
			
		||||
    while (arrVal.size() > size){arrVal.pop_front();}
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
  if (myType == OBJECT){
 | 
			
		||||
    while (objVal.size() > size){objVal.erase(objVal.begin());}
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// For object JSON::Value objects, removes the member with
 | 
			
		||||
/// the given name, if it exists. Has no effect otherwise.
 | 
			
		||||
void JSON::Value::removeMember(const std::string & name){
 | 
			
		||||
  objVal.erase(name);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// For object JSON::Value objects, returns true if the
 | 
			
		||||
/// given name is a member. Returns false otherwise.
 | 
			
		||||
bool JSON::Value::isMember(const std::string & name) const{
 | 
			
		||||
  return objVal.count(name) > 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Returns an iterator to the begin of the object map, if any.
 | 
			
		||||
JSON::ObjIter JSON::Value::ObjBegin(){
 | 
			
		||||
  return objVal.begin();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Returns an iterator to the end of the object map, if any.
 | 
			
		||||
JSON::ObjIter JSON::Value::ObjEnd(){
 | 
			
		||||
  return objVal.end();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Returns an iterator to the begin of the array, if any.
 | 
			
		||||
JSON::ArrIter JSON::Value::ArrBegin(){
 | 
			
		||||
  return arrVal.begin();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Returns an iterator to the end of the array, if any.
 | 
			
		||||
JSON::ArrIter JSON::Value::ArrEnd(){
 | 
			
		||||
  return arrVal.end();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Completely clears the contents of this value,
 | 
			
		||||
/// changing its type to NULL in the process.
 | 
			
		||||
void JSON::Value::null(){
 | 
			
		||||
  objVal.clear();
 | 
			
		||||
  arrVal.clear();
 | 
			
		||||
  strVal.clear();
 | 
			
		||||
  intVal = 0;
 | 
			
		||||
  myType = EMPTY;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Converts a std::string to a JSON::Value.
 | 
			
		||||
JSON::Value JSON::fromString(std::string json){
 | 
			
		||||
  std::istringstream is(json);
 | 
			
		||||
  return JSON::Value(is);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Converts a file to a JSON::Value.
 | 
			
		||||
JSON::Value JSON::fromFile(std::string filename){
 | 
			
		||||
  std::string Result;
 | 
			
		||||
  std::ifstream File;
 | 
			
		||||
  File.open(filename.c_str());
 | 
			
		||||
  while (File.good()){Result += File.get();}
 | 
			
		||||
  File.close( );
 | 
			
		||||
  return fromString(Result);
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										74
									
								
								util/json.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										74
									
								
								util/json.h
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,74 @@
 | 
			
		|||
/// \file json.h Holds all JSON-related headers.
 | 
			
		||||
 | 
			
		||||
#include <string>
 | 
			
		||||
#include <deque>
 | 
			
		||||
#include <map>
 | 
			
		||||
#include <istream>
 | 
			
		||||
 | 
			
		||||
/// JSON-related classes and functions
 | 
			
		||||
namespace JSON{
 | 
			
		||||
 | 
			
		||||
  /// Lists all types of JSON::Value.
 | 
			
		||||
  enum ValueType{ EMPTY, BOOL, INTEGER, STRING, ARRAY, OBJECT };
 | 
			
		||||
 | 
			
		||||
  class Value;//forward declaration for below typedef
 | 
			
		||||
 | 
			
		||||
  typedef std::map<std::string, Value>::iterator ObjIter;
 | 
			
		||||
  typedef std::deque<Value>::iterator ArrIter;
 | 
			
		||||
  
 | 
			
		||||
  /// A JSON::Value is either a string or an integer, but may also be an object, array or null.
 | 
			
		||||
  class Value{
 | 
			
		||||
    private:
 | 
			
		||||
      ValueType myType;
 | 
			
		||||
      long long int intVal;
 | 
			
		||||
      std::string strVal;
 | 
			
		||||
      std::deque<Value> arrVal;
 | 
			
		||||
      std::map<std::string, Value> objVal;
 | 
			
		||||
      std::string read_string(int separator, std::istream & fromstream);
 | 
			
		||||
      std::string string_escape(std::string val);
 | 
			
		||||
      int c2hex(int c);
 | 
			
		||||
    public:
 | 
			
		||||
      //constructors
 | 
			
		||||
      Value();
 | 
			
		||||
      Value(std::istream & fromstream);
 | 
			
		||||
      Value(const std::string & val);
 | 
			
		||||
      Value(const char * val);
 | 
			
		||||
      Value(long long int val);
 | 
			
		||||
      Value(bool val);
 | 
			
		||||
      //comparison operators
 | 
			
		||||
      bool operator==(const Value &rhs) const;
 | 
			
		||||
      bool operator!=(const Value &rhs) const;
 | 
			
		||||
      //assignment operators
 | 
			
		||||
      Value & operator=(const std::string &rhs);
 | 
			
		||||
      Value & operator=(const char * rhs);
 | 
			
		||||
      Value & operator=(const long long int &rhs);
 | 
			
		||||
      Value & operator=(const int &rhs);
 | 
			
		||||
      Value & operator=(const unsigned int &rhs);
 | 
			
		||||
      Value & operator=(const bool &rhs);
 | 
			
		||||
      //converts to basic types
 | 
			
		||||
      operator long long int();
 | 
			
		||||
      operator std::string();
 | 
			
		||||
      operator bool();
 | 
			
		||||
      //array operator for maps and arrays
 | 
			
		||||
      Value & operator[](const std::string i);
 | 
			
		||||
      Value & operator[](const char * i);
 | 
			
		||||
      Value & operator[](unsigned int i);
 | 
			
		||||
      //handy functions and others
 | 
			
		||||
      std::string toString();
 | 
			
		||||
      void append(const Value & rhs);
 | 
			
		||||
      void prepend(const Value & rhs);
 | 
			
		||||
      void shrink(unsigned int size);
 | 
			
		||||
      void removeMember(const std::string & name);
 | 
			
		||||
      bool isMember(const std::string & name) const;
 | 
			
		||||
      ObjIter ObjBegin();
 | 
			
		||||
      ObjIter ObjEnd();
 | 
			
		||||
      ArrIter ArrBegin();
 | 
			
		||||
      ArrIter ArrEnd();
 | 
			
		||||
      unsigned int size();
 | 
			
		||||
      void null();
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  Value fromString(std::string json);
 | 
			
		||||
  Value fromFile(std::string filename);
 | 
			
		||||
  
 | 
			
		||||
};
 | 
			
		||||
| 
						 | 
				
			
			@ -1,19 +0,0 @@
 | 
			
		|||
#ifndef JSON_AUTOLINK_H_INCLUDED
 | 
			
		||||
# define JSON_AUTOLINK_H_INCLUDED
 | 
			
		||||
 | 
			
		||||
# include "config.h"
 | 
			
		||||
 | 
			
		||||
# ifdef JSON_IN_CPPTL
 | 
			
		||||
#  include <cpptl/cpptl_autolink.h>
 | 
			
		||||
# endif
 | 
			
		||||
 | 
			
		||||
# if !defined(JSON_NO_AUTOLINK)  &&  !defined(JSON_DLL_BUILD)  &&  !defined(JSON_IN_CPPTL)
 | 
			
		||||
#  define CPPTL_AUTOLINK_NAME "json"
 | 
			
		||||
#  undef CPPTL_AUTOLINK_DLL
 | 
			
		||||
#  ifdef JSON_DLL
 | 
			
		||||
#   define CPPTL_AUTOLINK_DLL
 | 
			
		||||
#  endif
 | 
			
		||||
#  include "autolink.h"
 | 
			
		||||
# endif
 | 
			
		||||
 | 
			
		||||
#endif // JSON_AUTOLINK_H_INCLUDED
 | 
			
		||||
| 
						 | 
				
			
			@ -1,43 +0,0 @@
 | 
			
		|||
#ifndef JSON_CONFIG_H_INCLUDED
 | 
			
		||||
# define JSON_CONFIG_H_INCLUDED
 | 
			
		||||
 | 
			
		||||
/// If defined, indicates that json library is embedded in CppTL library.
 | 
			
		||||
//# define JSON_IN_CPPTL 1
 | 
			
		||||
 | 
			
		||||
/// If defined, indicates that json may leverage CppTL library
 | 
			
		||||
//#  define JSON_USE_CPPTL 1
 | 
			
		||||
/// If defined, indicates that cpptl vector based map should be used instead of std::map
 | 
			
		||||
/// as Value container.
 | 
			
		||||
//#  define JSON_USE_CPPTL_SMALLMAP 1
 | 
			
		||||
/// If defined, indicates that Json specific container should be used
 | 
			
		||||
/// (hash table & simple deque container with customizable allocator).
 | 
			
		||||
/// THIS FEATURE IS STILL EXPERIMENTAL!
 | 
			
		||||
//#  define JSON_VALUE_USE_INTERNAL_MAP 1
 | 
			
		||||
/// Force usage of standard new/malloc based allocator instead of memory pool based allocator.
 | 
			
		||||
/// The memory pools allocator used optimization (initializing Value and ValueInternalLink
 | 
			
		||||
/// as if it was a POD) that may cause some validation tool to report errors.
 | 
			
		||||
/// Only has effects if JSON_VALUE_USE_INTERNAL_MAP is defined.
 | 
			
		||||
//#  define JSON_USE_SIMPLE_INTERNAL_ALLOCATOR 1
 | 
			
		||||
 | 
			
		||||
/// If defined, indicates that Json use exception to report invalid type manipulation
 | 
			
		||||
/// instead of C assert macro.
 | 
			
		||||
# define JSON_USE_EXCEPTION 1
 | 
			
		||||
 | 
			
		||||
# ifdef JSON_IN_CPPTL
 | 
			
		||||
#  include <cpptl/config.h>
 | 
			
		||||
#  ifndef JSON_USE_CPPTL
 | 
			
		||||
#   define JSON_USE_CPPTL 1
 | 
			
		||||
#  endif
 | 
			
		||||
# endif
 | 
			
		||||
 | 
			
		||||
# ifdef JSON_IN_CPPTL
 | 
			
		||||
#  define JSON_API CPPTL_API
 | 
			
		||||
# elif defined(JSON_DLL_BUILD)
 | 
			
		||||
#  define JSON_API __declspec(dllexport)
 | 
			
		||||
# elif defined(JSON_DLL)
 | 
			
		||||
#  define JSON_API __declspec(dllimport)
 | 
			
		||||
# else
 | 
			
		||||
#  define JSON_API
 | 
			
		||||
# endif
 | 
			
		||||
 | 
			
		||||
#endif // JSON_CONFIG_H_INCLUDED
 | 
			
		||||
| 
						 | 
				
			
			@ -1,42 +0,0 @@
 | 
			
		|||
#ifndef CPPTL_JSON_FEATURES_H_INCLUDED
 | 
			
		||||
# define CPPTL_JSON_FEATURES_H_INCLUDED
 | 
			
		||||
 | 
			
		||||
# include "forwards.h"
 | 
			
		||||
 | 
			
		||||
namespace Json {
 | 
			
		||||
 | 
			
		||||
   /** \brief Configuration passed to reader and writer.
 | 
			
		||||
    * This configuration object can be used to force the Reader or Writer
 | 
			
		||||
    * to behave in a standard conforming way.
 | 
			
		||||
    */
 | 
			
		||||
   class JSON_API Features
 | 
			
		||||
   {
 | 
			
		||||
   public:
 | 
			
		||||
      /** \brief A configuration that allows all features and assumes all strings are UTF-8.
 | 
			
		||||
       * - C & C++ comments are allowed
 | 
			
		||||
       * - Root object can be any JSON value
 | 
			
		||||
       * - Assumes Value strings are encoded in UTF-8
 | 
			
		||||
       */
 | 
			
		||||
      static Features all();
 | 
			
		||||
 | 
			
		||||
      /** \brief A configuration that is strictly compatible with the JSON specification.
 | 
			
		||||
       * - Comments are forbidden.
 | 
			
		||||
       * - Root object must be either an array or an object value.
 | 
			
		||||
       * - Assumes Value strings are encoded in UTF-8
 | 
			
		||||
       */
 | 
			
		||||
      static Features strictMode();
 | 
			
		||||
 | 
			
		||||
      /** \brief Initialize the configuration like JsonConfig::allFeatures;
 | 
			
		||||
       */
 | 
			
		||||
      Features();
 | 
			
		||||
 | 
			
		||||
      /// \c true if comments are allowed. Default: \c true.
 | 
			
		||||
      bool allowComments_;
 | 
			
		||||
 | 
			
		||||
      /// \c true if root must be either an array or an object value. Default: \c false.
 | 
			
		||||
      bool strictRoot_;
 | 
			
		||||
   };
 | 
			
		||||
 | 
			
		||||
} // namespace Json
 | 
			
		||||
 | 
			
		||||
#endif // CPPTL_JSON_FEATURES_H_INCLUDED
 | 
			
		||||
| 
						 | 
				
			
			@ -1,39 +0,0 @@
 | 
			
		|||
#ifndef JSON_FORWARDS_H_INCLUDED
 | 
			
		||||
# define JSON_FORWARDS_H_INCLUDED
 | 
			
		||||
 | 
			
		||||
# include "config.h"
 | 
			
		||||
 | 
			
		||||
namespace Json {
 | 
			
		||||
 | 
			
		||||
   // writer.h
 | 
			
		||||
   class FastWriter;
 | 
			
		||||
   class StyledWriter;
 | 
			
		||||
 | 
			
		||||
   // reader.h
 | 
			
		||||
   class Reader;
 | 
			
		||||
 | 
			
		||||
   // features.h
 | 
			
		||||
   class Features;
 | 
			
		||||
 | 
			
		||||
   // value.h
 | 
			
		||||
   typedef int Int;
 | 
			
		||||
   typedef unsigned int UInt;
 | 
			
		||||
   class StaticString;
 | 
			
		||||
   class Path;
 | 
			
		||||
   class PathArgument;
 | 
			
		||||
   class Value;
 | 
			
		||||
   class ValueIteratorBase;
 | 
			
		||||
   class ValueIterator;
 | 
			
		||||
   class ValueConstIterator;
 | 
			
		||||
#ifdef JSON_VALUE_USE_INTERNAL_MAP
 | 
			
		||||
   class ValueAllocator;
 | 
			
		||||
   class ValueMapAllocator;
 | 
			
		||||
   class ValueInternalLink;
 | 
			
		||||
   class ValueInternalArray;
 | 
			
		||||
   class ValueInternalMap;
 | 
			
		||||
#endif // #ifdef JSON_VALUE_USE_INTERNAL_MAP
 | 
			
		||||
 | 
			
		||||
} // namespace Json
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
#endif // JSON_FORWARDS_H_INCLUDED
 | 
			
		||||
| 
						 | 
				
			
			@ -1,10 +0,0 @@
 | 
			
		|||
#ifndef JSON_JSON_H_INCLUDED
 | 
			
		||||
# define JSON_JSON_H_INCLUDED
 | 
			
		||||
 | 
			
		||||
# include "autolink.h"
 | 
			
		||||
# include "value.h"
 | 
			
		||||
# include "reader.h"
 | 
			
		||||
# include "writer.h"
 | 
			
		||||
# include "features.h"
 | 
			
		||||
 | 
			
		||||
#endif // JSON_JSON_H_INCLUDED
 | 
			
		||||
| 
						 | 
				
			
			@ -1,125 +0,0 @@
 | 
			
		|||
#ifndef JSONCPP_BATCHALLOCATOR_H_INCLUDED
 | 
			
		||||
# define JSONCPP_BATCHALLOCATOR_H_INCLUDED
 | 
			
		||||
 | 
			
		||||
# include <stdlib.h>
 | 
			
		||||
# include <assert.h>
 | 
			
		||||
 | 
			
		||||
# ifndef JSONCPP_DOC_EXCLUDE_IMPLEMENTATION
 | 
			
		||||
 | 
			
		||||
namespace Json {
 | 
			
		||||
 | 
			
		||||
/* Fast memory allocator.
 | 
			
		||||
 *
 | 
			
		||||
 * This memory allocator allocates memory for a batch of object (specified by
 | 
			
		||||
 * the page size, the number of object in each page).
 | 
			
		||||
 *
 | 
			
		||||
 * It does not allow the destruction of a single object. All the allocated objects
 | 
			
		||||
 * can be destroyed at once. The memory can be either released or reused for future
 | 
			
		||||
 * allocation.
 | 
			
		||||
 * 
 | 
			
		||||
 * The in-place new operator must be used to construct the object using the pointer
 | 
			
		||||
 * returned by allocate.
 | 
			
		||||
 */
 | 
			
		||||
template<typename AllocatedType
 | 
			
		||||
        ,const unsigned int objectPerAllocation>
 | 
			
		||||
class BatchAllocator
 | 
			
		||||
{
 | 
			
		||||
public:
 | 
			
		||||
   typedef AllocatedType Type;
 | 
			
		||||
 | 
			
		||||
   BatchAllocator( unsigned int objectsPerPage = 255 )
 | 
			
		||||
      : freeHead_( 0 )
 | 
			
		||||
      , objectsPerPage_( objectsPerPage )
 | 
			
		||||
   {
 | 
			
		||||
//      printf( "Size: %d => %s\n", sizeof(AllocatedType), typeid(AllocatedType).name() );
 | 
			
		||||
      assert( sizeof(AllocatedType) * objectPerAllocation >= sizeof(AllocatedType *) ); // We must be able to store a slist in the object free space.
 | 
			
		||||
      assert( objectsPerPage >= 16 );
 | 
			
		||||
      batches_ = allocateBatch( 0 );   // allocated a dummy page
 | 
			
		||||
      currentBatch_ = batches_;
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   ~BatchAllocator()
 | 
			
		||||
   {
 | 
			
		||||
      for ( BatchInfo *batch = batches_; batch;  )
 | 
			
		||||
      {
 | 
			
		||||
         BatchInfo *nextBatch = batch->next_;
 | 
			
		||||
         free( batch );
 | 
			
		||||
         batch = nextBatch;
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   /// allocate space for an array of objectPerAllocation object.
 | 
			
		||||
   /// @warning it is the responsability of the caller to call objects constructors.
 | 
			
		||||
   AllocatedType *allocate()
 | 
			
		||||
   {
 | 
			
		||||
      if ( freeHead_ ) // returns node from free list.
 | 
			
		||||
      {
 | 
			
		||||
         AllocatedType *object = freeHead_;
 | 
			
		||||
         freeHead_ = *(AllocatedType **)object;
 | 
			
		||||
         return object;
 | 
			
		||||
      }
 | 
			
		||||
      if ( currentBatch_->used_ == currentBatch_->end_ )
 | 
			
		||||
      {
 | 
			
		||||
         currentBatch_ = currentBatch_->next_;
 | 
			
		||||
         while ( currentBatch_  &&  currentBatch_->used_ == currentBatch_->end_ )
 | 
			
		||||
            currentBatch_ = currentBatch_->next_;
 | 
			
		||||
 | 
			
		||||
         if ( !currentBatch_  ) // no free batch found, allocate a new one
 | 
			
		||||
         { 
 | 
			
		||||
            currentBatch_ = allocateBatch( objectsPerPage_ );
 | 
			
		||||
            currentBatch_->next_ = batches_; // insert at the head of the list
 | 
			
		||||
            batches_ = currentBatch_;
 | 
			
		||||
         }
 | 
			
		||||
      }
 | 
			
		||||
      AllocatedType *allocated = currentBatch_->used_;
 | 
			
		||||
      currentBatch_->used_ += objectPerAllocation;
 | 
			
		||||
      return allocated;
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   /// Release the object.
 | 
			
		||||
   /// @warning it is the responsability of the caller to actually destruct the object.
 | 
			
		||||
   void release( AllocatedType *object )
 | 
			
		||||
   {
 | 
			
		||||
      assert( object != 0 );
 | 
			
		||||
      *(AllocatedType **)object = freeHead_;
 | 
			
		||||
      freeHead_ = object;
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
private:
 | 
			
		||||
   struct BatchInfo
 | 
			
		||||
   {
 | 
			
		||||
      BatchInfo *next_;
 | 
			
		||||
      AllocatedType *used_;
 | 
			
		||||
      AllocatedType *end_;
 | 
			
		||||
      AllocatedType buffer_[objectPerAllocation];
 | 
			
		||||
   };
 | 
			
		||||
 | 
			
		||||
   // disabled copy constructor and assignement operator.
 | 
			
		||||
   BatchAllocator( const BatchAllocator & );
 | 
			
		||||
   void operator =( const BatchAllocator &);
 | 
			
		||||
 | 
			
		||||
   static BatchInfo *allocateBatch( unsigned int objectsPerPage )
 | 
			
		||||
   {
 | 
			
		||||
      const unsigned int mallocSize = sizeof(BatchInfo) - sizeof(AllocatedType)* objectPerAllocation
 | 
			
		||||
                                + sizeof(AllocatedType) * objectPerAllocation * objectsPerPage;
 | 
			
		||||
      BatchInfo *batch = static_cast<BatchInfo*>( malloc( mallocSize ) );
 | 
			
		||||
      batch->next_ = 0;
 | 
			
		||||
      batch->used_ = batch->buffer_;
 | 
			
		||||
      batch->end_ = batch->buffer_ + objectsPerPage;
 | 
			
		||||
      return batch;
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   BatchInfo *batches_;
 | 
			
		||||
   BatchInfo *currentBatch_;
 | 
			
		||||
   /// Head of a single linked list within the allocated space of freeed object
 | 
			
		||||
   AllocatedType *freeHead_;
 | 
			
		||||
   unsigned int objectsPerPage_;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
} // namespace Json
 | 
			
		||||
 | 
			
		||||
# endif // ifndef JSONCPP_DOC_INCLUDE_IMPLEMENTATION
 | 
			
		||||
 | 
			
		||||
#endif // JSONCPP_BATCHALLOCATOR_H_INCLUDED
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -1,448 +0,0 @@
 | 
			
		|||
// included by json_value.cpp
 | 
			
		||||
// everything is within Json namespace
 | 
			
		||||
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// class ValueInternalArray
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
 | 
			
		||||
ValueArrayAllocator::~ValueArrayAllocator()
 | 
			
		||||
{
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// class DefaultValueArrayAllocator
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
#ifdef JSON_USE_SIMPLE_INTERNAL_ALLOCATOR
 | 
			
		||||
class DefaultValueArrayAllocator : public ValueArrayAllocator
 | 
			
		||||
{
 | 
			
		||||
public: // overridden from ValueArrayAllocator
 | 
			
		||||
   virtual ~DefaultValueArrayAllocator()
 | 
			
		||||
   {
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   virtual ValueInternalArray *newArray()
 | 
			
		||||
   {
 | 
			
		||||
      return new ValueInternalArray();
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   virtual ValueInternalArray *newArrayCopy( const ValueInternalArray &other )
 | 
			
		||||
   {
 | 
			
		||||
      return new ValueInternalArray( other );
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   virtual void destructArray( ValueInternalArray *array )
 | 
			
		||||
   {
 | 
			
		||||
      delete array;
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   virtual void reallocateArrayPageIndex( Value **&indexes, 
 | 
			
		||||
                                          ValueInternalArray::PageIndex &indexCount,
 | 
			
		||||
                                          ValueInternalArray::PageIndex minNewIndexCount )
 | 
			
		||||
   {
 | 
			
		||||
      ValueInternalArray::PageIndex newIndexCount = (indexCount*3)/2 + 1;
 | 
			
		||||
      if ( minNewIndexCount > newIndexCount )
 | 
			
		||||
         newIndexCount = minNewIndexCount;
 | 
			
		||||
      void *newIndexes = realloc( indexes, sizeof(Value*) * newIndexCount );
 | 
			
		||||
      if ( !newIndexes )
 | 
			
		||||
         throw std::bad_alloc();
 | 
			
		||||
      indexCount = newIndexCount;
 | 
			
		||||
      indexes = static_cast<Value **>( newIndexes );
 | 
			
		||||
   }
 | 
			
		||||
   virtual void releaseArrayPageIndex( Value **indexes, 
 | 
			
		||||
                                       ValueInternalArray::PageIndex indexCount )
 | 
			
		||||
   {
 | 
			
		||||
      if ( indexes )
 | 
			
		||||
         free( indexes );
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   virtual Value *allocateArrayPage()
 | 
			
		||||
   {
 | 
			
		||||
      return static_cast<Value *>( malloc( sizeof(Value) * ValueInternalArray::itemsPerPage ) );
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   virtual void releaseArrayPage( Value *value )
 | 
			
		||||
   {
 | 
			
		||||
      if ( value )
 | 
			
		||||
         free( value );
 | 
			
		||||
   }
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
#else // #ifdef JSON_USE_SIMPLE_INTERNAL_ALLOCATOR
 | 
			
		||||
/// @todo make this thread-safe (lock when accessign batch allocator)
 | 
			
		||||
class DefaultValueArrayAllocator : public ValueArrayAllocator
 | 
			
		||||
{
 | 
			
		||||
public: // overridden from ValueArrayAllocator
 | 
			
		||||
   virtual ~DefaultValueArrayAllocator()
 | 
			
		||||
   {
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   virtual ValueInternalArray *newArray()
 | 
			
		||||
   {
 | 
			
		||||
      ValueInternalArray *array = arraysAllocator_.allocate();
 | 
			
		||||
      new (array) ValueInternalArray(); // placement new
 | 
			
		||||
      return array;
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   virtual ValueInternalArray *newArrayCopy( const ValueInternalArray &other )
 | 
			
		||||
   {
 | 
			
		||||
      ValueInternalArray *array = arraysAllocator_.allocate();
 | 
			
		||||
      new (array) ValueInternalArray( other ); // placement new
 | 
			
		||||
      return array;
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   virtual void destructArray( ValueInternalArray *array )
 | 
			
		||||
   {
 | 
			
		||||
      if ( array )
 | 
			
		||||
      {
 | 
			
		||||
         array->~ValueInternalArray();
 | 
			
		||||
         arraysAllocator_.release( array );
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   virtual void reallocateArrayPageIndex( Value **&indexes, 
 | 
			
		||||
                                          ValueInternalArray::PageIndex &indexCount,
 | 
			
		||||
                                          ValueInternalArray::PageIndex minNewIndexCount )
 | 
			
		||||
   {
 | 
			
		||||
      ValueInternalArray::PageIndex newIndexCount = (indexCount*3)/2 + 1;
 | 
			
		||||
      if ( minNewIndexCount > newIndexCount )
 | 
			
		||||
         newIndexCount = minNewIndexCount;
 | 
			
		||||
      void *newIndexes = realloc( indexes, sizeof(Value*) * newIndexCount );
 | 
			
		||||
      if ( !newIndexes )
 | 
			
		||||
         throw std::bad_alloc();
 | 
			
		||||
      indexCount = newIndexCount;
 | 
			
		||||
      indexes = static_cast<Value **>( newIndexes );
 | 
			
		||||
   }
 | 
			
		||||
   virtual void releaseArrayPageIndex( Value **indexes, 
 | 
			
		||||
                                       ValueInternalArray::PageIndex indexCount )
 | 
			
		||||
   {
 | 
			
		||||
      if ( indexes )
 | 
			
		||||
         free( indexes );
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   virtual Value *allocateArrayPage()
 | 
			
		||||
   {
 | 
			
		||||
      return static_cast<Value *>( pagesAllocator_.allocate() );
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   virtual void releaseArrayPage( Value *value )
 | 
			
		||||
   {
 | 
			
		||||
      if ( value )
 | 
			
		||||
         pagesAllocator_.release( value );
 | 
			
		||||
   }
 | 
			
		||||
private:
 | 
			
		||||
   BatchAllocator<ValueInternalArray,1> arraysAllocator_;
 | 
			
		||||
   BatchAllocator<Value,ValueInternalArray::itemsPerPage> pagesAllocator_;
 | 
			
		||||
};
 | 
			
		||||
#endif // #ifdef JSON_USE_SIMPLE_INTERNAL_ALLOCATOR
 | 
			
		||||
 | 
			
		||||
static ValueArrayAllocator *&arrayAllocator()
 | 
			
		||||
{
 | 
			
		||||
   static DefaultValueArrayAllocator defaultAllocator;
 | 
			
		||||
   static ValueArrayAllocator *arrayAllocator = &defaultAllocator;
 | 
			
		||||
   return arrayAllocator;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static struct DummyArrayAllocatorInitializer {
 | 
			
		||||
   DummyArrayAllocatorInitializer() 
 | 
			
		||||
   {
 | 
			
		||||
      arrayAllocator();      // ensure arrayAllocator() statics are initialized before main().
 | 
			
		||||
   }
 | 
			
		||||
} dummyArrayAllocatorInitializer;
 | 
			
		||||
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// class ValueInternalArray
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
bool 
 | 
			
		||||
ValueInternalArray::equals( const IteratorState &x, 
 | 
			
		||||
                            const IteratorState &other )
 | 
			
		||||
{
 | 
			
		||||
   return x.array_ == other.array_  
 | 
			
		||||
          &&  x.currentItemIndex_ == other.currentItemIndex_  
 | 
			
		||||
          &&  x.currentPageIndex_ == other.currentPageIndex_;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
ValueInternalArray::increment( IteratorState &it )
 | 
			
		||||
{
 | 
			
		||||
   JSON_ASSERT_MESSAGE( it.array_  &&
 | 
			
		||||
      (it.currentPageIndex_ - it.array_->pages_)*itemsPerPage + it.currentItemIndex_
 | 
			
		||||
      != it.array_->size_,
 | 
			
		||||
      "ValueInternalArray::increment(): moving iterator beyond end" );
 | 
			
		||||
   ++(it.currentItemIndex_);
 | 
			
		||||
   if ( it.currentItemIndex_ == itemsPerPage )
 | 
			
		||||
   {
 | 
			
		||||
      it.currentItemIndex_ = 0;
 | 
			
		||||
      ++(it.currentPageIndex_);
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
ValueInternalArray::decrement( IteratorState &it )
 | 
			
		||||
{
 | 
			
		||||
   JSON_ASSERT_MESSAGE( it.array_  &&  it.currentPageIndex_ == it.array_->pages_ 
 | 
			
		||||
                        &&  it.currentItemIndex_ == 0,
 | 
			
		||||
      "ValueInternalArray::decrement(): moving iterator beyond end" );
 | 
			
		||||
   if ( it.currentItemIndex_ == 0 )
 | 
			
		||||
   {
 | 
			
		||||
      it.currentItemIndex_ = itemsPerPage-1;
 | 
			
		||||
      --(it.currentPageIndex_);
 | 
			
		||||
   }
 | 
			
		||||
   else
 | 
			
		||||
   {
 | 
			
		||||
      --(it.currentItemIndex_);
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
Value &
 | 
			
		||||
ValueInternalArray::unsafeDereference( const IteratorState &it )
 | 
			
		||||
{
 | 
			
		||||
   return (*(it.currentPageIndex_))[it.currentItemIndex_];
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
Value &
 | 
			
		||||
ValueInternalArray::dereference( const IteratorState &it )
 | 
			
		||||
{
 | 
			
		||||
   JSON_ASSERT_MESSAGE( it.array_  &&
 | 
			
		||||
      (it.currentPageIndex_ - it.array_->pages_)*itemsPerPage + it.currentItemIndex_
 | 
			
		||||
      < it.array_->size_,
 | 
			
		||||
      "ValueInternalArray::dereference(): dereferencing invalid iterator" );
 | 
			
		||||
   return unsafeDereference( it );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
ValueInternalArray::makeBeginIterator( IteratorState &it ) const
 | 
			
		||||
{
 | 
			
		||||
   it.array_ = const_cast<ValueInternalArray *>( this );
 | 
			
		||||
   it.currentItemIndex_ = 0;
 | 
			
		||||
   it.currentPageIndex_ = pages_;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
ValueInternalArray::makeIterator( IteratorState &it, ArrayIndex index ) const
 | 
			
		||||
{
 | 
			
		||||
   it.array_ = const_cast<ValueInternalArray *>( this );
 | 
			
		||||
   it.currentItemIndex_ = index % itemsPerPage;
 | 
			
		||||
   it.currentPageIndex_ = pages_ + index / itemsPerPage;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
ValueInternalArray::makeEndIterator( IteratorState &it ) const
 | 
			
		||||
{
 | 
			
		||||
   makeIterator( it, size_ );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
ValueInternalArray::ValueInternalArray()
 | 
			
		||||
   : pages_( 0 )
 | 
			
		||||
   , size_( 0 )
 | 
			
		||||
   , pageCount_( 0 )
 | 
			
		||||
{
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
ValueInternalArray::ValueInternalArray( const ValueInternalArray &other )
 | 
			
		||||
   : pages_( 0 )
 | 
			
		||||
   , pageCount_( 0 )
 | 
			
		||||
   , size_( other.size_ )
 | 
			
		||||
{
 | 
			
		||||
   PageIndex minNewPages = other.size_ / itemsPerPage;
 | 
			
		||||
   arrayAllocator()->reallocateArrayPageIndex( pages_, pageCount_, minNewPages );
 | 
			
		||||
   JSON_ASSERT_MESSAGE( pageCount_ >= minNewPages, 
 | 
			
		||||
                        "ValueInternalArray::reserve(): bad reallocation" );
 | 
			
		||||
   IteratorState itOther;
 | 
			
		||||
   other.makeBeginIterator( itOther );
 | 
			
		||||
   Value *value;
 | 
			
		||||
   for ( ArrayIndex index = 0; index < size_; ++index, increment(itOther) )
 | 
			
		||||
   {
 | 
			
		||||
      if ( index % itemsPerPage == 0 )
 | 
			
		||||
      {
 | 
			
		||||
         PageIndex pageIndex = index / itemsPerPage;
 | 
			
		||||
         value = arrayAllocator()->allocateArrayPage();
 | 
			
		||||
         pages_[pageIndex] = value;
 | 
			
		||||
      }
 | 
			
		||||
      new (value) Value( dereference( itOther ) );
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
ValueInternalArray &
 | 
			
		||||
ValueInternalArray::operator =( const ValueInternalArray &other )
 | 
			
		||||
{
 | 
			
		||||
   ValueInternalArray temp( other );
 | 
			
		||||
   swap( temp );
 | 
			
		||||
   return *this;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
ValueInternalArray::~ValueInternalArray()
 | 
			
		||||
{
 | 
			
		||||
   // destroy all constructed items
 | 
			
		||||
   IteratorState it;
 | 
			
		||||
   IteratorState itEnd;
 | 
			
		||||
   makeBeginIterator( it);
 | 
			
		||||
   makeEndIterator( itEnd );
 | 
			
		||||
   for ( ; !equals(it,itEnd); increment(it) )
 | 
			
		||||
   {
 | 
			
		||||
      Value *value = &dereference(it);
 | 
			
		||||
      value->~Value();
 | 
			
		||||
   }
 | 
			
		||||
   // release all pages
 | 
			
		||||
   PageIndex lastPageIndex = size_ / itemsPerPage;
 | 
			
		||||
   for ( PageIndex pageIndex = 0; pageIndex < lastPageIndex; ++pageIndex )
 | 
			
		||||
      arrayAllocator()->releaseArrayPage( pages_[pageIndex] );
 | 
			
		||||
   // release pages index
 | 
			
		||||
   arrayAllocator()->releaseArrayPageIndex( pages_, pageCount_ );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
ValueInternalArray::swap( ValueInternalArray &other )
 | 
			
		||||
{
 | 
			
		||||
   Value **tempPages = pages_;
 | 
			
		||||
   pages_ = other.pages_;
 | 
			
		||||
   other.pages_ = tempPages;
 | 
			
		||||
   ArrayIndex tempSize = size_;
 | 
			
		||||
   size_ = other.size_;
 | 
			
		||||
   other.size_ = tempSize;
 | 
			
		||||
   PageIndex tempPageCount = pageCount_;
 | 
			
		||||
   pageCount_ = other.pageCount_;
 | 
			
		||||
   other.pageCount_ = tempPageCount;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
ValueInternalArray::clear()
 | 
			
		||||
{
 | 
			
		||||
   ValueInternalArray dummy;
 | 
			
		||||
   swap( dummy );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
ValueInternalArray::resize( ArrayIndex newSize )
 | 
			
		||||
{
 | 
			
		||||
   if ( newSize == 0 )
 | 
			
		||||
      clear();
 | 
			
		||||
   else if ( newSize < size_ )
 | 
			
		||||
   {
 | 
			
		||||
      IteratorState it;
 | 
			
		||||
      IteratorState itEnd;
 | 
			
		||||
      makeIterator( it, newSize );
 | 
			
		||||
      makeIterator( itEnd, size_ );
 | 
			
		||||
      for ( ; !equals(it,itEnd); increment(it) )
 | 
			
		||||
      {
 | 
			
		||||
         Value *value = &dereference(it);
 | 
			
		||||
         value->~Value();
 | 
			
		||||
      }
 | 
			
		||||
      PageIndex pageIndex = (newSize + itemsPerPage - 1) / itemsPerPage;
 | 
			
		||||
      PageIndex lastPageIndex = size_ / itemsPerPage;
 | 
			
		||||
      for ( ; pageIndex < lastPageIndex; ++pageIndex )
 | 
			
		||||
         arrayAllocator()->releaseArrayPage( pages_[pageIndex] );
 | 
			
		||||
      size_ = newSize;
 | 
			
		||||
   }
 | 
			
		||||
   else if ( newSize > size_ )
 | 
			
		||||
      resolveReference( newSize );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
ValueInternalArray::makeIndexValid( ArrayIndex index )
 | 
			
		||||
{
 | 
			
		||||
   // Need to enlarge page index ?
 | 
			
		||||
   if ( index >= pageCount_ * itemsPerPage )
 | 
			
		||||
   {
 | 
			
		||||
      PageIndex minNewPages = (index + 1) / itemsPerPage;
 | 
			
		||||
      arrayAllocator()->reallocateArrayPageIndex( pages_, pageCount_, minNewPages );
 | 
			
		||||
      JSON_ASSERT_MESSAGE( pageCount_ >= minNewPages, "ValueInternalArray::reserve(): bad reallocation" );
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   // Need to allocate new pages ?
 | 
			
		||||
   ArrayIndex nextPageIndex = 
 | 
			
		||||
      (size_ % itemsPerPage) != 0 ? size_ - (size_%itemsPerPage) + itemsPerPage
 | 
			
		||||
                                  : size_;
 | 
			
		||||
   if ( nextPageIndex <= index )
 | 
			
		||||
   {
 | 
			
		||||
      PageIndex pageIndex = nextPageIndex / itemsPerPage;
 | 
			
		||||
      PageIndex pageToAllocate = (index - nextPageIndex) / itemsPerPage + 1;
 | 
			
		||||
      for ( ; pageToAllocate-- > 0; ++pageIndex )
 | 
			
		||||
         pages_[pageIndex] = arrayAllocator()->allocateArrayPage();
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   // Initialize all new entries
 | 
			
		||||
   IteratorState it;
 | 
			
		||||
   IteratorState itEnd;
 | 
			
		||||
   makeIterator( it, size_ );
 | 
			
		||||
   size_ = index + 1;
 | 
			
		||||
   makeIterator( itEnd, size_ );
 | 
			
		||||
   for ( ; !equals(it,itEnd); increment(it) )
 | 
			
		||||
   {
 | 
			
		||||
      Value *value = &dereference(it);
 | 
			
		||||
      new (value) Value(); // Construct a default value using placement new
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Value &
 | 
			
		||||
ValueInternalArray::resolveReference( ArrayIndex index )
 | 
			
		||||
{
 | 
			
		||||
   if ( index >= size_ )
 | 
			
		||||
      makeIndexValid( index );
 | 
			
		||||
   return pages_[index/itemsPerPage][index%itemsPerPage];
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
Value *
 | 
			
		||||
ValueInternalArray::find( ArrayIndex index ) const
 | 
			
		||||
{
 | 
			
		||||
   if ( index >= size_ )
 | 
			
		||||
      return 0;
 | 
			
		||||
   return &(pages_[index/itemsPerPage][index%itemsPerPage]);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
ValueInternalArray::ArrayIndex 
 | 
			
		||||
ValueInternalArray::size() const
 | 
			
		||||
{
 | 
			
		||||
   return size_;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
int 
 | 
			
		||||
ValueInternalArray::distance( const IteratorState &x, const IteratorState &y )
 | 
			
		||||
{
 | 
			
		||||
   return indexOf(y) - indexOf(x);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
ValueInternalArray::ArrayIndex 
 | 
			
		||||
ValueInternalArray::indexOf( const IteratorState &iterator )
 | 
			
		||||
{
 | 
			
		||||
   if ( !iterator.array_ )
 | 
			
		||||
      return ArrayIndex(-1);
 | 
			
		||||
   return ArrayIndex(
 | 
			
		||||
      (iterator.currentPageIndex_ - iterator.array_->pages_) * itemsPerPage 
 | 
			
		||||
      + iterator.currentItemIndex_ );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
int 
 | 
			
		||||
ValueInternalArray::compare( const ValueInternalArray &other ) const
 | 
			
		||||
{
 | 
			
		||||
   int sizeDiff( size_ - other.size_ );
 | 
			
		||||
   if ( sizeDiff != 0 )
 | 
			
		||||
      return sizeDiff;
 | 
			
		||||
   
 | 
			
		||||
   for ( ArrayIndex index =0; index < size_; ++index )
 | 
			
		||||
   {
 | 
			
		||||
      int diff = pages_[index/itemsPerPage][index%itemsPerPage].compare( 
 | 
			
		||||
         other.pages_[index/itemsPerPage][index%itemsPerPage] );
 | 
			
		||||
      if ( diff != 0 )
 | 
			
		||||
         return diff;
 | 
			
		||||
   }
 | 
			
		||||
   return 0;
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			@ -1,607 +0,0 @@
 | 
			
		|||
// included by json_value.cpp
 | 
			
		||||
// everything is within Json namespace
 | 
			
		||||
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// class ValueInternalMap
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
 | 
			
		||||
/** \internal MUST be safely initialized using memset( this, 0, sizeof(ValueInternalLink) );
 | 
			
		||||
   * This optimization is used by the fast allocator.
 | 
			
		||||
   */
 | 
			
		||||
ValueInternalLink::ValueInternalLink()
 | 
			
		||||
   : previous_( 0 )
 | 
			
		||||
   , next_( 0 )
 | 
			
		||||
{
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
ValueInternalLink::~ValueInternalLink()
 | 
			
		||||
{ 
 | 
			
		||||
   for ( int index =0; index < itemPerLink; ++index )
 | 
			
		||||
   {
 | 
			
		||||
      if ( !items_[index].isItemAvailable() )
 | 
			
		||||
      {
 | 
			
		||||
         if ( !items_[index].isMemberNameStatic() )
 | 
			
		||||
            free( keys_[index] );
 | 
			
		||||
      }
 | 
			
		||||
      else
 | 
			
		||||
         break;
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
ValueMapAllocator::~ValueMapAllocator()
 | 
			
		||||
{
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
#ifdef JSON_USE_SIMPLE_INTERNAL_ALLOCATOR
 | 
			
		||||
class DefaultValueMapAllocator : public ValueMapAllocator
 | 
			
		||||
{
 | 
			
		||||
public: // overridden from ValueMapAllocator
 | 
			
		||||
   virtual ValueInternalMap *newMap()
 | 
			
		||||
   {
 | 
			
		||||
      return new ValueInternalMap();
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   virtual ValueInternalMap *newMapCopy( const ValueInternalMap &other )
 | 
			
		||||
   {
 | 
			
		||||
      return new ValueInternalMap( other );
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   virtual void destructMap( ValueInternalMap *map )
 | 
			
		||||
   {
 | 
			
		||||
      delete map;
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   virtual ValueInternalLink *allocateMapBuckets( unsigned int size )
 | 
			
		||||
   {
 | 
			
		||||
      return new ValueInternalLink[size];
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   virtual void releaseMapBuckets( ValueInternalLink *links )
 | 
			
		||||
   {
 | 
			
		||||
      delete [] links;
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   virtual ValueInternalLink *allocateMapLink()
 | 
			
		||||
   {
 | 
			
		||||
      return new ValueInternalLink();
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   virtual void releaseMapLink( ValueInternalLink *link )
 | 
			
		||||
   {
 | 
			
		||||
      delete link;
 | 
			
		||||
   }
 | 
			
		||||
};
 | 
			
		||||
#else
 | 
			
		||||
/// @todo make this thread-safe (lock when accessign batch allocator)
 | 
			
		||||
class DefaultValueMapAllocator : public ValueMapAllocator
 | 
			
		||||
{
 | 
			
		||||
public: // overridden from ValueMapAllocator
 | 
			
		||||
   virtual ValueInternalMap *newMap()
 | 
			
		||||
   {
 | 
			
		||||
      ValueInternalMap *map = mapsAllocator_.allocate();
 | 
			
		||||
      new (map) ValueInternalMap(); // placement new
 | 
			
		||||
      return map;
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   virtual ValueInternalMap *newMapCopy( const ValueInternalMap &other )
 | 
			
		||||
   {
 | 
			
		||||
      ValueInternalMap *map = mapsAllocator_.allocate();
 | 
			
		||||
      new (map) ValueInternalMap( other ); // placement new
 | 
			
		||||
      return map;
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   virtual void destructMap( ValueInternalMap *map )
 | 
			
		||||
   {
 | 
			
		||||
      if ( map )
 | 
			
		||||
      {
 | 
			
		||||
         map->~ValueInternalMap();
 | 
			
		||||
         mapsAllocator_.release( map );
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   virtual ValueInternalLink *allocateMapBuckets( unsigned int size )
 | 
			
		||||
   {
 | 
			
		||||
      return new ValueInternalLink[size];
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   virtual void releaseMapBuckets( ValueInternalLink *links )
 | 
			
		||||
   {
 | 
			
		||||
      delete [] links;
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   virtual ValueInternalLink *allocateMapLink()
 | 
			
		||||
   {
 | 
			
		||||
      ValueInternalLink *link = linksAllocator_.allocate();
 | 
			
		||||
      memset( link, 0, sizeof(ValueInternalLink) );
 | 
			
		||||
      return link;
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   virtual void releaseMapLink( ValueInternalLink *link )
 | 
			
		||||
   {
 | 
			
		||||
      link->~ValueInternalLink();
 | 
			
		||||
      linksAllocator_.release( link );
 | 
			
		||||
   }
 | 
			
		||||
private:
 | 
			
		||||
   BatchAllocator<ValueInternalMap,1> mapsAllocator_;
 | 
			
		||||
   BatchAllocator<ValueInternalLink,1> linksAllocator_;
 | 
			
		||||
};
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
static ValueMapAllocator *&mapAllocator()
 | 
			
		||||
{
 | 
			
		||||
   static DefaultValueMapAllocator defaultAllocator;
 | 
			
		||||
   static ValueMapAllocator *mapAllocator = &defaultAllocator;
 | 
			
		||||
   return mapAllocator;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static struct DummyMapAllocatorInitializer {
 | 
			
		||||
   DummyMapAllocatorInitializer() 
 | 
			
		||||
   {
 | 
			
		||||
      mapAllocator();      // ensure mapAllocator() statics are initialized before main().
 | 
			
		||||
   }
 | 
			
		||||
} dummyMapAllocatorInitializer;
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
// h(K) = value * K >> w ; with w = 32 & K prime w.r.t. 2^32.
 | 
			
		||||
 | 
			
		||||
/*
 | 
			
		||||
use linked list hash map. 
 | 
			
		||||
buckets array is a container.
 | 
			
		||||
linked list element contains 6 key/values. (memory = (16+4) * 6 + 4 = 124)
 | 
			
		||||
value have extra state: valid, available, deleted
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
ValueInternalMap::ValueInternalMap()
 | 
			
		||||
   : buckets_( 0 )
 | 
			
		||||
   , tailLink_( 0 )
 | 
			
		||||
   , bucketsSize_( 0 )
 | 
			
		||||
   , itemCount_( 0 )
 | 
			
		||||
{
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
ValueInternalMap::ValueInternalMap( const ValueInternalMap &other )
 | 
			
		||||
   : buckets_( 0 )
 | 
			
		||||
   , tailLink_( 0 )
 | 
			
		||||
   , bucketsSize_( 0 )
 | 
			
		||||
   , itemCount_( 0 )
 | 
			
		||||
{
 | 
			
		||||
   reserve( other.itemCount_ );
 | 
			
		||||
   IteratorState it;
 | 
			
		||||
   IteratorState itEnd;
 | 
			
		||||
   other.makeBeginIterator( it );
 | 
			
		||||
   other.makeEndIterator( itEnd );
 | 
			
		||||
   for ( ; !equals(it,itEnd); increment(it) )
 | 
			
		||||
   {
 | 
			
		||||
      bool isStatic;
 | 
			
		||||
      const char *memberName = key( it, isStatic );
 | 
			
		||||
      const Value &aValue = value( it );
 | 
			
		||||
      resolveReference(memberName, isStatic) = aValue;
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
ValueInternalMap &
 | 
			
		||||
ValueInternalMap::operator =( const ValueInternalMap &other )
 | 
			
		||||
{
 | 
			
		||||
   ValueInternalMap dummy( other );
 | 
			
		||||
   swap( dummy );
 | 
			
		||||
   return *this;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
ValueInternalMap::~ValueInternalMap()
 | 
			
		||||
{
 | 
			
		||||
   if ( buckets_ )
 | 
			
		||||
   {
 | 
			
		||||
      for ( BucketIndex bucketIndex =0; bucketIndex < bucketsSize_; ++bucketIndex )
 | 
			
		||||
      {
 | 
			
		||||
         ValueInternalLink *link = buckets_[bucketIndex].next_;
 | 
			
		||||
         while ( link )
 | 
			
		||||
         {
 | 
			
		||||
            ValueInternalLink *linkToRelease = link;
 | 
			
		||||
            link = link->next_;
 | 
			
		||||
            mapAllocator()->releaseMapLink( linkToRelease );
 | 
			
		||||
         }
 | 
			
		||||
      }
 | 
			
		||||
      mapAllocator()->releaseMapBuckets( buckets_ );
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
ValueInternalMap::swap( ValueInternalMap &other )
 | 
			
		||||
{
 | 
			
		||||
   ValueInternalLink *tempBuckets = buckets_;
 | 
			
		||||
   buckets_ = other.buckets_;
 | 
			
		||||
   other.buckets_ = tempBuckets;
 | 
			
		||||
   ValueInternalLink *tempTailLink = tailLink_;
 | 
			
		||||
   tailLink_ = other.tailLink_;
 | 
			
		||||
   other.tailLink_ = tempTailLink;
 | 
			
		||||
   BucketIndex tempBucketsSize = bucketsSize_;
 | 
			
		||||
   bucketsSize_ = other.bucketsSize_;
 | 
			
		||||
   other.bucketsSize_ = tempBucketsSize;
 | 
			
		||||
   BucketIndex tempItemCount = itemCount_;
 | 
			
		||||
   itemCount_ = other.itemCount_;
 | 
			
		||||
   other.itemCount_ = tempItemCount;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
ValueInternalMap::clear()
 | 
			
		||||
{
 | 
			
		||||
   ValueInternalMap dummy;
 | 
			
		||||
   swap( dummy );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
ValueInternalMap::BucketIndex 
 | 
			
		||||
ValueInternalMap::size() const
 | 
			
		||||
{
 | 
			
		||||
   return itemCount_;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
bool 
 | 
			
		||||
ValueInternalMap::reserveDelta( BucketIndex growth )
 | 
			
		||||
{
 | 
			
		||||
   return reserve( itemCount_ + growth );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
bool 
 | 
			
		||||
ValueInternalMap::reserve( BucketIndex newItemCount )
 | 
			
		||||
{
 | 
			
		||||
   if ( !buckets_  &&  newItemCount > 0 )
 | 
			
		||||
   {
 | 
			
		||||
      buckets_ = mapAllocator()->allocateMapBuckets( 1 );
 | 
			
		||||
      bucketsSize_ = 1;
 | 
			
		||||
      tailLink_ = &buckets_[0];
 | 
			
		||||
   }
 | 
			
		||||
//   BucketIndex idealBucketCount = (newItemCount + ValueInternalLink::itemPerLink) / ValueInternalLink::itemPerLink;
 | 
			
		||||
   return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
const Value *
 | 
			
		||||
ValueInternalMap::find( const char *key ) const
 | 
			
		||||
{
 | 
			
		||||
   if ( !bucketsSize_ )
 | 
			
		||||
      return 0;
 | 
			
		||||
   HashKey hashedKey = hash( key );
 | 
			
		||||
   BucketIndex bucketIndex = hashedKey % bucketsSize_;
 | 
			
		||||
   for ( const ValueInternalLink *current = &buckets_[bucketIndex]; 
 | 
			
		||||
         current != 0; 
 | 
			
		||||
         current = current->next_ )
 | 
			
		||||
   {
 | 
			
		||||
      for ( BucketIndex index=0; index < ValueInternalLink::itemPerLink; ++index )
 | 
			
		||||
      {
 | 
			
		||||
         if ( current->items_[index].isItemAvailable() )
 | 
			
		||||
            return 0;
 | 
			
		||||
         if ( strcmp( key, current->keys_[index] ) == 0 )
 | 
			
		||||
            return ¤t->items_[index];
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
   return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
Value *
 | 
			
		||||
ValueInternalMap::find( const char *key )
 | 
			
		||||
{
 | 
			
		||||
   const ValueInternalMap *constThis = this;
 | 
			
		||||
   return const_cast<Value *>( constThis->find( key ) );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
Value &
 | 
			
		||||
ValueInternalMap::resolveReference( const char *key,
 | 
			
		||||
                                    bool isStatic )
 | 
			
		||||
{
 | 
			
		||||
   HashKey hashedKey = hash( key );
 | 
			
		||||
   if ( bucketsSize_ )
 | 
			
		||||
   {
 | 
			
		||||
      BucketIndex bucketIndex = hashedKey % bucketsSize_;
 | 
			
		||||
      ValueInternalLink **previous = 0;
 | 
			
		||||
      BucketIndex index;
 | 
			
		||||
      for ( ValueInternalLink *current = &buckets_[bucketIndex]; 
 | 
			
		||||
            current != 0; 
 | 
			
		||||
            previous = ¤t->next_, current = current->next_ )
 | 
			
		||||
      {
 | 
			
		||||
         for ( index=0; index < ValueInternalLink::itemPerLink; ++index )
 | 
			
		||||
         {
 | 
			
		||||
            if ( current->items_[index].isItemAvailable() )
 | 
			
		||||
               return setNewItem( key, isStatic, current, index );
 | 
			
		||||
            if ( strcmp( key, current->keys_[index] ) == 0 )
 | 
			
		||||
               return current->items_[index];
 | 
			
		||||
         }
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   reserveDelta( 1 );
 | 
			
		||||
   return unsafeAdd( key, isStatic, hashedKey );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
ValueInternalMap::remove( const char *key )
 | 
			
		||||
{
 | 
			
		||||
   HashKey hashedKey = hash( key );
 | 
			
		||||
   if ( !bucketsSize_ )
 | 
			
		||||
      return;
 | 
			
		||||
   BucketIndex bucketIndex = hashedKey % bucketsSize_;
 | 
			
		||||
   for ( ValueInternalLink *link = &buckets_[bucketIndex]; 
 | 
			
		||||
         link != 0; 
 | 
			
		||||
         link = link->next_ )
 | 
			
		||||
   {
 | 
			
		||||
      BucketIndex index;
 | 
			
		||||
      for ( index =0; index < ValueInternalLink::itemPerLink; ++index )
 | 
			
		||||
      {
 | 
			
		||||
         if ( link->items_[index].isItemAvailable() )
 | 
			
		||||
            return;
 | 
			
		||||
         if ( strcmp( key, link->keys_[index] ) == 0 )
 | 
			
		||||
         {
 | 
			
		||||
            doActualRemove( link, index, bucketIndex );
 | 
			
		||||
            return;
 | 
			
		||||
         }
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
ValueInternalMap::doActualRemove( ValueInternalLink *link, 
 | 
			
		||||
                                  BucketIndex index,
 | 
			
		||||
                                  BucketIndex bucketIndex )
 | 
			
		||||
{
 | 
			
		||||
   // find last item of the bucket and swap it with the 'removed' one.
 | 
			
		||||
   // set removed items flags to 'available'.
 | 
			
		||||
   // if last page only contains 'available' items, then desallocate it (it's empty)
 | 
			
		||||
   ValueInternalLink *&lastLink = getLastLinkInBucket( index );
 | 
			
		||||
   BucketIndex lastItemIndex = 1; // a link can never be empty, so start at 1
 | 
			
		||||
   for ( ;   
 | 
			
		||||
         lastItemIndex < ValueInternalLink::itemPerLink; 
 | 
			
		||||
         ++lastItemIndex ) // may be optimized with dicotomic search
 | 
			
		||||
   {
 | 
			
		||||
      if ( lastLink->items_[lastItemIndex].isItemAvailable() )
 | 
			
		||||
         break;
 | 
			
		||||
   }
 | 
			
		||||
   
 | 
			
		||||
   BucketIndex lastUsedIndex = lastItemIndex - 1;
 | 
			
		||||
   Value *valueToDelete = &link->items_[index];
 | 
			
		||||
   Value *valueToPreserve = &lastLink->items_[lastUsedIndex];
 | 
			
		||||
   if ( valueToDelete != valueToPreserve )
 | 
			
		||||
      valueToDelete->swap( *valueToPreserve );
 | 
			
		||||
   if ( lastUsedIndex == 0 )  // page is now empty
 | 
			
		||||
   {  // remove it from bucket linked list and delete it.
 | 
			
		||||
      ValueInternalLink *linkPreviousToLast = lastLink->previous_;
 | 
			
		||||
      if ( linkPreviousToLast != 0 )   // can not deleted bucket link.
 | 
			
		||||
      {
 | 
			
		||||
         mapAllocator()->releaseMapLink( lastLink );
 | 
			
		||||
         linkPreviousToLast->next_ = 0;
 | 
			
		||||
         lastLink = linkPreviousToLast;
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
   else
 | 
			
		||||
   {
 | 
			
		||||
      Value dummy;
 | 
			
		||||
      valueToPreserve->swap( dummy ); // restore deleted to default Value.
 | 
			
		||||
      valueToPreserve->setItemUsed( false );
 | 
			
		||||
   }
 | 
			
		||||
   --itemCount_;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
ValueInternalLink *&
 | 
			
		||||
ValueInternalMap::getLastLinkInBucket( BucketIndex bucketIndex )
 | 
			
		||||
{
 | 
			
		||||
   if ( bucketIndex == bucketsSize_ - 1 )
 | 
			
		||||
      return tailLink_;
 | 
			
		||||
   ValueInternalLink *&previous = buckets_[bucketIndex+1].previous_;
 | 
			
		||||
   if ( !previous )
 | 
			
		||||
      previous = &buckets_[bucketIndex];
 | 
			
		||||
   return previous;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
Value &
 | 
			
		||||
ValueInternalMap::setNewItem( const char *key, 
 | 
			
		||||
                              bool isStatic,
 | 
			
		||||
                              ValueInternalLink *link, 
 | 
			
		||||
                              BucketIndex index )
 | 
			
		||||
{
 | 
			
		||||
   char *duplicatedKey = valueAllocator()->makeMemberName( key );
 | 
			
		||||
   ++itemCount_;
 | 
			
		||||
   link->keys_[index] = duplicatedKey;
 | 
			
		||||
   link->items_[index].setItemUsed();
 | 
			
		||||
   link->items_[index].setMemberNameIsStatic( isStatic );
 | 
			
		||||
   return link->items_[index]; // items already default constructed.
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
Value &
 | 
			
		||||
ValueInternalMap::unsafeAdd( const char *key, 
 | 
			
		||||
                             bool isStatic, 
 | 
			
		||||
                             HashKey hashedKey )
 | 
			
		||||
{
 | 
			
		||||
   JSON_ASSERT_MESSAGE( bucketsSize_ > 0, "ValueInternalMap::unsafeAdd(): internal logic error." );
 | 
			
		||||
   BucketIndex bucketIndex = hashedKey % bucketsSize_;
 | 
			
		||||
   ValueInternalLink *&previousLink = getLastLinkInBucket( bucketIndex );
 | 
			
		||||
   ValueInternalLink *link = previousLink;
 | 
			
		||||
   BucketIndex index;
 | 
			
		||||
   for ( index =0; index < ValueInternalLink::itemPerLink; ++index )
 | 
			
		||||
   {
 | 
			
		||||
      if ( link->items_[index].isItemAvailable() )
 | 
			
		||||
         break;
 | 
			
		||||
   }
 | 
			
		||||
   if ( index == ValueInternalLink::itemPerLink ) // need to add a new page
 | 
			
		||||
   {
 | 
			
		||||
      ValueInternalLink *newLink = mapAllocator()->allocateMapLink();
 | 
			
		||||
      index = 0;
 | 
			
		||||
      link->next_ = newLink;
 | 
			
		||||
      previousLink = newLink;
 | 
			
		||||
      link = newLink;
 | 
			
		||||
   }
 | 
			
		||||
   return setNewItem( key, isStatic, link, index );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
ValueInternalMap::HashKey 
 | 
			
		||||
ValueInternalMap::hash( const char *key ) const
 | 
			
		||||
{
 | 
			
		||||
   HashKey hash = 0;
 | 
			
		||||
   while ( *key )
 | 
			
		||||
      hash += *key++ * 37;
 | 
			
		||||
   return hash;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
int 
 | 
			
		||||
ValueInternalMap::compare( const ValueInternalMap &other ) const
 | 
			
		||||
{
 | 
			
		||||
   int sizeDiff( itemCount_ - other.itemCount_ );
 | 
			
		||||
   if ( sizeDiff != 0 )
 | 
			
		||||
      return sizeDiff;
 | 
			
		||||
   // Strict order guaranty is required. Compare all keys FIRST, then compare values.
 | 
			
		||||
   IteratorState it;
 | 
			
		||||
   IteratorState itEnd;
 | 
			
		||||
   makeBeginIterator( it );
 | 
			
		||||
   makeEndIterator( itEnd );
 | 
			
		||||
   for ( ; !equals(it,itEnd); increment(it) )
 | 
			
		||||
   {
 | 
			
		||||
      if ( !other.find( key( it ) ) )
 | 
			
		||||
         return 1;
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   // All keys are equals, let's compare values
 | 
			
		||||
   makeBeginIterator( it );
 | 
			
		||||
   for ( ; !equals(it,itEnd); increment(it) )
 | 
			
		||||
   {
 | 
			
		||||
      const Value *otherValue = other.find( key( it ) );
 | 
			
		||||
      int valueDiff = value(it).compare( *otherValue );
 | 
			
		||||
      if ( valueDiff != 0 )
 | 
			
		||||
         return valueDiff;
 | 
			
		||||
   }
 | 
			
		||||
   return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
ValueInternalMap::makeBeginIterator( IteratorState &it ) const
 | 
			
		||||
{
 | 
			
		||||
   it.map_ = const_cast<ValueInternalMap *>( this );
 | 
			
		||||
   it.bucketIndex_ = 0;
 | 
			
		||||
   it.itemIndex_ = 0;
 | 
			
		||||
   it.link_ = buckets_;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
ValueInternalMap::makeEndIterator( IteratorState &it ) const
 | 
			
		||||
{
 | 
			
		||||
   it.map_ = const_cast<ValueInternalMap *>( this );
 | 
			
		||||
   it.bucketIndex_ = bucketsSize_;
 | 
			
		||||
   it.itemIndex_ = 0;
 | 
			
		||||
   it.link_ = 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
bool 
 | 
			
		||||
ValueInternalMap::equals( const IteratorState &x, const IteratorState &other )
 | 
			
		||||
{
 | 
			
		||||
   return x.map_ == other.map_  
 | 
			
		||||
          &&  x.bucketIndex_ == other.bucketIndex_  
 | 
			
		||||
          &&  x.link_ == other.link_
 | 
			
		||||
          &&  x.itemIndex_ == other.itemIndex_;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
ValueInternalMap::incrementBucket( IteratorState &iterator )
 | 
			
		||||
{
 | 
			
		||||
   ++iterator.bucketIndex_;
 | 
			
		||||
   JSON_ASSERT_MESSAGE( iterator.bucketIndex_ <= iterator.map_->bucketsSize_,
 | 
			
		||||
      "ValueInternalMap::increment(): attempting to iterate beyond end." );
 | 
			
		||||
   if ( iterator.bucketIndex_ == iterator.map_->bucketsSize_ )
 | 
			
		||||
      iterator.link_ = 0;
 | 
			
		||||
   else
 | 
			
		||||
      iterator.link_ = &(iterator.map_->buckets_[iterator.bucketIndex_]);
 | 
			
		||||
   iterator.itemIndex_ = 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
ValueInternalMap::increment( IteratorState &iterator )
 | 
			
		||||
{
 | 
			
		||||
   JSON_ASSERT_MESSAGE( iterator.map_, "Attempting to iterator using invalid iterator." );
 | 
			
		||||
   ++iterator.itemIndex_;
 | 
			
		||||
   if ( iterator.itemIndex_ == ValueInternalLink::itemPerLink )
 | 
			
		||||
   {
 | 
			
		||||
      JSON_ASSERT_MESSAGE( iterator.link_ != 0,
 | 
			
		||||
         "ValueInternalMap::increment(): attempting to iterate beyond end." );
 | 
			
		||||
      iterator.link_ = iterator.link_->next_;
 | 
			
		||||
      if ( iterator.link_ == 0 )
 | 
			
		||||
         incrementBucket( iterator );
 | 
			
		||||
   }
 | 
			
		||||
   else if ( iterator.link_->items_[iterator.itemIndex_].isItemAvailable() )
 | 
			
		||||
   {
 | 
			
		||||
      incrementBucket( iterator );
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
ValueInternalMap::decrement( IteratorState &iterator )
 | 
			
		||||
{
 | 
			
		||||
   if ( iterator.itemIndex_ == 0 )
 | 
			
		||||
   {
 | 
			
		||||
      JSON_ASSERT_MESSAGE( iterator.map_, "Attempting to iterate using invalid iterator." );
 | 
			
		||||
      if ( iterator.link_ == &iterator.map_->buckets_[iterator.bucketIndex_] )
 | 
			
		||||
      {
 | 
			
		||||
         JSON_ASSERT_MESSAGE( iterator.bucketIndex_ > 0, "Attempting to iterate beyond beginning." );
 | 
			
		||||
         --(iterator.bucketIndex_);
 | 
			
		||||
      }
 | 
			
		||||
      iterator.link_ = iterator.link_->previous_;
 | 
			
		||||
      iterator.itemIndex_ = ValueInternalLink::itemPerLink - 1;
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
const char *
 | 
			
		||||
ValueInternalMap::key( const IteratorState &iterator )
 | 
			
		||||
{
 | 
			
		||||
   JSON_ASSERT_MESSAGE( iterator.link_, "Attempting to iterate using invalid iterator." );
 | 
			
		||||
   return iterator.link_->keys_[iterator.itemIndex_];
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
const char *
 | 
			
		||||
ValueInternalMap::key( const IteratorState &iterator, bool &isStatic )
 | 
			
		||||
{
 | 
			
		||||
   JSON_ASSERT_MESSAGE( iterator.link_, "Attempting to iterate using invalid iterator." );
 | 
			
		||||
   isStatic = iterator.link_->items_[iterator.itemIndex_].isMemberNameStatic();
 | 
			
		||||
   return iterator.link_->keys_[iterator.itemIndex_];
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
Value &
 | 
			
		||||
ValueInternalMap::value( const IteratorState &iterator )
 | 
			
		||||
{
 | 
			
		||||
   JSON_ASSERT_MESSAGE( iterator.link_, "Attempting to iterate using invalid iterator." );
 | 
			
		||||
   return iterator.link_->items_[iterator.itemIndex_];
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
int 
 | 
			
		||||
ValueInternalMap::distance( const IteratorState &x, const IteratorState &y )
 | 
			
		||||
{
 | 
			
		||||
   int offset = 0;
 | 
			
		||||
   IteratorState it = x;
 | 
			
		||||
   while ( !equals( it, y ) )
 | 
			
		||||
      increment( it );
 | 
			
		||||
   return offset;
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			@ -1,885 +0,0 @@
 | 
			
		|||
#include "reader.h"
 | 
			
		||||
#include "value.h"
 | 
			
		||||
#include <utility>
 | 
			
		||||
#include <cstdio>
 | 
			
		||||
#include <cassert>
 | 
			
		||||
#include <cstring>
 | 
			
		||||
#include <iostream>
 | 
			
		||||
#include <stdexcept>
 | 
			
		||||
 | 
			
		||||
#if _MSC_VER >= 1400 // VC++ 8.0
 | 
			
		||||
#pragma warning( disable : 4996 )   // disable warning about strdup being deprecated.
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
namespace Json {
 | 
			
		||||
 | 
			
		||||
// Implementation of class Features
 | 
			
		||||
// ////////////////////////////////
 | 
			
		||||
 | 
			
		||||
Features::Features()
 | 
			
		||||
   : allowComments_( true )
 | 
			
		||||
   , strictRoot_( false )
 | 
			
		||||
{
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
Features 
 | 
			
		||||
Features::all()
 | 
			
		||||
{
 | 
			
		||||
   return Features();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
Features 
 | 
			
		||||
Features::strictMode()
 | 
			
		||||
{
 | 
			
		||||
   Features features;
 | 
			
		||||
   features.allowComments_ = false;
 | 
			
		||||
   features.strictRoot_ = true;
 | 
			
		||||
   return features;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Implementation of class Reader
 | 
			
		||||
// ////////////////////////////////
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
static inline bool 
 | 
			
		||||
in( Reader::Char c, Reader::Char c1, Reader::Char c2, Reader::Char c3, Reader::Char c4 )
 | 
			
		||||
{
 | 
			
		||||
   return c == c1  ||  c == c2  ||  c == c3  ||  c == c4;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static inline bool 
 | 
			
		||||
in( Reader::Char c, Reader::Char c1, Reader::Char c2, Reader::Char c3, Reader::Char c4, Reader::Char c5 )
 | 
			
		||||
{
 | 
			
		||||
   return c == c1  ||  c == c2  ||  c == c3  ||  c == c4  ||  c == c5;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
static bool 
 | 
			
		||||
containsNewLine( Reader::Location begin, 
 | 
			
		||||
                 Reader::Location end )
 | 
			
		||||
{
 | 
			
		||||
   for ( ;begin < end; ++begin )
 | 
			
		||||
      if ( *begin == '\n'  ||  *begin == '\r' )
 | 
			
		||||
         return true;
 | 
			
		||||
   return false;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static std::string codePointToUTF8(unsigned int cp)
 | 
			
		||||
{
 | 
			
		||||
   std::string result;
 | 
			
		||||
   
 | 
			
		||||
   // based on description from http://en.wikipedia.org/wiki/UTF-8
 | 
			
		||||
 | 
			
		||||
   if (cp <= 0x7f) 
 | 
			
		||||
   {
 | 
			
		||||
      result.resize(1);
 | 
			
		||||
      result[0] = static_cast<char>(cp);
 | 
			
		||||
   } 
 | 
			
		||||
   else if (cp <= 0x7FF) 
 | 
			
		||||
   {
 | 
			
		||||
      result.resize(2);
 | 
			
		||||
      result[1] = static_cast<char>(0x80 | (0x3f & cp));
 | 
			
		||||
      result[0] = static_cast<char>(0xC0 | (0x1f & (cp >> 6)));
 | 
			
		||||
   } 
 | 
			
		||||
   else if (cp <= 0xFFFF) 
 | 
			
		||||
   {
 | 
			
		||||
      result.resize(3);
 | 
			
		||||
      result[2] = static_cast<char>(0x80 | (0x3f & cp));
 | 
			
		||||
      result[1] = 0x80 | static_cast<char>((0x3f & (cp >> 6)));
 | 
			
		||||
      result[0] = 0xE0 | static_cast<char>((0xf & (cp >> 12)));
 | 
			
		||||
   }
 | 
			
		||||
   else if (cp <= 0x10FFFF) 
 | 
			
		||||
   {
 | 
			
		||||
      result.resize(4);
 | 
			
		||||
      result[3] = static_cast<char>(0x80 | (0x3f & cp));
 | 
			
		||||
      result[2] = static_cast<char>(0x80 | (0x3f & (cp >> 6)));
 | 
			
		||||
      result[1] = static_cast<char>(0x80 | (0x3f & (cp >> 12)));
 | 
			
		||||
      result[0] = static_cast<char>(0xF0 | (0x7 & (cp >> 18)));
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   return result;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
// Class Reader
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
 | 
			
		||||
Reader::Reader()
 | 
			
		||||
   : features_( Features::all() )
 | 
			
		||||
{
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
Reader::Reader( const Features &features )
 | 
			
		||||
   : features_( features )
 | 
			
		||||
{
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
bool
 | 
			
		||||
Reader::parse( const std::string &document, 
 | 
			
		||||
               Value &root,
 | 
			
		||||
               bool collectComments )
 | 
			
		||||
{
 | 
			
		||||
   document_ = document;
 | 
			
		||||
   const char *begin = document_.c_str();
 | 
			
		||||
   const char *end = begin + document_.length();
 | 
			
		||||
   return parse( begin, end, root, collectComments );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
bool
 | 
			
		||||
Reader::parse( std::istream& sin,
 | 
			
		||||
               Value &root,
 | 
			
		||||
               bool collectComments )
 | 
			
		||||
{
 | 
			
		||||
   //std::istream_iterator<char> begin(sin);
 | 
			
		||||
   //std::istream_iterator<char> end;
 | 
			
		||||
   // Those would allow streamed input from a file, if parse() were a
 | 
			
		||||
   // template function.
 | 
			
		||||
 | 
			
		||||
   // Since std::string is reference-counted, this at least does not
 | 
			
		||||
   // create an extra copy.
 | 
			
		||||
   std::string doc;
 | 
			
		||||
   std::getline(sin, doc, (char)EOF);
 | 
			
		||||
   return parse( doc, root, collectComments );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
bool 
 | 
			
		||||
Reader::parse( const char *beginDoc, const char *endDoc, 
 | 
			
		||||
               Value &root,
 | 
			
		||||
               bool collectComments )
 | 
			
		||||
{
 | 
			
		||||
   if ( !features_.allowComments_ )
 | 
			
		||||
   {
 | 
			
		||||
      collectComments = false;
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   begin_ = beginDoc;
 | 
			
		||||
   end_ = endDoc;
 | 
			
		||||
   collectComments_ = collectComments;
 | 
			
		||||
   current_ = begin_;
 | 
			
		||||
   lastValueEnd_ = 0;
 | 
			
		||||
   lastValue_ = 0;
 | 
			
		||||
   commentsBefore_ = "";
 | 
			
		||||
   errors_.clear();
 | 
			
		||||
   while ( !nodes_.empty() )
 | 
			
		||||
      nodes_.pop();
 | 
			
		||||
   nodes_.push( &root );
 | 
			
		||||
   
 | 
			
		||||
   bool successful = readValue();
 | 
			
		||||
   Token token;
 | 
			
		||||
   skipCommentTokens( token );
 | 
			
		||||
   if ( collectComments_  &&  !commentsBefore_.empty() )
 | 
			
		||||
      root.setComment( commentsBefore_, commentAfter );
 | 
			
		||||
   if ( features_.strictRoot_ )
 | 
			
		||||
   {
 | 
			
		||||
      if ( !root.isArray()  &&  !root.isObject() )
 | 
			
		||||
      {
 | 
			
		||||
         // Set error location to start of doc, ideally should be first token found in doc
 | 
			
		||||
         token.type_ = tokenError;
 | 
			
		||||
         token.start_ = beginDoc;
 | 
			
		||||
         token.end_ = endDoc;
 | 
			
		||||
         addError( "A valid JSON document must be either an array or an object value.",
 | 
			
		||||
                   token );
 | 
			
		||||
         return false;
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
   return successful;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
bool
 | 
			
		||||
Reader::readValue()
 | 
			
		||||
{
 | 
			
		||||
   Token token;
 | 
			
		||||
   skipCommentTokens( token );
 | 
			
		||||
   bool successful = true;
 | 
			
		||||
 | 
			
		||||
   if ( collectComments_  &&  !commentsBefore_.empty() )
 | 
			
		||||
   {
 | 
			
		||||
      currentValue().setComment( commentsBefore_, commentBefore );
 | 
			
		||||
      commentsBefore_ = "";
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   switch ( token.type_ )
 | 
			
		||||
   {
 | 
			
		||||
   case tokenObjectBegin:
 | 
			
		||||
      successful = readObject( token );
 | 
			
		||||
      break;
 | 
			
		||||
   case tokenArrayBegin:
 | 
			
		||||
      successful = readArray( token );
 | 
			
		||||
      break;
 | 
			
		||||
   case tokenNumber:
 | 
			
		||||
      successful = decodeNumber( token );
 | 
			
		||||
      break;
 | 
			
		||||
   case tokenString:
 | 
			
		||||
      successful = decodeString( token );
 | 
			
		||||
      break;
 | 
			
		||||
   case tokenTrue:
 | 
			
		||||
      currentValue() = true;
 | 
			
		||||
      break;
 | 
			
		||||
   case tokenFalse:
 | 
			
		||||
      currentValue() = false;
 | 
			
		||||
      break;
 | 
			
		||||
   case tokenNull:
 | 
			
		||||
      currentValue() = Value();
 | 
			
		||||
      break;
 | 
			
		||||
   default:
 | 
			
		||||
      return addError( "Syntax error: value, object or array expected.", token );
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   if ( collectComments_ )
 | 
			
		||||
   {
 | 
			
		||||
      lastValueEnd_ = current_;
 | 
			
		||||
      lastValue_ = ¤tValue();
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   return successful;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
Reader::skipCommentTokens( Token &token )
 | 
			
		||||
{
 | 
			
		||||
   if ( features_.allowComments_ )
 | 
			
		||||
   {
 | 
			
		||||
      do
 | 
			
		||||
      {
 | 
			
		||||
         readToken( token );
 | 
			
		||||
      }
 | 
			
		||||
      while ( token.type_ == tokenComment );
 | 
			
		||||
   }
 | 
			
		||||
   else
 | 
			
		||||
   {
 | 
			
		||||
      readToken( token );
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
bool 
 | 
			
		||||
Reader::expectToken( TokenType type, Token &token, const char *message )
 | 
			
		||||
{
 | 
			
		||||
   readToken( token );
 | 
			
		||||
   if ( token.type_ != type )
 | 
			
		||||
      return addError( message, token );
 | 
			
		||||
   return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
bool 
 | 
			
		||||
Reader::readToken( Token &token )
 | 
			
		||||
{
 | 
			
		||||
   skipSpaces();
 | 
			
		||||
   token.start_ = current_;
 | 
			
		||||
   Char c = getNextChar();
 | 
			
		||||
   bool ok = true;
 | 
			
		||||
   switch ( c )
 | 
			
		||||
   {
 | 
			
		||||
   case '{':
 | 
			
		||||
      token.type_ = tokenObjectBegin;
 | 
			
		||||
      break;
 | 
			
		||||
   case '}':
 | 
			
		||||
      token.type_ = tokenObjectEnd;
 | 
			
		||||
      break;
 | 
			
		||||
   case '[':
 | 
			
		||||
      token.type_ = tokenArrayBegin;
 | 
			
		||||
      break;
 | 
			
		||||
   case ']':
 | 
			
		||||
      token.type_ = tokenArrayEnd;
 | 
			
		||||
      break;
 | 
			
		||||
   case '"':
 | 
			
		||||
      token.type_ = tokenString;
 | 
			
		||||
      ok = readString();
 | 
			
		||||
      break;
 | 
			
		||||
   case '/':
 | 
			
		||||
      token.type_ = tokenComment;
 | 
			
		||||
      ok = readComment();
 | 
			
		||||
      break;
 | 
			
		||||
   case '0':
 | 
			
		||||
   case '1':
 | 
			
		||||
   case '2':
 | 
			
		||||
   case '3':
 | 
			
		||||
   case '4':
 | 
			
		||||
   case '5':
 | 
			
		||||
   case '6':
 | 
			
		||||
   case '7':
 | 
			
		||||
   case '8':
 | 
			
		||||
   case '9':
 | 
			
		||||
   case '-':
 | 
			
		||||
      token.type_ = tokenNumber;
 | 
			
		||||
      readNumber();
 | 
			
		||||
      break;
 | 
			
		||||
   case 't':
 | 
			
		||||
      token.type_ = tokenTrue;
 | 
			
		||||
      ok = match( "rue", 3 );
 | 
			
		||||
      break;
 | 
			
		||||
   case 'f':
 | 
			
		||||
      token.type_ = tokenFalse;
 | 
			
		||||
      ok = match( "alse", 4 );
 | 
			
		||||
      break;
 | 
			
		||||
   case 'n':
 | 
			
		||||
      token.type_ = tokenNull;
 | 
			
		||||
      ok = match( "ull", 3 );
 | 
			
		||||
      break;
 | 
			
		||||
   case ',':
 | 
			
		||||
      token.type_ = tokenArraySeparator;
 | 
			
		||||
      break;
 | 
			
		||||
   case ':':
 | 
			
		||||
      token.type_ = tokenMemberSeparator;
 | 
			
		||||
      break;
 | 
			
		||||
   case 0:
 | 
			
		||||
      token.type_ = tokenEndOfStream;
 | 
			
		||||
      break;
 | 
			
		||||
   default:
 | 
			
		||||
      ok = false;
 | 
			
		||||
      break;
 | 
			
		||||
   }
 | 
			
		||||
   if ( !ok )
 | 
			
		||||
      token.type_ = tokenError;
 | 
			
		||||
   token.end_ = current_;
 | 
			
		||||
   return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
Reader::skipSpaces()
 | 
			
		||||
{
 | 
			
		||||
   while ( current_ != end_ )
 | 
			
		||||
   {
 | 
			
		||||
      Char c = *current_;
 | 
			
		||||
      if ( c == ' '  ||  c == '\t'  ||  c == '\r'  ||  c == '\n' )
 | 
			
		||||
         ++current_;
 | 
			
		||||
      else
 | 
			
		||||
         break;
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
bool 
 | 
			
		||||
Reader::match( Location pattern, 
 | 
			
		||||
               int patternLength )
 | 
			
		||||
{
 | 
			
		||||
   if ( end_ - current_ < patternLength )
 | 
			
		||||
      return false;
 | 
			
		||||
   int index = patternLength;
 | 
			
		||||
   while ( index-- )
 | 
			
		||||
      if ( current_[index] != pattern[index] )
 | 
			
		||||
         return false;
 | 
			
		||||
   current_ += patternLength;
 | 
			
		||||
   return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
bool
 | 
			
		||||
Reader::readComment()
 | 
			
		||||
{
 | 
			
		||||
   Location commentBegin = current_ - 1;
 | 
			
		||||
   Char c = getNextChar();
 | 
			
		||||
   bool successful = false;
 | 
			
		||||
   if ( c == '*' )
 | 
			
		||||
      successful = readCStyleComment();
 | 
			
		||||
   else if ( c == '/' )
 | 
			
		||||
      successful = readCppStyleComment();
 | 
			
		||||
   if ( !successful )
 | 
			
		||||
      return false;
 | 
			
		||||
 | 
			
		||||
   if ( collectComments_ )
 | 
			
		||||
   {
 | 
			
		||||
      CommentPlacement placement = commentBefore;
 | 
			
		||||
      if ( lastValueEnd_  &&  !containsNewLine( lastValueEnd_, commentBegin ) )
 | 
			
		||||
      {
 | 
			
		||||
         if ( c != '*'  ||  !containsNewLine( commentBegin, current_ ) )
 | 
			
		||||
            placement = commentAfterOnSameLine;
 | 
			
		||||
      }
 | 
			
		||||
 | 
			
		||||
      addComment( commentBegin, current_, placement );
 | 
			
		||||
   }
 | 
			
		||||
   return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
Reader::addComment( Location begin, 
 | 
			
		||||
                    Location end, 
 | 
			
		||||
                    CommentPlacement placement )
 | 
			
		||||
{
 | 
			
		||||
   assert( collectComments_ );
 | 
			
		||||
   if ( placement == commentAfterOnSameLine )
 | 
			
		||||
   {
 | 
			
		||||
      assert( lastValue_ != 0 );
 | 
			
		||||
      lastValue_->setComment( std::string( begin, end ), placement );
 | 
			
		||||
   }
 | 
			
		||||
   else
 | 
			
		||||
   {
 | 
			
		||||
      if ( !commentsBefore_.empty() )
 | 
			
		||||
         commentsBefore_ += "\n";
 | 
			
		||||
      commentsBefore_ += std::string( begin, end );
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
bool 
 | 
			
		||||
Reader::readCStyleComment()
 | 
			
		||||
{
 | 
			
		||||
   while ( current_ != end_ )
 | 
			
		||||
   {
 | 
			
		||||
      Char c = getNextChar();
 | 
			
		||||
      if ( c == '*'  &&  *current_ == '/' )
 | 
			
		||||
         break;
 | 
			
		||||
   }
 | 
			
		||||
   return getNextChar() == '/';
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
bool 
 | 
			
		||||
Reader::readCppStyleComment()
 | 
			
		||||
{
 | 
			
		||||
   while ( current_ != end_ )
 | 
			
		||||
   {
 | 
			
		||||
      Char c = getNextChar();
 | 
			
		||||
      if (  c == '\r'  ||  c == '\n' )
 | 
			
		||||
         break;
 | 
			
		||||
   }
 | 
			
		||||
   return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
Reader::readNumber()
 | 
			
		||||
{
 | 
			
		||||
   while ( current_ != end_ )
 | 
			
		||||
   {
 | 
			
		||||
      if ( !(*current_ >= '0'  &&  *current_ <= '9')  &&
 | 
			
		||||
           !in( *current_, '.', 'e', 'E', '+', '-' ) )
 | 
			
		||||
         break;
 | 
			
		||||
      ++current_;
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
bool
 | 
			
		||||
Reader::readString()
 | 
			
		||||
{
 | 
			
		||||
   Char c = 0;
 | 
			
		||||
   while ( current_ != end_ )
 | 
			
		||||
   {
 | 
			
		||||
      c = getNextChar();
 | 
			
		||||
      if ( c == '\\' )
 | 
			
		||||
         getNextChar();
 | 
			
		||||
      else if ( c == '"' )
 | 
			
		||||
         break;
 | 
			
		||||
   }
 | 
			
		||||
   return c == '"';
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
bool 
 | 
			
		||||
Reader::readObject( Token &tokenStart )
 | 
			
		||||
{
 | 
			
		||||
   Token tokenName;
 | 
			
		||||
   std::string name;
 | 
			
		||||
   currentValue() = Value( objectValue );
 | 
			
		||||
   while ( readToken( tokenName ) )
 | 
			
		||||
   {
 | 
			
		||||
      bool initialTokenOk = true;
 | 
			
		||||
      while ( tokenName.type_ == tokenComment  &&  initialTokenOk )
 | 
			
		||||
         initialTokenOk = readToken( tokenName );
 | 
			
		||||
      if  ( !initialTokenOk )
 | 
			
		||||
         break;
 | 
			
		||||
      if ( tokenName.type_ == tokenObjectEnd  &&  name.empty() )  // empty object
 | 
			
		||||
         return true;
 | 
			
		||||
      if ( tokenName.type_ != tokenString )
 | 
			
		||||
         break;
 | 
			
		||||
      
 | 
			
		||||
      name = "";
 | 
			
		||||
      if ( !decodeString( tokenName, name ) )
 | 
			
		||||
         return recoverFromError( tokenObjectEnd );
 | 
			
		||||
 | 
			
		||||
      Token colon;
 | 
			
		||||
      if ( !readToken( colon ) ||  colon.type_ != tokenMemberSeparator )
 | 
			
		||||
      {
 | 
			
		||||
         return addErrorAndRecover( "Missing ':' after object member name", 
 | 
			
		||||
                                    colon, 
 | 
			
		||||
                                    tokenObjectEnd );
 | 
			
		||||
      }
 | 
			
		||||
      Value &value = currentValue()[ name ];
 | 
			
		||||
      nodes_.push( &value );
 | 
			
		||||
      bool ok = readValue();
 | 
			
		||||
      nodes_.pop();
 | 
			
		||||
      if ( !ok ) // error already set
 | 
			
		||||
         return recoverFromError( tokenObjectEnd );
 | 
			
		||||
 | 
			
		||||
      Token comma;
 | 
			
		||||
      if ( !readToken( comma )
 | 
			
		||||
            ||  ( comma.type_ != tokenObjectEnd  &&  
 | 
			
		||||
                  comma.type_ != tokenArraySeparator &&
 | 
			
		||||
		  comma.type_ != tokenComment ) )
 | 
			
		||||
      {
 | 
			
		||||
         return addErrorAndRecover( "Missing ',' or '}' in object declaration", 
 | 
			
		||||
                                    comma, 
 | 
			
		||||
                                    tokenObjectEnd );
 | 
			
		||||
      }
 | 
			
		||||
      bool finalizeTokenOk = true;
 | 
			
		||||
      while ( comma.type_ == tokenComment &&
 | 
			
		||||
              finalizeTokenOk )
 | 
			
		||||
         finalizeTokenOk = readToken( comma );
 | 
			
		||||
      if ( comma.type_ == tokenObjectEnd )
 | 
			
		||||
         return true;
 | 
			
		||||
   }
 | 
			
		||||
   return addErrorAndRecover( "Missing '}' or object member name", 
 | 
			
		||||
                              tokenName, 
 | 
			
		||||
                              tokenObjectEnd );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
bool 
 | 
			
		||||
Reader::readArray( Token &tokenStart )
 | 
			
		||||
{
 | 
			
		||||
   currentValue() = Value( arrayValue );
 | 
			
		||||
   skipSpaces();
 | 
			
		||||
   if ( *current_ == ']' ) // empty array
 | 
			
		||||
   {
 | 
			
		||||
      Token endArray;
 | 
			
		||||
      readToken( endArray );
 | 
			
		||||
      return true;
 | 
			
		||||
   }
 | 
			
		||||
   int index = 0;
 | 
			
		||||
   while ( true )
 | 
			
		||||
   {
 | 
			
		||||
      Value &value = currentValue()[ index++ ];
 | 
			
		||||
      nodes_.push( &value );
 | 
			
		||||
      bool ok = readValue();
 | 
			
		||||
      nodes_.pop();
 | 
			
		||||
      if ( !ok ) // error already set
 | 
			
		||||
         return recoverFromError( tokenArrayEnd );
 | 
			
		||||
 | 
			
		||||
      Token token;
 | 
			
		||||
      // Accept Comment after last item in the array.
 | 
			
		||||
      ok = readToken( token );
 | 
			
		||||
      while ( token.type_ == tokenComment  &&  ok )
 | 
			
		||||
      {
 | 
			
		||||
         ok = readToken( token );
 | 
			
		||||
      }
 | 
			
		||||
      bool badTokenType = ( token.type_ == tokenArraySeparator  &&  
 | 
			
		||||
                            token.type_ == tokenArrayEnd );
 | 
			
		||||
      if ( !ok  ||  badTokenType )
 | 
			
		||||
      {
 | 
			
		||||
         return addErrorAndRecover( "Missing ',' or ']' in array declaration", 
 | 
			
		||||
                                    token, 
 | 
			
		||||
                                    tokenArrayEnd );
 | 
			
		||||
      }
 | 
			
		||||
      if ( token.type_ == tokenArrayEnd )
 | 
			
		||||
         break;
 | 
			
		||||
   }
 | 
			
		||||
   return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
bool 
 | 
			
		||||
Reader::decodeNumber( Token &token )
 | 
			
		||||
{
 | 
			
		||||
   bool isDouble = false;
 | 
			
		||||
   for ( Location inspect = token.start_; inspect != token.end_; ++inspect )
 | 
			
		||||
   {
 | 
			
		||||
      isDouble = isDouble  
 | 
			
		||||
                 ||  in( *inspect, '.', 'e', 'E', '+' )  
 | 
			
		||||
                 ||  ( *inspect == '-'  &&  inspect != token.start_ );
 | 
			
		||||
   }
 | 
			
		||||
   if ( isDouble )
 | 
			
		||||
      return decodeDouble( token );
 | 
			
		||||
   Location current = token.start_;
 | 
			
		||||
   bool isNegative = *current == '-';
 | 
			
		||||
   if ( isNegative )
 | 
			
		||||
      ++current;
 | 
			
		||||
   Value::UInt threshold = (isNegative ? Value::UInt(-Value::minInt) 
 | 
			
		||||
                                       : Value::maxUInt) / 10;
 | 
			
		||||
   Value::UInt value = 0;
 | 
			
		||||
   while ( current < token.end_ )
 | 
			
		||||
   {
 | 
			
		||||
      Char c = *current++;
 | 
			
		||||
      if ( c < '0'  ||  c > '9' )
 | 
			
		||||
         return addError( "'" + std::string( token.start_, token.end_ ) + "' is not a number.", token );
 | 
			
		||||
      if ( value >= threshold )
 | 
			
		||||
         return decodeDouble( token );
 | 
			
		||||
      value = value * 10 + Value::UInt(c - '0');
 | 
			
		||||
   }
 | 
			
		||||
   if ( isNegative )
 | 
			
		||||
      currentValue() = -Value::Int( value );
 | 
			
		||||
   else if ( value <= Value::UInt(Value::maxInt) )
 | 
			
		||||
      currentValue() = Value::Int( value );
 | 
			
		||||
   else
 | 
			
		||||
      currentValue() = value;
 | 
			
		||||
   return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
bool 
 | 
			
		||||
Reader::decodeDouble( Token &token )
 | 
			
		||||
{
 | 
			
		||||
   double value = 0;
 | 
			
		||||
   const int bufferSize = 32;
 | 
			
		||||
   int count;
 | 
			
		||||
   int length = int(token.end_ - token.start_);
 | 
			
		||||
   if ( length <= bufferSize )
 | 
			
		||||
   {
 | 
			
		||||
      Char buffer[bufferSize];
 | 
			
		||||
      memcpy( buffer, token.start_, length );
 | 
			
		||||
      buffer[length] = 0;
 | 
			
		||||
      count = sscanf( buffer, "%lf", &value );
 | 
			
		||||
   }
 | 
			
		||||
   else
 | 
			
		||||
   {
 | 
			
		||||
      std::string buffer( token.start_, token.end_ );
 | 
			
		||||
      count = sscanf( buffer.c_str(), "%lf", &value );
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
   if ( count != 1 )
 | 
			
		||||
      return addError( "'" + std::string( token.start_, token.end_ ) + "' is not a number.", token );
 | 
			
		||||
   currentValue() = value;
 | 
			
		||||
   return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
bool 
 | 
			
		||||
Reader::decodeString( Token &token )
 | 
			
		||||
{
 | 
			
		||||
   std::string decoded;
 | 
			
		||||
   if ( !decodeString( token, decoded ) )
 | 
			
		||||
      return false;
 | 
			
		||||
   currentValue() = decoded;
 | 
			
		||||
   return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
bool 
 | 
			
		||||
Reader::decodeString( Token &token, std::string &decoded )
 | 
			
		||||
{
 | 
			
		||||
   decoded.reserve( token.end_ - token.start_ - 2 );
 | 
			
		||||
   Location current = token.start_ + 1; // skip '"'
 | 
			
		||||
   Location end = token.end_ - 1;      // do not include '"'
 | 
			
		||||
   while ( current != end )
 | 
			
		||||
   {
 | 
			
		||||
      Char c = *current++;
 | 
			
		||||
      if ( c == '"' )
 | 
			
		||||
         break;
 | 
			
		||||
      else if ( c == '\\' )
 | 
			
		||||
      {
 | 
			
		||||
         if ( current == end )
 | 
			
		||||
            return addError( "Empty escape sequence in string", token, current );
 | 
			
		||||
         Char escape = *current++;
 | 
			
		||||
         switch ( escape )
 | 
			
		||||
         {
 | 
			
		||||
         case '"': decoded += '"'; break;
 | 
			
		||||
         case '/': decoded += '/'; break;
 | 
			
		||||
         case '\\': decoded += '\\'; break;
 | 
			
		||||
         case 'b': decoded += '\b'; break;
 | 
			
		||||
         case 'f': decoded += '\f'; break;
 | 
			
		||||
         case 'n': decoded += '\n'; break;
 | 
			
		||||
         case 'r': decoded += '\r'; break;
 | 
			
		||||
         case 't': decoded += '\t'; break;
 | 
			
		||||
         case 'u':
 | 
			
		||||
            {
 | 
			
		||||
               unsigned int unicode;
 | 
			
		||||
               if ( !decodeUnicodeCodePoint( token, current, end, unicode ) )
 | 
			
		||||
                  return false;
 | 
			
		||||
               decoded += codePointToUTF8(unicode);
 | 
			
		||||
            }
 | 
			
		||||
            break;
 | 
			
		||||
         default:
 | 
			
		||||
            return addError( "Bad escape sequence in string", token, current );
 | 
			
		||||
         }
 | 
			
		||||
      }
 | 
			
		||||
      else
 | 
			
		||||
      {
 | 
			
		||||
         decoded += c;
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
   return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
bool
 | 
			
		||||
Reader::decodeUnicodeCodePoint( Token &token, 
 | 
			
		||||
                                     Location ¤t, 
 | 
			
		||||
                                     Location end, 
 | 
			
		||||
                                     unsigned int &unicode )
 | 
			
		||||
{
 | 
			
		||||
 | 
			
		||||
   if ( !decodeUnicodeEscapeSequence( token, current, end, unicode ) )
 | 
			
		||||
      return false;
 | 
			
		||||
   if (unicode >= 0xD800 && unicode <= 0xDBFF)
 | 
			
		||||
   {
 | 
			
		||||
      // surrogate pairs
 | 
			
		||||
      if (end - current < 6)
 | 
			
		||||
         return addError( "additional six characters expected to parse unicode surrogate pair.", token, current );
 | 
			
		||||
      unsigned int surrogatePair;
 | 
			
		||||
      if (*(current++) == '\\' && *(current++)== 'u')
 | 
			
		||||
      {
 | 
			
		||||
         if (decodeUnicodeEscapeSequence( token, current, end, surrogatePair ))
 | 
			
		||||
         {
 | 
			
		||||
            unicode = 0x10000 + ((unicode & 0x3FF) << 10) + (surrogatePair & 0x3FF);
 | 
			
		||||
         } 
 | 
			
		||||
         else
 | 
			
		||||
            return false;
 | 
			
		||||
      } 
 | 
			
		||||
      else
 | 
			
		||||
         return addError( "expecting another \\u token to begin the second half of a unicode surrogate pair", token, current );
 | 
			
		||||
   }
 | 
			
		||||
   return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
bool 
 | 
			
		||||
Reader::decodeUnicodeEscapeSequence( Token &token, 
 | 
			
		||||
                                     Location ¤t, 
 | 
			
		||||
                                     Location end, 
 | 
			
		||||
                                     unsigned int &unicode )
 | 
			
		||||
{
 | 
			
		||||
   if ( end - current < 4 )
 | 
			
		||||
      return addError( "Bad unicode escape sequence in string: four digits expected.", token, current );
 | 
			
		||||
   unicode = 0;
 | 
			
		||||
   for ( int index =0; index < 4; ++index )
 | 
			
		||||
   {
 | 
			
		||||
      Char c = *current++;
 | 
			
		||||
      unicode *= 16;
 | 
			
		||||
      if ( c >= '0'  &&  c <= '9' )
 | 
			
		||||
         unicode += c - '0';
 | 
			
		||||
      else if ( c >= 'a'  &&  c <= 'f' )
 | 
			
		||||
         unicode += c - 'a' + 10;
 | 
			
		||||
      else if ( c >= 'A'  &&  c <= 'F' )
 | 
			
		||||
         unicode += c - 'A' + 10;
 | 
			
		||||
      else
 | 
			
		||||
         return addError( "Bad unicode escape sequence in string: hexadecimal digit expected.", token, current );
 | 
			
		||||
   }
 | 
			
		||||
   return true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
bool 
 | 
			
		||||
Reader::addError( const std::string &message, 
 | 
			
		||||
                  Token &token,
 | 
			
		||||
                  Location extra )
 | 
			
		||||
{
 | 
			
		||||
   ErrorInfo info;
 | 
			
		||||
   info.token_ = token;
 | 
			
		||||
   info.message_ = message;
 | 
			
		||||
   info.extra_ = extra;
 | 
			
		||||
   errors_.push_back( info );
 | 
			
		||||
   return false;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
bool 
 | 
			
		||||
Reader::recoverFromError( TokenType skipUntilToken )
 | 
			
		||||
{
 | 
			
		||||
   int errorCount = int(errors_.size());
 | 
			
		||||
   Token skip;
 | 
			
		||||
   while ( true )
 | 
			
		||||
   {
 | 
			
		||||
      if ( !readToken(skip) )
 | 
			
		||||
         errors_.resize( errorCount ); // discard errors caused by recovery
 | 
			
		||||
      if ( skip.type_ == skipUntilToken  ||  skip.type_ == tokenEndOfStream )
 | 
			
		||||
         break;
 | 
			
		||||
   }
 | 
			
		||||
   errors_.resize( errorCount );
 | 
			
		||||
   return false;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
bool 
 | 
			
		||||
Reader::addErrorAndRecover( const std::string &message, 
 | 
			
		||||
                            Token &token,
 | 
			
		||||
                            TokenType skipUntilToken )
 | 
			
		||||
{
 | 
			
		||||
   addError( message, token );
 | 
			
		||||
   return recoverFromError( skipUntilToken );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
Value &
 | 
			
		||||
Reader::currentValue()
 | 
			
		||||
{
 | 
			
		||||
   return *(nodes_.top());
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
Reader::Char 
 | 
			
		||||
Reader::getNextChar()
 | 
			
		||||
{
 | 
			
		||||
   if ( current_ == end_ )
 | 
			
		||||
      return 0;
 | 
			
		||||
   return *current_++;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
Reader::getLocationLineAndColumn( Location location,
 | 
			
		||||
                                  int &line,
 | 
			
		||||
                                  int &column ) const
 | 
			
		||||
{
 | 
			
		||||
   Location current = begin_;
 | 
			
		||||
   Location lastLineStart = current;
 | 
			
		||||
   line = 0;
 | 
			
		||||
   while ( current < location  &&  current != end_ )
 | 
			
		||||
   {
 | 
			
		||||
      Char c = *current++;
 | 
			
		||||
      if ( c == '\r' )
 | 
			
		||||
      {
 | 
			
		||||
         if ( *current == '\n' )
 | 
			
		||||
            ++current;
 | 
			
		||||
         lastLineStart = current;
 | 
			
		||||
         ++line;
 | 
			
		||||
      }
 | 
			
		||||
      else if ( c == '\n' )
 | 
			
		||||
      {
 | 
			
		||||
         lastLineStart = current;
 | 
			
		||||
         ++line;
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
   // column & line start at 1
 | 
			
		||||
   column = int(location - lastLineStart) + 1;
 | 
			
		||||
   ++line;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
std::string
 | 
			
		||||
Reader::getLocationLineAndColumn( Location location ) const
 | 
			
		||||
{
 | 
			
		||||
   int line, column;
 | 
			
		||||
   getLocationLineAndColumn( location, line, column );
 | 
			
		||||
   char buffer[18+16+16+1];
 | 
			
		||||
   sprintf( buffer, "Line %d, Column %d", line, column );
 | 
			
		||||
   return buffer;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
std::string 
 | 
			
		||||
Reader::getFormatedErrorMessages() const
 | 
			
		||||
{
 | 
			
		||||
   std::string formattedMessage;
 | 
			
		||||
   for ( Errors::const_iterator itError = errors_.begin();
 | 
			
		||||
         itError != errors_.end();
 | 
			
		||||
         ++itError )
 | 
			
		||||
   {
 | 
			
		||||
      const ErrorInfo &error = *itError;
 | 
			
		||||
      formattedMessage += "* " + getLocationLineAndColumn( error.token_.start_ ) + "\n";
 | 
			
		||||
      formattedMessage += "  " + error.message_ + "\n";
 | 
			
		||||
      if ( error.extra_ )
 | 
			
		||||
         formattedMessage += "See " + getLocationLineAndColumn( error.extra_ ) + " for detail.\n";
 | 
			
		||||
   }
 | 
			
		||||
   return formattedMessage;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
std::istream& operator>>( std::istream &sin, Value &root )
 | 
			
		||||
{
 | 
			
		||||
    Json::Reader reader;
 | 
			
		||||
    bool ok = reader.parse(sin, root, true);
 | 
			
		||||
    //JSON_ASSERT( ok );
 | 
			
		||||
    if (!ok) throw std::runtime_error(reader.getFormatedErrorMessages());
 | 
			
		||||
    return sin;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
} // namespace Json
 | 
			
		||||
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							| 
						 | 
				
			
			@ -1,292 +0,0 @@
 | 
			
		|||
// included by json_value.cpp
 | 
			
		||||
// everything is within Json namespace
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// class ValueIteratorBase
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
 | 
			
		||||
ValueIteratorBase::ValueIteratorBase()
 | 
			
		||||
#ifndef JSON_VALUE_USE_INTERNAL_MAP
 | 
			
		||||
   : current_()
 | 
			
		||||
   , isNull_( true )
 | 
			
		||||
{
 | 
			
		||||
}
 | 
			
		||||
#else
 | 
			
		||||
   : isArray_( true )
 | 
			
		||||
   , isNull_( true )
 | 
			
		||||
{
 | 
			
		||||
   iterator_.array_ = ValueInternalArray::IteratorState();
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
#ifndef JSON_VALUE_USE_INTERNAL_MAP
 | 
			
		||||
ValueIteratorBase::ValueIteratorBase( const Value::ObjectValues::iterator ¤t )
 | 
			
		||||
   : current_( current )
 | 
			
		||||
   , isNull_( false )
 | 
			
		||||
{
 | 
			
		||||
}
 | 
			
		||||
#else
 | 
			
		||||
ValueIteratorBase::ValueIteratorBase( const ValueInternalArray::IteratorState &state )
 | 
			
		||||
   : isArray_( true )
 | 
			
		||||
{
 | 
			
		||||
   iterator_.array_ = state;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
ValueIteratorBase::ValueIteratorBase( const ValueInternalMap::IteratorState &state )
 | 
			
		||||
   : isArray_( false )
 | 
			
		||||
{
 | 
			
		||||
   iterator_.map_ = state;
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
Value &
 | 
			
		||||
ValueIteratorBase::deref() const
 | 
			
		||||
{
 | 
			
		||||
#ifndef JSON_VALUE_USE_INTERNAL_MAP
 | 
			
		||||
   return current_->second;
 | 
			
		||||
#else
 | 
			
		||||
   if ( isArray_ )
 | 
			
		||||
      return ValueInternalArray::dereference( iterator_.array_ );
 | 
			
		||||
   return ValueInternalMap::value( iterator_.map_ );
 | 
			
		||||
#endif
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
ValueIteratorBase::increment()
 | 
			
		||||
{
 | 
			
		||||
#ifndef JSON_VALUE_USE_INTERNAL_MAP
 | 
			
		||||
   ++current_;
 | 
			
		||||
#else
 | 
			
		||||
   if ( isArray_ )
 | 
			
		||||
      ValueInternalArray::increment( iterator_.array_ );
 | 
			
		||||
   ValueInternalMap::increment( iterator_.map_ );
 | 
			
		||||
#endif
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
ValueIteratorBase::decrement()
 | 
			
		||||
{
 | 
			
		||||
#ifndef JSON_VALUE_USE_INTERNAL_MAP
 | 
			
		||||
   --current_;
 | 
			
		||||
#else
 | 
			
		||||
   if ( isArray_ )
 | 
			
		||||
      ValueInternalArray::decrement( iterator_.array_ );
 | 
			
		||||
   ValueInternalMap::decrement( iterator_.map_ );
 | 
			
		||||
#endif
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
ValueIteratorBase::difference_type 
 | 
			
		||||
ValueIteratorBase::computeDistance( const SelfType &other ) const
 | 
			
		||||
{
 | 
			
		||||
#ifndef JSON_VALUE_USE_INTERNAL_MAP
 | 
			
		||||
# ifdef JSON_USE_CPPTL_SMALLMAP
 | 
			
		||||
   return current_ - other.current_;
 | 
			
		||||
# else
 | 
			
		||||
   // Iterator for null value are initialized using the default
 | 
			
		||||
   // constructor, which initialize current_ to the default
 | 
			
		||||
   // std::map::iterator. As begin() and end() are two instance 
 | 
			
		||||
   // of the default std::map::iterator, they can not be compared.
 | 
			
		||||
   // To allow this, we handle this comparison specifically.
 | 
			
		||||
   if ( isNull_  &&  other.isNull_ )
 | 
			
		||||
   {
 | 
			
		||||
      return 0;
 | 
			
		||||
   }
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
   // Usage of std::distance is not portable (does not compile with Sun Studio 12 RogueWave STL,
 | 
			
		||||
   // which is the one used by default).
 | 
			
		||||
   // Using a portable hand-made version for non random iterator instead:
 | 
			
		||||
   //   return difference_type( std::distance( current_, other.current_ ) );
 | 
			
		||||
   difference_type myDistance = 0;
 | 
			
		||||
   for ( Value::ObjectValues::iterator it = current_; it != other.current_; ++it )
 | 
			
		||||
   {
 | 
			
		||||
      ++myDistance;
 | 
			
		||||
   }
 | 
			
		||||
   return myDistance;
 | 
			
		||||
# endif
 | 
			
		||||
#else
 | 
			
		||||
   if ( isArray_ )
 | 
			
		||||
      return ValueInternalArray::distance( iterator_.array_, other.iterator_.array_ );
 | 
			
		||||
   return ValueInternalMap::distance( iterator_.map_, other.iterator_.map_ );
 | 
			
		||||
#endif
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
bool 
 | 
			
		||||
ValueIteratorBase::isEqual( const SelfType &other ) const
 | 
			
		||||
{
 | 
			
		||||
#ifndef JSON_VALUE_USE_INTERNAL_MAP
 | 
			
		||||
   if ( isNull_ )
 | 
			
		||||
   {
 | 
			
		||||
      return other.isNull_;
 | 
			
		||||
   }
 | 
			
		||||
   return current_ == other.current_;
 | 
			
		||||
#else
 | 
			
		||||
   if ( isArray_ )
 | 
			
		||||
      return ValueInternalArray::equals( iterator_.array_, other.iterator_.array_ );
 | 
			
		||||
   return ValueInternalMap::equals( iterator_.map_, other.iterator_.map_ );
 | 
			
		||||
#endif
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
ValueIteratorBase::copy( const SelfType &other )
 | 
			
		||||
{
 | 
			
		||||
#ifndef JSON_VALUE_USE_INTERNAL_MAP
 | 
			
		||||
   current_ = other.current_;
 | 
			
		||||
#else
 | 
			
		||||
   if ( isArray_ )
 | 
			
		||||
      iterator_.array_ = other.iterator_.array_;
 | 
			
		||||
   iterator_.map_ = other.iterator_.map_;
 | 
			
		||||
#endif
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
Value 
 | 
			
		||||
ValueIteratorBase::key() const
 | 
			
		||||
{
 | 
			
		||||
#ifndef JSON_VALUE_USE_INTERNAL_MAP
 | 
			
		||||
   const Value::CZString czstring = (*current_).first;
 | 
			
		||||
   if ( czstring.c_str() )
 | 
			
		||||
   {
 | 
			
		||||
      if ( czstring.isStaticString() )
 | 
			
		||||
         return Value( StaticString( czstring.c_str() ) );
 | 
			
		||||
      return Value( czstring.c_str() );
 | 
			
		||||
   }
 | 
			
		||||
   return Value( czstring.index() );
 | 
			
		||||
#else
 | 
			
		||||
   if ( isArray_ )
 | 
			
		||||
      return Value( ValueInternalArray::indexOf( iterator_.array_ ) );
 | 
			
		||||
   bool isStatic;
 | 
			
		||||
   const char *memberName = ValueInternalMap::key( iterator_.map_, isStatic );
 | 
			
		||||
   if ( isStatic )
 | 
			
		||||
      return Value( StaticString( memberName ) );
 | 
			
		||||
   return Value( memberName );
 | 
			
		||||
#endif
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
UInt 
 | 
			
		||||
ValueIteratorBase::index() const
 | 
			
		||||
{
 | 
			
		||||
#ifndef JSON_VALUE_USE_INTERNAL_MAP
 | 
			
		||||
   const Value::CZString czstring = (*current_).first;
 | 
			
		||||
   if ( !czstring.c_str() )
 | 
			
		||||
      return czstring.index();
 | 
			
		||||
   return Value::UInt( -1 );
 | 
			
		||||
#else
 | 
			
		||||
   if ( isArray_ )
 | 
			
		||||
      return Value::UInt( ValueInternalArray::indexOf( iterator_.array_ ) );
 | 
			
		||||
   return Value::UInt( -1 );
 | 
			
		||||
#endif
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
const char *
 | 
			
		||||
ValueIteratorBase::memberName() const
 | 
			
		||||
{
 | 
			
		||||
#ifndef JSON_VALUE_USE_INTERNAL_MAP
 | 
			
		||||
   const char *name = (*current_).first.c_str();
 | 
			
		||||
   return name ? name : "";
 | 
			
		||||
#else
 | 
			
		||||
   if ( !isArray_ )
 | 
			
		||||
      return ValueInternalMap::key( iterator_.map_ );
 | 
			
		||||
   return "";
 | 
			
		||||
#endif
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// class ValueConstIterator
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
 | 
			
		||||
ValueConstIterator::ValueConstIterator()
 | 
			
		||||
{
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
#ifndef JSON_VALUE_USE_INTERNAL_MAP
 | 
			
		||||
ValueConstIterator::ValueConstIterator( const Value::ObjectValues::iterator ¤t )
 | 
			
		||||
   : ValueIteratorBase( current )
 | 
			
		||||
{
 | 
			
		||||
}
 | 
			
		||||
#else
 | 
			
		||||
ValueConstIterator::ValueConstIterator( const ValueInternalArray::IteratorState &state )
 | 
			
		||||
   : ValueIteratorBase( state )
 | 
			
		||||
{
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
ValueConstIterator::ValueConstIterator( const ValueInternalMap::IteratorState &state )
 | 
			
		||||
   : ValueIteratorBase( state )
 | 
			
		||||
{
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
ValueConstIterator &
 | 
			
		||||
ValueConstIterator::operator =( const ValueIteratorBase &other )
 | 
			
		||||
{
 | 
			
		||||
   copy( other );
 | 
			
		||||
   return *this;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// class ValueIterator
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
 | 
			
		||||
ValueIterator::ValueIterator()
 | 
			
		||||
{
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
#ifndef JSON_VALUE_USE_INTERNAL_MAP
 | 
			
		||||
ValueIterator::ValueIterator( const Value::ObjectValues::iterator ¤t )
 | 
			
		||||
   : ValueIteratorBase( current )
 | 
			
		||||
{
 | 
			
		||||
}
 | 
			
		||||
#else
 | 
			
		||||
ValueIterator::ValueIterator( const ValueInternalArray::IteratorState &state )
 | 
			
		||||
   : ValueIteratorBase( state )
 | 
			
		||||
{
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
ValueIterator::ValueIterator( const ValueInternalMap::IteratorState &state )
 | 
			
		||||
   : ValueIteratorBase( state )
 | 
			
		||||
{
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
ValueIterator::ValueIterator( const ValueConstIterator &other )
 | 
			
		||||
   : ValueIteratorBase( other )
 | 
			
		||||
{
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
ValueIterator::ValueIterator( const ValueIterator &other )
 | 
			
		||||
   : ValueIteratorBase( other )
 | 
			
		||||
{
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
ValueIterator &
 | 
			
		||||
ValueIterator::operator =( const SelfType &other )
 | 
			
		||||
{
 | 
			
		||||
   copy( other );
 | 
			
		||||
   return *this;
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			@ -1,829 +0,0 @@
 | 
			
		|||
#include "writer.h"
 | 
			
		||||
#include <utility>
 | 
			
		||||
#include <assert.h>
 | 
			
		||||
#include <stdio.h>
 | 
			
		||||
#include <string.h>
 | 
			
		||||
#include <iostream>
 | 
			
		||||
#include <sstream>
 | 
			
		||||
#include <iomanip>
 | 
			
		||||
 | 
			
		||||
#if _MSC_VER >= 1400 // VC++ 8.0
 | 
			
		||||
#pragma warning( disable : 4996 )   // disable warning about strdup being deprecated.
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
namespace Json {
 | 
			
		||||
 | 
			
		||||
static bool isControlCharacter(char ch)
 | 
			
		||||
{
 | 
			
		||||
   return ch > 0 && ch <= 0x1F;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
static bool containsControlCharacter( const char* str )
 | 
			
		||||
{
 | 
			
		||||
   while ( *str ) 
 | 
			
		||||
   {
 | 
			
		||||
      if ( isControlCharacter( *(str++) ) )
 | 
			
		||||
         return true;
 | 
			
		||||
   }
 | 
			
		||||
   return false;
 | 
			
		||||
}
 | 
			
		||||
static void uintToString( unsigned int value, 
 | 
			
		||||
                          char *¤t )
 | 
			
		||||
{
 | 
			
		||||
   *--current = 0;
 | 
			
		||||
   do
 | 
			
		||||
   {
 | 
			
		||||
      *--current = (value % 10) + '0';
 | 
			
		||||
      value /= 10;
 | 
			
		||||
   }
 | 
			
		||||
   while ( value != 0 );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
std::string valueToString( Int value )
 | 
			
		||||
{
 | 
			
		||||
   char buffer[32];
 | 
			
		||||
   char *current = buffer + sizeof(buffer);
 | 
			
		||||
   bool isNegative = value < 0;
 | 
			
		||||
   if ( isNegative )
 | 
			
		||||
      value = -value;
 | 
			
		||||
   uintToString( UInt(value), current );
 | 
			
		||||
   if ( isNegative )
 | 
			
		||||
      *--current = '-';
 | 
			
		||||
   assert( current >= buffer );
 | 
			
		||||
   return current;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
std::string valueToString( UInt value )
 | 
			
		||||
{
 | 
			
		||||
   char buffer[32];
 | 
			
		||||
   char *current = buffer + sizeof(buffer);
 | 
			
		||||
   uintToString( value, current );
 | 
			
		||||
   assert( current >= buffer );
 | 
			
		||||
   return current;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
std::string valueToString( double value )
 | 
			
		||||
{
 | 
			
		||||
   char buffer[32];
 | 
			
		||||
#if defined(_MSC_VER) && defined(__STDC_SECURE_LIB__) // Use secure version with visual studio 2005 to avoid warning. 
 | 
			
		||||
   sprintf_s(buffer, sizeof(buffer), "%#.16g", value); 
 | 
			
		||||
#else	
 | 
			
		||||
   sprintf(buffer, "%#.16g", value); 
 | 
			
		||||
#endif
 | 
			
		||||
   char* ch = buffer + strlen(buffer) - 1;
 | 
			
		||||
   if (*ch != '0') return buffer; // nothing to truncate, so save time
 | 
			
		||||
   while(ch > buffer && *ch == '0'){
 | 
			
		||||
     --ch;
 | 
			
		||||
   }
 | 
			
		||||
   char* last_nonzero = ch;
 | 
			
		||||
   while(ch >= buffer){
 | 
			
		||||
     switch(*ch){
 | 
			
		||||
     case '0':
 | 
			
		||||
     case '1':
 | 
			
		||||
     case '2':
 | 
			
		||||
     case '3':
 | 
			
		||||
     case '4':
 | 
			
		||||
     case '5':
 | 
			
		||||
     case '6':
 | 
			
		||||
     case '7':
 | 
			
		||||
     case '8':
 | 
			
		||||
     case '9':
 | 
			
		||||
       --ch;
 | 
			
		||||
       continue;
 | 
			
		||||
     case '.':
 | 
			
		||||
       // Truncate zeroes to save bytes in output, but keep one.
 | 
			
		||||
       *(last_nonzero+2) = '\0';
 | 
			
		||||
       return buffer;
 | 
			
		||||
     default:
 | 
			
		||||
       return buffer;
 | 
			
		||||
     }
 | 
			
		||||
   }
 | 
			
		||||
   return buffer;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
std::string valueToString( bool value )
 | 
			
		||||
{
 | 
			
		||||
   return value ? "true" : "false";
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
std::string valueToQuotedString( const char *value )
 | 
			
		||||
{
 | 
			
		||||
   // Not sure how to handle unicode...
 | 
			
		||||
   if (strpbrk(value, "\"\\\b\f\n\r\t") == NULL && !containsControlCharacter( value ))
 | 
			
		||||
      return std::string("\"") + value + "\"";
 | 
			
		||||
   // We have to walk value and escape any special characters.
 | 
			
		||||
   // Appending to std::string is not efficient, but this should be rare.
 | 
			
		||||
   // (Note: forward slashes are *not* rare, but I am not escaping them.)
 | 
			
		||||
   unsigned maxsize = strlen(value)*2 + 3; // allescaped+quotes+NULL
 | 
			
		||||
   std::string result;
 | 
			
		||||
   result.reserve(maxsize); // to avoid lots of mallocs
 | 
			
		||||
   result += "\"";
 | 
			
		||||
   for (const char* c=value; *c != 0; ++c)
 | 
			
		||||
   {
 | 
			
		||||
      switch(*c)
 | 
			
		||||
      {
 | 
			
		||||
         case '\"':
 | 
			
		||||
            result += "\\\"";
 | 
			
		||||
            break;
 | 
			
		||||
         case '\\':
 | 
			
		||||
            result += "\\\\";
 | 
			
		||||
            break;
 | 
			
		||||
         case '\b':
 | 
			
		||||
            result += "\\b";
 | 
			
		||||
            break;
 | 
			
		||||
         case '\f':
 | 
			
		||||
            result += "\\f";
 | 
			
		||||
            break;
 | 
			
		||||
         case '\n':
 | 
			
		||||
            result += "\\n";
 | 
			
		||||
            break;
 | 
			
		||||
         case '\r':
 | 
			
		||||
            result += "\\r";
 | 
			
		||||
            break;
 | 
			
		||||
         case '\t':
 | 
			
		||||
            result += "\\t";
 | 
			
		||||
            break;
 | 
			
		||||
         //case '/':
 | 
			
		||||
            // Even though \/ is considered a legal escape in JSON, a bare
 | 
			
		||||
            // slash is also legal, so I see no reason to escape it.
 | 
			
		||||
            // (I hope I am not misunderstanding something.
 | 
			
		||||
            // blep notes: actually escaping \/ may be useful in javascript to avoid </ 
 | 
			
		||||
            // sequence.
 | 
			
		||||
            // Should add a flag to allow this compatibility mode and prevent this 
 | 
			
		||||
            // sequence from occurring.
 | 
			
		||||
         default:
 | 
			
		||||
            if ( isControlCharacter( *c ) )
 | 
			
		||||
            {
 | 
			
		||||
               std::ostringstream oss;
 | 
			
		||||
               oss << "\\u" << std::hex << std::uppercase << std::setfill('0') << std::setw(4) << static_cast<int>(*c);
 | 
			
		||||
               result += oss.str();
 | 
			
		||||
            }
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
               result += *c;
 | 
			
		||||
            }
 | 
			
		||||
            break;
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
   result += "\"";
 | 
			
		||||
   return result;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Class Writer
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
Writer::~Writer()
 | 
			
		||||
{
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
// Class FastWriter
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
 | 
			
		||||
FastWriter::FastWriter()
 | 
			
		||||
   : yamlCompatiblityEnabled_( false )
 | 
			
		||||
{
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
FastWriter::enableYAMLCompatibility()
 | 
			
		||||
{
 | 
			
		||||
   yamlCompatiblityEnabled_ = true;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
std::string 
 | 
			
		||||
FastWriter::write( const Value &root )
 | 
			
		||||
{
 | 
			
		||||
   document_ = "";
 | 
			
		||||
   writeValue( root );
 | 
			
		||||
   document_ += "\n";
 | 
			
		||||
   return document_;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
FastWriter::writeValue( const Value &value )
 | 
			
		||||
{
 | 
			
		||||
   switch ( value.type() )
 | 
			
		||||
   {
 | 
			
		||||
   case nullValue:
 | 
			
		||||
      document_ += "null";
 | 
			
		||||
      break;
 | 
			
		||||
   case intValue:
 | 
			
		||||
      document_ += valueToString( value.asInt() );
 | 
			
		||||
      break;
 | 
			
		||||
   case uintValue:
 | 
			
		||||
      document_ += valueToString( value.asUInt() );
 | 
			
		||||
      break;
 | 
			
		||||
   case realValue:
 | 
			
		||||
      document_ += valueToString( value.asDouble() );
 | 
			
		||||
      break;
 | 
			
		||||
   case stringValue:
 | 
			
		||||
      document_ += valueToQuotedString( value.asCString() );
 | 
			
		||||
      break;
 | 
			
		||||
   case booleanValue:
 | 
			
		||||
      document_ += valueToString( value.asBool() );
 | 
			
		||||
      break;
 | 
			
		||||
   case arrayValue:
 | 
			
		||||
      {
 | 
			
		||||
         document_ += "[";
 | 
			
		||||
         int size = value.size();
 | 
			
		||||
         for ( int index =0; index < size; ++index )
 | 
			
		||||
         {
 | 
			
		||||
            if ( index > 0 )
 | 
			
		||||
               document_ += ",";
 | 
			
		||||
            writeValue( value[index] );
 | 
			
		||||
         }
 | 
			
		||||
         document_ += "]";
 | 
			
		||||
      }
 | 
			
		||||
      break;
 | 
			
		||||
   case objectValue:
 | 
			
		||||
      {
 | 
			
		||||
         Value::Members members( value.getMemberNames() );
 | 
			
		||||
         document_ += "{";
 | 
			
		||||
         for ( Value::Members::iterator it = members.begin(); 
 | 
			
		||||
               it != members.end(); 
 | 
			
		||||
               ++it )
 | 
			
		||||
         {
 | 
			
		||||
            const std::string &name = *it;
 | 
			
		||||
            if ( it != members.begin() )
 | 
			
		||||
               document_ += ",";
 | 
			
		||||
            document_ += valueToQuotedString( name.c_str() );
 | 
			
		||||
            document_ += yamlCompatiblityEnabled_ ? ": " 
 | 
			
		||||
                                                  : ":";
 | 
			
		||||
            writeValue( value[name] );
 | 
			
		||||
         }
 | 
			
		||||
         document_ += "}";
 | 
			
		||||
      }
 | 
			
		||||
      break;
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
// Class StyledWriter
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
 | 
			
		||||
StyledWriter::StyledWriter()
 | 
			
		||||
   : rightMargin_( 74 )
 | 
			
		||||
   , indentSize_( 3 )
 | 
			
		||||
{
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
std::string 
 | 
			
		||||
StyledWriter::write( const Value &root )
 | 
			
		||||
{
 | 
			
		||||
   document_ = "";
 | 
			
		||||
   addChildValues_ = false;
 | 
			
		||||
   indentString_ = "";
 | 
			
		||||
   writeCommentBeforeValue( root );
 | 
			
		||||
   writeValue( root );
 | 
			
		||||
   writeCommentAfterValueOnSameLine( root );
 | 
			
		||||
   document_ += "\n";
 | 
			
		||||
   return document_;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
StyledWriter::writeValue( const Value &value )
 | 
			
		||||
{
 | 
			
		||||
   switch ( value.type() )
 | 
			
		||||
   {
 | 
			
		||||
   case nullValue:
 | 
			
		||||
      pushValue( "null" );
 | 
			
		||||
      break;
 | 
			
		||||
   case intValue:
 | 
			
		||||
      pushValue( valueToString( value.asInt() ) );
 | 
			
		||||
      break;
 | 
			
		||||
   case uintValue:
 | 
			
		||||
      pushValue( valueToString( value.asUInt() ) );
 | 
			
		||||
      break;
 | 
			
		||||
   case realValue:
 | 
			
		||||
      pushValue( valueToString( value.asDouble() ) );
 | 
			
		||||
      break;
 | 
			
		||||
   case stringValue:
 | 
			
		||||
      pushValue( valueToQuotedString( value.asCString() ) );
 | 
			
		||||
      break;
 | 
			
		||||
   case booleanValue:
 | 
			
		||||
      pushValue( valueToString( value.asBool() ) );
 | 
			
		||||
      break;
 | 
			
		||||
   case arrayValue:
 | 
			
		||||
      writeArrayValue( value);
 | 
			
		||||
      break;
 | 
			
		||||
   case objectValue:
 | 
			
		||||
      {
 | 
			
		||||
         Value::Members members( value.getMemberNames() );
 | 
			
		||||
         if ( members.empty() )
 | 
			
		||||
            pushValue( "{}" );
 | 
			
		||||
         else
 | 
			
		||||
         {
 | 
			
		||||
            writeWithIndent( "{" );
 | 
			
		||||
            indent();
 | 
			
		||||
            Value::Members::iterator it = members.begin();
 | 
			
		||||
            while ( true )
 | 
			
		||||
            {
 | 
			
		||||
               const std::string &name = *it;
 | 
			
		||||
               const Value &childValue = value[name];
 | 
			
		||||
               writeCommentBeforeValue( childValue );
 | 
			
		||||
               writeWithIndent( valueToQuotedString( name.c_str() ) );
 | 
			
		||||
               document_ += " : ";
 | 
			
		||||
               writeValue( childValue );
 | 
			
		||||
               if ( ++it == members.end() )
 | 
			
		||||
               {
 | 
			
		||||
                  writeCommentAfterValueOnSameLine( childValue );
 | 
			
		||||
                  break;
 | 
			
		||||
               }
 | 
			
		||||
               document_ += ",";
 | 
			
		||||
               writeCommentAfterValueOnSameLine( childValue );
 | 
			
		||||
            }
 | 
			
		||||
            unindent();
 | 
			
		||||
            writeWithIndent( "}" );
 | 
			
		||||
         }
 | 
			
		||||
      }
 | 
			
		||||
      break;
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
StyledWriter::writeArrayValue( const Value &value )
 | 
			
		||||
{
 | 
			
		||||
   unsigned size = value.size();
 | 
			
		||||
   if ( size == 0 )
 | 
			
		||||
      pushValue( "[]" );
 | 
			
		||||
   else
 | 
			
		||||
   {
 | 
			
		||||
      bool isArrayMultiLine = isMultineArray( value );
 | 
			
		||||
      if ( isArrayMultiLine )
 | 
			
		||||
      {
 | 
			
		||||
         writeWithIndent( "[" );
 | 
			
		||||
         indent();
 | 
			
		||||
         bool hasChildValue = !childValues_.empty();
 | 
			
		||||
         unsigned index =0;
 | 
			
		||||
         while ( true )
 | 
			
		||||
         {
 | 
			
		||||
            const Value &childValue = value[index];
 | 
			
		||||
            writeCommentBeforeValue( childValue );
 | 
			
		||||
            if ( hasChildValue )
 | 
			
		||||
               writeWithIndent( childValues_[index] );
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
               writeIndent();
 | 
			
		||||
               writeValue( childValue );
 | 
			
		||||
            }
 | 
			
		||||
            if ( ++index == size )
 | 
			
		||||
            {
 | 
			
		||||
               writeCommentAfterValueOnSameLine( childValue );
 | 
			
		||||
               break;
 | 
			
		||||
            }
 | 
			
		||||
            document_ += ",";
 | 
			
		||||
            writeCommentAfterValueOnSameLine( childValue );
 | 
			
		||||
         }
 | 
			
		||||
         unindent();
 | 
			
		||||
         writeWithIndent( "]" );
 | 
			
		||||
      }
 | 
			
		||||
      else // output on a single line
 | 
			
		||||
      {
 | 
			
		||||
         assert( childValues_.size() == size );
 | 
			
		||||
         document_ += "[ ";
 | 
			
		||||
         for ( unsigned index =0; index < size; ++index )
 | 
			
		||||
         {
 | 
			
		||||
            if ( index > 0 )
 | 
			
		||||
               document_ += ", ";
 | 
			
		||||
            document_ += childValues_[index];
 | 
			
		||||
         }
 | 
			
		||||
         document_ += " ]";
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
bool 
 | 
			
		||||
StyledWriter::isMultineArray( const Value &value )
 | 
			
		||||
{
 | 
			
		||||
   int size = value.size();
 | 
			
		||||
   bool isMultiLine = size*3 >= rightMargin_ ;
 | 
			
		||||
   childValues_.clear();
 | 
			
		||||
   for ( int index =0; index < size  &&  !isMultiLine; ++index )
 | 
			
		||||
   {
 | 
			
		||||
      const Value &childValue = value[index];
 | 
			
		||||
      isMultiLine = isMultiLine  ||
 | 
			
		||||
                     ( (childValue.isArray()  ||  childValue.isObject())  &&  
 | 
			
		||||
                        childValue.size() > 0 );
 | 
			
		||||
   }
 | 
			
		||||
   if ( !isMultiLine ) // check if line length > max line length
 | 
			
		||||
   {
 | 
			
		||||
      childValues_.reserve( size );
 | 
			
		||||
      addChildValues_ = true;
 | 
			
		||||
      int lineLength = 4 + (size-1)*2; // '[ ' + ', '*n + ' ]'
 | 
			
		||||
      for ( int index =0; index < size  &&  !isMultiLine; ++index )
 | 
			
		||||
      {
 | 
			
		||||
         writeValue( value[index] );
 | 
			
		||||
         lineLength += int( childValues_[index].length() );
 | 
			
		||||
         isMultiLine = isMultiLine  &&  hasCommentForValue( value[index] );
 | 
			
		||||
      }
 | 
			
		||||
      addChildValues_ = false;
 | 
			
		||||
      isMultiLine = isMultiLine  ||  lineLength >= rightMargin_;
 | 
			
		||||
   }
 | 
			
		||||
   return isMultiLine;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
StyledWriter::pushValue( const std::string &value )
 | 
			
		||||
{
 | 
			
		||||
   if ( addChildValues_ )
 | 
			
		||||
      childValues_.push_back( value );
 | 
			
		||||
   else
 | 
			
		||||
      document_ += value;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
StyledWriter::writeIndent()
 | 
			
		||||
{
 | 
			
		||||
   if ( !document_.empty() )
 | 
			
		||||
   {
 | 
			
		||||
      char last = document_[document_.length()-1];
 | 
			
		||||
      if ( last == ' ' )     // already indented
 | 
			
		||||
         return;
 | 
			
		||||
      if ( last != '\n' )    // Comments may add new-line
 | 
			
		||||
         document_ += '\n';
 | 
			
		||||
   }
 | 
			
		||||
   document_ += indentString_;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
StyledWriter::writeWithIndent( const std::string &value )
 | 
			
		||||
{
 | 
			
		||||
   writeIndent();
 | 
			
		||||
   document_ += value;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
StyledWriter::indent()
 | 
			
		||||
{
 | 
			
		||||
   indentString_ += std::string( indentSize_, ' ' );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
StyledWriter::unindent()
 | 
			
		||||
{
 | 
			
		||||
   assert( int(indentString_.size()) >= indentSize_ );
 | 
			
		||||
   indentString_.resize( indentString_.size() - indentSize_ );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
StyledWriter::writeCommentBeforeValue( const Value &root )
 | 
			
		||||
{
 | 
			
		||||
   if ( !root.hasComment( commentBefore ) )
 | 
			
		||||
      return;
 | 
			
		||||
   document_ += normalizeEOL( root.getComment( commentBefore ) );
 | 
			
		||||
   document_ += "\n";
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
StyledWriter::writeCommentAfterValueOnSameLine( const Value &root )
 | 
			
		||||
{
 | 
			
		||||
   if ( root.hasComment( commentAfterOnSameLine ) )
 | 
			
		||||
      document_ += " " + normalizeEOL( root.getComment( commentAfterOnSameLine ) );
 | 
			
		||||
 | 
			
		||||
   if ( root.hasComment( commentAfter ) )
 | 
			
		||||
   {
 | 
			
		||||
      document_ += "\n";
 | 
			
		||||
      document_ += normalizeEOL( root.getComment( commentAfter ) );
 | 
			
		||||
      document_ += "\n";
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
bool 
 | 
			
		||||
StyledWriter::hasCommentForValue( const Value &value )
 | 
			
		||||
{
 | 
			
		||||
   return value.hasComment( commentBefore )
 | 
			
		||||
          ||  value.hasComment( commentAfterOnSameLine )
 | 
			
		||||
          ||  value.hasComment( commentAfter );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
std::string 
 | 
			
		||||
StyledWriter::normalizeEOL( const std::string &text )
 | 
			
		||||
{
 | 
			
		||||
   std::string normalized;
 | 
			
		||||
   normalized.reserve( text.length() );
 | 
			
		||||
   const char *begin = text.c_str();
 | 
			
		||||
   const char *end = begin + text.length();
 | 
			
		||||
   const char *current = begin;
 | 
			
		||||
   while ( current != end )
 | 
			
		||||
   {
 | 
			
		||||
      char c = *current++;
 | 
			
		||||
      if ( c == '\r' ) // mac or dos EOL
 | 
			
		||||
      {
 | 
			
		||||
         if ( *current == '\n' ) // convert dos EOL
 | 
			
		||||
            ++current;
 | 
			
		||||
         normalized += '\n';
 | 
			
		||||
      }
 | 
			
		||||
      else // handle unix EOL & other char
 | 
			
		||||
         normalized += c;
 | 
			
		||||
   }
 | 
			
		||||
   return normalized;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
// Class StyledStreamWriter
 | 
			
		||||
// //////////////////////////////////////////////////////////////////
 | 
			
		||||
 | 
			
		||||
StyledStreamWriter::StyledStreamWriter( std::string indentation )
 | 
			
		||||
   : document_(NULL)
 | 
			
		||||
   , rightMargin_( 74 )
 | 
			
		||||
   , indentation_( indentation )
 | 
			
		||||
{
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void
 | 
			
		||||
StyledStreamWriter::write( std::ostream &out, const Value &root )
 | 
			
		||||
{
 | 
			
		||||
   document_ = &out;
 | 
			
		||||
   addChildValues_ = false;
 | 
			
		||||
   indentString_ = "";
 | 
			
		||||
   writeCommentBeforeValue( root );
 | 
			
		||||
   writeValue( root );
 | 
			
		||||
   writeCommentAfterValueOnSameLine( root );
 | 
			
		||||
   *document_ << "\n";
 | 
			
		||||
   document_ = NULL; // Forget the stream, for safety.
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
StyledStreamWriter::writeValue( const Value &value )
 | 
			
		||||
{
 | 
			
		||||
   switch ( value.type() )
 | 
			
		||||
   {
 | 
			
		||||
   case nullValue:
 | 
			
		||||
      pushValue( "null" );
 | 
			
		||||
      break;
 | 
			
		||||
   case intValue:
 | 
			
		||||
      pushValue( valueToString( value.asInt() ) );
 | 
			
		||||
      break;
 | 
			
		||||
   case uintValue:
 | 
			
		||||
      pushValue( valueToString( value.asUInt() ) );
 | 
			
		||||
      break;
 | 
			
		||||
   case realValue:
 | 
			
		||||
      pushValue( valueToString( value.asDouble() ) );
 | 
			
		||||
      break;
 | 
			
		||||
   case stringValue:
 | 
			
		||||
      pushValue( valueToQuotedString( value.asCString() ) );
 | 
			
		||||
      break;
 | 
			
		||||
   case booleanValue:
 | 
			
		||||
      pushValue( valueToString( value.asBool() ) );
 | 
			
		||||
      break;
 | 
			
		||||
   case arrayValue:
 | 
			
		||||
      writeArrayValue( value);
 | 
			
		||||
      break;
 | 
			
		||||
   case objectValue:
 | 
			
		||||
      {
 | 
			
		||||
         Value::Members members( value.getMemberNames() );
 | 
			
		||||
         if ( members.empty() )
 | 
			
		||||
            pushValue( "{}" );
 | 
			
		||||
         else
 | 
			
		||||
         {
 | 
			
		||||
            writeWithIndent( "{" );
 | 
			
		||||
            indent();
 | 
			
		||||
            Value::Members::iterator it = members.begin();
 | 
			
		||||
            while ( true )
 | 
			
		||||
            {
 | 
			
		||||
               const std::string &name = *it;
 | 
			
		||||
               const Value &childValue = value[name];
 | 
			
		||||
               writeCommentBeforeValue( childValue );
 | 
			
		||||
               writeWithIndent( valueToQuotedString( name.c_str() ) );
 | 
			
		||||
               *document_ << " : ";
 | 
			
		||||
               writeValue( childValue );
 | 
			
		||||
               if ( ++it == members.end() )
 | 
			
		||||
               {
 | 
			
		||||
                  writeCommentAfterValueOnSameLine( childValue );
 | 
			
		||||
                  break;
 | 
			
		||||
               }
 | 
			
		||||
               *document_ << ",";
 | 
			
		||||
               writeCommentAfterValueOnSameLine( childValue );
 | 
			
		||||
            }
 | 
			
		||||
            unindent();
 | 
			
		||||
            writeWithIndent( "}" );
 | 
			
		||||
         }
 | 
			
		||||
      }
 | 
			
		||||
      break;
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
StyledStreamWriter::writeArrayValue( const Value &value )
 | 
			
		||||
{
 | 
			
		||||
   unsigned size = value.size();
 | 
			
		||||
   if ( size == 0 )
 | 
			
		||||
      pushValue( "[]" );
 | 
			
		||||
   else
 | 
			
		||||
   {
 | 
			
		||||
      bool isArrayMultiLine = isMultineArray( value );
 | 
			
		||||
      if ( isArrayMultiLine )
 | 
			
		||||
      {
 | 
			
		||||
         writeWithIndent( "[" );
 | 
			
		||||
         indent();
 | 
			
		||||
         bool hasChildValue = !childValues_.empty();
 | 
			
		||||
         unsigned index =0;
 | 
			
		||||
         while ( true )
 | 
			
		||||
         {
 | 
			
		||||
            const Value &childValue = value[index];
 | 
			
		||||
            writeCommentBeforeValue( childValue );
 | 
			
		||||
            if ( hasChildValue )
 | 
			
		||||
               writeWithIndent( childValues_[index] );
 | 
			
		||||
            else
 | 
			
		||||
            {
 | 
			
		||||
	       writeIndent();
 | 
			
		||||
               writeValue( childValue );
 | 
			
		||||
            }
 | 
			
		||||
            if ( ++index == size )
 | 
			
		||||
            {
 | 
			
		||||
               writeCommentAfterValueOnSameLine( childValue );
 | 
			
		||||
               break;
 | 
			
		||||
            }
 | 
			
		||||
            *document_ << ",";
 | 
			
		||||
            writeCommentAfterValueOnSameLine( childValue );
 | 
			
		||||
         }
 | 
			
		||||
         unindent();
 | 
			
		||||
         writeWithIndent( "]" );
 | 
			
		||||
      }
 | 
			
		||||
      else // output on a single line
 | 
			
		||||
      {
 | 
			
		||||
         assert( childValues_.size() == size );
 | 
			
		||||
         *document_ << "[ ";
 | 
			
		||||
         for ( unsigned index =0; index < size; ++index )
 | 
			
		||||
         {
 | 
			
		||||
            if ( index > 0 )
 | 
			
		||||
               *document_ << ", ";
 | 
			
		||||
            *document_ << childValues_[index];
 | 
			
		||||
         }
 | 
			
		||||
         *document_ << " ]";
 | 
			
		||||
      }
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
bool 
 | 
			
		||||
StyledStreamWriter::isMultineArray( const Value &value )
 | 
			
		||||
{
 | 
			
		||||
   int size = value.size();
 | 
			
		||||
   bool isMultiLine = size*3 >= rightMargin_ ;
 | 
			
		||||
   childValues_.clear();
 | 
			
		||||
   for ( int index =0; index < size  &&  !isMultiLine; ++index )
 | 
			
		||||
   {
 | 
			
		||||
      const Value &childValue = value[index];
 | 
			
		||||
      isMultiLine = isMultiLine  ||
 | 
			
		||||
                     ( (childValue.isArray()  ||  childValue.isObject())  &&  
 | 
			
		||||
                        childValue.size() > 0 );
 | 
			
		||||
   }
 | 
			
		||||
   if ( !isMultiLine ) // check if line length > max line length
 | 
			
		||||
   {
 | 
			
		||||
      childValues_.reserve( size );
 | 
			
		||||
      addChildValues_ = true;
 | 
			
		||||
      int lineLength = 4 + (size-1)*2; // '[ ' + ', '*n + ' ]'
 | 
			
		||||
      for ( int index =0; index < size  &&  !isMultiLine; ++index )
 | 
			
		||||
      {
 | 
			
		||||
         writeValue( value[index] );
 | 
			
		||||
         lineLength += int( childValues_[index].length() );
 | 
			
		||||
         isMultiLine = isMultiLine  &&  hasCommentForValue( value[index] );
 | 
			
		||||
      }
 | 
			
		||||
      addChildValues_ = false;
 | 
			
		||||
      isMultiLine = isMultiLine  ||  lineLength >= rightMargin_;
 | 
			
		||||
   }
 | 
			
		||||
   return isMultiLine;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
StyledStreamWriter::pushValue( const std::string &value )
 | 
			
		||||
{
 | 
			
		||||
   if ( addChildValues_ )
 | 
			
		||||
      childValues_.push_back( value );
 | 
			
		||||
   else
 | 
			
		||||
      *document_ << value;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
StyledStreamWriter::writeIndent()
 | 
			
		||||
{
 | 
			
		||||
  /*
 | 
			
		||||
    Some comments in this method would have been nice. ;-)
 | 
			
		||||
 | 
			
		||||
   if ( !document_.empty() )
 | 
			
		||||
   {
 | 
			
		||||
      char last = document_[document_.length()-1];
 | 
			
		||||
      if ( last == ' ' )     // already indented
 | 
			
		||||
         return;
 | 
			
		||||
      if ( last != '\n' )    // Comments may add new-line
 | 
			
		||||
         *document_ << '\n';
 | 
			
		||||
   }
 | 
			
		||||
  */
 | 
			
		||||
   *document_ << '\n' << indentString_;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
StyledStreamWriter::writeWithIndent( const std::string &value )
 | 
			
		||||
{
 | 
			
		||||
   writeIndent();
 | 
			
		||||
   *document_ << value;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
StyledStreamWriter::indent()
 | 
			
		||||
{
 | 
			
		||||
   indentString_ += indentation_;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
StyledStreamWriter::unindent()
 | 
			
		||||
{
 | 
			
		||||
   assert( indentString_.size() >= indentation_.size() );
 | 
			
		||||
   indentString_.resize( indentString_.size() - indentation_.size() );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
StyledStreamWriter::writeCommentBeforeValue( const Value &root )
 | 
			
		||||
{
 | 
			
		||||
   if ( !root.hasComment( commentBefore ) )
 | 
			
		||||
      return;
 | 
			
		||||
   *document_ << normalizeEOL( root.getComment( commentBefore ) );
 | 
			
		||||
   *document_ << "\n";
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
void 
 | 
			
		||||
StyledStreamWriter::writeCommentAfterValueOnSameLine( const Value &root )
 | 
			
		||||
{
 | 
			
		||||
   if ( root.hasComment( commentAfterOnSameLine ) )
 | 
			
		||||
      *document_ << " " + normalizeEOL( root.getComment( commentAfterOnSameLine ) );
 | 
			
		||||
 | 
			
		||||
   if ( root.hasComment( commentAfter ) )
 | 
			
		||||
   {
 | 
			
		||||
      *document_ << "\n";
 | 
			
		||||
      *document_ << normalizeEOL( root.getComment( commentAfter ) );
 | 
			
		||||
      *document_ << "\n";
 | 
			
		||||
   }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
bool 
 | 
			
		||||
StyledStreamWriter::hasCommentForValue( const Value &value )
 | 
			
		||||
{
 | 
			
		||||
   return value.hasComment( commentBefore )
 | 
			
		||||
          ||  value.hasComment( commentAfterOnSameLine )
 | 
			
		||||
          ||  value.hasComment( commentAfter );
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
std::string 
 | 
			
		||||
StyledStreamWriter::normalizeEOL( const std::string &text )
 | 
			
		||||
{
 | 
			
		||||
   std::string normalized;
 | 
			
		||||
   normalized.reserve( text.length() );
 | 
			
		||||
   const char *begin = text.c_str();
 | 
			
		||||
   const char *end = begin + text.length();
 | 
			
		||||
   const char *current = begin;
 | 
			
		||||
   while ( current != end )
 | 
			
		||||
   {
 | 
			
		||||
      char c = *current++;
 | 
			
		||||
      if ( c == '\r' ) // mac or dos EOL
 | 
			
		||||
      {
 | 
			
		||||
         if ( *current == '\n' ) // convert dos EOL
 | 
			
		||||
            ++current;
 | 
			
		||||
         normalized += '\n';
 | 
			
		||||
      }
 | 
			
		||||
      else // handle unix EOL & other char
 | 
			
		||||
         normalized += c;
 | 
			
		||||
   }
 | 
			
		||||
   return normalized;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
std::ostream& operator<<( std::ostream &sout, const Value &root )
 | 
			
		||||
{
 | 
			
		||||
   Json::StyledStreamWriter writer;
 | 
			
		||||
   writer.write(sout, root);
 | 
			
		||||
   return sout;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
} // namespace Json
 | 
			
		||||
| 
						 | 
				
			
			@ -1,196 +0,0 @@
 | 
			
		|||
#ifndef CPPTL_JSON_READER_H_INCLUDED
 | 
			
		||||
# define CPPTL_JSON_READER_H_INCLUDED
 | 
			
		||||
 | 
			
		||||
# include "features.h"
 | 
			
		||||
# include "value.h"
 | 
			
		||||
# include <deque>
 | 
			
		||||
# include <stack>
 | 
			
		||||
# include <string>
 | 
			
		||||
# include <iostream>
 | 
			
		||||
 | 
			
		||||
namespace Json {
 | 
			
		||||
 | 
			
		||||
   /** \brief Unserialize a <a HREF="http://www.json.org">JSON</a> document into a Value.
 | 
			
		||||
    *
 | 
			
		||||
    */
 | 
			
		||||
   class JSON_API Reader
 | 
			
		||||
   {
 | 
			
		||||
   public:
 | 
			
		||||
      typedef char Char;
 | 
			
		||||
      typedef const Char *Location;
 | 
			
		||||
 | 
			
		||||
      /** \brief Constructs a Reader allowing all features
 | 
			
		||||
       * for parsing.
 | 
			
		||||
       */
 | 
			
		||||
      Reader();
 | 
			
		||||
 | 
			
		||||
      /** \brief Constructs a Reader allowing the specified feature set
 | 
			
		||||
       * for parsing.
 | 
			
		||||
       */
 | 
			
		||||
      Reader( const Features &features );
 | 
			
		||||
 | 
			
		||||
      /** \brief Read a Value from a <a HREF="http://www.json.org">JSON</a> document.
 | 
			
		||||
       * \param document UTF-8 encoded string containing the document to read.
 | 
			
		||||
       * \param root [out] Contains the root value of the document if it was
 | 
			
		||||
       *             successfully parsed.
 | 
			
		||||
       * \param collectComments \c true to collect comment and allow writing them back during
 | 
			
		||||
       *                        serialization, \c false to discard comments.
 | 
			
		||||
       *                        This parameter is ignored if Features::allowComments_
 | 
			
		||||
       *                        is \c false.
 | 
			
		||||
       * \return \c true if the document was successfully parsed, \c false if an error occurred.
 | 
			
		||||
       */
 | 
			
		||||
      bool parse( const std::string &document, 
 | 
			
		||||
                  Value &root,
 | 
			
		||||
                  bool collectComments = true );
 | 
			
		||||
 | 
			
		||||
      /** \brief Read a Value from a <a HREF="http://www.json.org">JSON</a> document.
 | 
			
		||||
       * \param document UTF-8 encoded string containing the document to read.
 | 
			
		||||
       * \param root [out] Contains the root value of the document if it was
 | 
			
		||||
       *             successfully parsed.
 | 
			
		||||
       * \param collectComments \c true to collect comment and allow writing them back during
 | 
			
		||||
       *                        serialization, \c false to discard comments.
 | 
			
		||||
       *                        This parameter is ignored if Features::allowComments_
 | 
			
		||||
       *                        is \c false.
 | 
			
		||||
       * \return \c true if the document was successfully parsed, \c false if an error occurred.
 | 
			
		||||
       */
 | 
			
		||||
      bool parse( const char *beginDoc, const char *endDoc, 
 | 
			
		||||
                  Value &root,
 | 
			
		||||
                  bool collectComments = true );
 | 
			
		||||
 | 
			
		||||
      /// \brief Parse from input stream.
 | 
			
		||||
      /// \see Json::operator>>(std::istream&, Json::Value&).
 | 
			
		||||
      bool parse( std::istream &is,
 | 
			
		||||
                  Value &root,
 | 
			
		||||
                  bool collectComments = true );
 | 
			
		||||
 | 
			
		||||
      /** \brief Returns a user friendly string that list errors in the parsed document.
 | 
			
		||||
       * \return Formatted error message with the list of errors with their location in 
 | 
			
		||||
       *         the parsed document. An empty string is returned if no error occurred
 | 
			
		||||
       *         during parsing.
 | 
			
		||||
       */
 | 
			
		||||
      std::string getFormatedErrorMessages() const;
 | 
			
		||||
 | 
			
		||||
   private:
 | 
			
		||||
      enum TokenType
 | 
			
		||||
      {
 | 
			
		||||
         tokenEndOfStream = 0,
 | 
			
		||||
         tokenObjectBegin,
 | 
			
		||||
         tokenObjectEnd,
 | 
			
		||||
         tokenArrayBegin,
 | 
			
		||||
         tokenArrayEnd,
 | 
			
		||||
         tokenString,
 | 
			
		||||
         tokenNumber,
 | 
			
		||||
         tokenTrue,
 | 
			
		||||
         tokenFalse,
 | 
			
		||||
         tokenNull,
 | 
			
		||||
         tokenArraySeparator,
 | 
			
		||||
         tokenMemberSeparator,
 | 
			
		||||
         tokenComment,
 | 
			
		||||
         tokenError
 | 
			
		||||
      };
 | 
			
		||||
 | 
			
		||||
      class Token
 | 
			
		||||
      {
 | 
			
		||||
      public:
 | 
			
		||||
         TokenType type_;
 | 
			
		||||
         Location start_;
 | 
			
		||||
         Location end_;
 | 
			
		||||
      };
 | 
			
		||||
 | 
			
		||||
      class ErrorInfo
 | 
			
		||||
      {
 | 
			
		||||
      public:
 | 
			
		||||
         Token token_;
 | 
			
		||||
         std::string message_;
 | 
			
		||||
         Location extra_;
 | 
			
		||||
      };
 | 
			
		||||
 | 
			
		||||
      typedef std::deque<ErrorInfo> Errors;
 | 
			
		||||
 | 
			
		||||
      bool expectToken( TokenType type, Token &token, const char *message );
 | 
			
		||||
      bool readToken( Token &token );
 | 
			
		||||
      void skipSpaces();
 | 
			
		||||
      bool match( Location pattern, 
 | 
			
		||||
                  int patternLength );
 | 
			
		||||
      bool readComment();
 | 
			
		||||
      bool readCStyleComment();
 | 
			
		||||
      bool readCppStyleComment();
 | 
			
		||||
      bool readString();
 | 
			
		||||
      void readNumber();
 | 
			
		||||
      bool readValue();
 | 
			
		||||
      bool readObject( Token &token );
 | 
			
		||||
      bool readArray( Token &token );
 | 
			
		||||
      bool decodeNumber( Token &token );
 | 
			
		||||
      bool decodeString( Token &token );
 | 
			
		||||
      bool decodeString( Token &token, std::string &decoded );
 | 
			
		||||
      bool decodeDouble( Token &token );
 | 
			
		||||
      bool decodeUnicodeCodePoint( Token &token, 
 | 
			
		||||
                                   Location ¤t, 
 | 
			
		||||
                                   Location end, 
 | 
			
		||||
                                   unsigned int &unicode );
 | 
			
		||||
      bool decodeUnicodeEscapeSequence( Token &token, 
 | 
			
		||||
                                        Location ¤t, 
 | 
			
		||||
                                        Location end, 
 | 
			
		||||
                                        unsigned int &unicode );
 | 
			
		||||
      bool addError( const std::string &message, 
 | 
			
		||||
                     Token &token,
 | 
			
		||||
                     Location extra = 0 );
 | 
			
		||||
      bool recoverFromError( TokenType skipUntilToken );
 | 
			
		||||
      bool addErrorAndRecover( const std::string &message, 
 | 
			
		||||
                               Token &token,
 | 
			
		||||
                               TokenType skipUntilToken );
 | 
			
		||||
      void skipUntilSpace();
 | 
			
		||||
      Value ¤tValue();
 | 
			
		||||
      Char getNextChar();
 | 
			
		||||
      void getLocationLineAndColumn( Location location,
 | 
			
		||||
                                     int &line,
 | 
			
		||||
                                     int &column ) const;
 | 
			
		||||
      std::string getLocationLineAndColumn( Location location ) const;
 | 
			
		||||
      void addComment( Location begin, 
 | 
			
		||||
                       Location end, 
 | 
			
		||||
                       CommentPlacement placement );
 | 
			
		||||
      void skipCommentTokens( Token &token );
 | 
			
		||||
   
 | 
			
		||||
      typedef std::stack<Value *> Nodes;
 | 
			
		||||
      Nodes nodes_;
 | 
			
		||||
      Errors errors_;
 | 
			
		||||
      std::string document_;
 | 
			
		||||
      Location begin_;
 | 
			
		||||
      Location end_;
 | 
			
		||||
      Location current_;
 | 
			
		||||
      Location lastValueEnd_;
 | 
			
		||||
      Value *lastValue_;
 | 
			
		||||
      std::string commentsBefore_;
 | 
			
		||||
      Features features_;
 | 
			
		||||
      bool collectComments_;
 | 
			
		||||
   };
 | 
			
		||||
 | 
			
		||||
   /** \brief Read from 'sin' into 'root'.
 | 
			
		||||
 | 
			
		||||
    Always keep comments from the input JSON.
 | 
			
		||||
 | 
			
		||||
    This can be used to read a file into a particular sub-object.
 | 
			
		||||
    For example:
 | 
			
		||||
    \code
 | 
			
		||||
    Json::Value root;
 | 
			
		||||
    cin >> root["dir"]["file"];
 | 
			
		||||
    cout << root;
 | 
			
		||||
    \endcode
 | 
			
		||||
    Result:
 | 
			
		||||
    \verbatim
 | 
			
		||||
    {
 | 
			
		||||
	"dir": {
 | 
			
		||||
	    "file": {
 | 
			
		||||
		// The input stream JSON would be nested here.
 | 
			
		||||
	    }
 | 
			
		||||
	}
 | 
			
		||||
    }
 | 
			
		||||
    \endverbatim
 | 
			
		||||
    \throw std::exception on parse error.
 | 
			
		||||
    \see Json::operator<<()
 | 
			
		||||
   */
 | 
			
		||||
   std::istream& operator>>( std::istream&, Value& );
 | 
			
		||||
 | 
			
		||||
} // namespace Json
 | 
			
		||||
 | 
			
		||||
#endif // CPPTL_JSON_READER_H_INCLUDED
 | 
			
		||||
							
								
								
									
										1069
									
								
								util/json/value.h
									
										
									
									
									
								
							
							
						
						
									
										1069
									
								
								util/json/value.h
									
										
									
									
									
								
							
										
											
												File diff suppressed because it is too large
												Load diff
											
										
									
								
							| 
						 | 
				
			
			@ -1,174 +0,0 @@
 | 
			
		|||
#ifndef JSON_WRITER_H_INCLUDED
 | 
			
		||||
# define JSON_WRITER_H_INCLUDED
 | 
			
		||||
 | 
			
		||||
# include "value.h"
 | 
			
		||||
# include <vector>
 | 
			
		||||
# include <string>
 | 
			
		||||
# include <iostream>
 | 
			
		||||
 | 
			
		||||
namespace Json {
 | 
			
		||||
 | 
			
		||||
   class Value;
 | 
			
		||||
 | 
			
		||||
   /** \brief Abstract class for writers.
 | 
			
		||||
    */
 | 
			
		||||
   class JSON_API Writer
 | 
			
		||||
   {
 | 
			
		||||
   public:
 | 
			
		||||
      virtual ~Writer();
 | 
			
		||||
 | 
			
		||||
      virtual std::string write( const Value &root ) = 0;
 | 
			
		||||
   };
 | 
			
		||||
 | 
			
		||||
   /** \brief Outputs a Value in <a HREF="http://www.json.org">JSON</a> format without formatting (not human friendly).
 | 
			
		||||
    *
 | 
			
		||||
    * The JSON document is written in a single line. It is not intended for 'human' consumption,
 | 
			
		||||
    * but may be usefull to support feature such as RPC where bandwith is limited.
 | 
			
		||||
    * \sa Reader, Value
 | 
			
		||||
    */
 | 
			
		||||
   class JSON_API FastWriter : public Writer
 | 
			
		||||
   {
 | 
			
		||||
   public:
 | 
			
		||||
      FastWriter();
 | 
			
		||||
      virtual ~FastWriter(){}
 | 
			
		||||
 | 
			
		||||
      void enableYAMLCompatibility();
 | 
			
		||||
 | 
			
		||||
   public: // overridden from Writer
 | 
			
		||||
      virtual std::string write( const Value &root );
 | 
			
		||||
 | 
			
		||||
   private:
 | 
			
		||||
      void writeValue( const Value &value );
 | 
			
		||||
 | 
			
		||||
      std::string document_;
 | 
			
		||||
      bool yamlCompatiblityEnabled_;
 | 
			
		||||
   };
 | 
			
		||||
 | 
			
		||||
   /** \brief Writes a Value in <a HREF="http://www.json.org">JSON</a> format in a human friendly way.
 | 
			
		||||
    *
 | 
			
		||||
    * The rules for line break and indent are as follow:
 | 
			
		||||
    * - Object value:
 | 
			
		||||
    *     - if empty then print {} without indent and line break
 | 
			
		||||
    *     - if not empty the print '{', line break & indent, print one value per line
 | 
			
		||||
    *       and then unindent and line break and print '}'.
 | 
			
		||||
    * - Array value:
 | 
			
		||||
    *     - if empty then print [] without indent and line break
 | 
			
		||||
    *     - if the array contains no object value, empty array or some other value types,
 | 
			
		||||
    *       and all the values fit on one lines, then print the array on a single line.
 | 
			
		||||
    *     - otherwise, it the values do not fit on one line, or the array contains
 | 
			
		||||
    *       object or non empty array, then print one value per line.
 | 
			
		||||
    *
 | 
			
		||||
    * If the Value have comments then they are outputed according to their #CommentPlacement.
 | 
			
		||||
    *
 | 
			
		||||
    * \sa Reader, Value, Value::setComment()
 | 
			
		||||
    */
 | 
			
		||||
   class JSON_API StyledWriter: public Writer
 | 
			
		||||
   {
 | 
			
		||||
   public:
 | 
			
		||||
      StyledWriter();
 | 
			
		||||
      virtual ~StyledWriter(){}
 | 
			
		||||
 | 
			
		||||
   public: // overridden from Writer
 | 
			
		||||
      /** \brief Serialize a Value in <a HREF="http://www.json.org">JSON</a> format.
 | 
			
		||||
       * \param root Value to serialize.
 | 
			
		||||
       * \return String containing the JSON document that represents the root value.
 | 
			
		||||
       */
 | 
			
		||||
      virtual std::string write( const Value &root );
 | 
			
		||||
 | 
			
		||||
   private:
 | 
			
		||||
      void writeValue( const Value &value );
 | 
			
		||||
      void writeArrayValue( const Value &value );
 | 
			
		||||
      bool isMultineArray( const Value &value );
 | 
			
		||||
      void pushValue( const std::string &value );
 | 
			
		||||
      void writeIndent();
 | 
			
		||||
      void writeWithIndent( const std::string &value );
 | 
			
		||||
      void indent();
 | 
			
		||||
      void unindent();
 | 
			
		||||
      void writeCommentBeforeValue( const Value &root );
 | 
			
		||||
      void writeCommentAfterValueOnSameLine( const Value &root );
 | 
			
		||||
      bool hasCommentForValue( const Value &value );
 | 
			
		||||
      static std::string normalizeEOL( const std::string &text );
 | 
			
		||||
 | 
			
		||||
      typedef std::vector<std::string> ChildValues;
 | 
			
		||||
 | 
			
		||||
      ChildValues childValues_;
 | 
			
		||||
      std::string document_;
 | 
			
		||||
      std::string indentString_;
 | 
			
		||||
      int rightMargin_;
 | 
			
		||||
      int indentSize_;
 | 
			
		||||
      bool addChildValues_;
 | 
			
		||||
   };
 | 
			
		||||
 | 
			
		||||
   /** \brief Writes a Value in <a HREF="http://www.json.org">JSON</a> format in a human friendly way,
 | 
			
		||||
        to a stream rather than to a string.
 | 
			
		||||
    *
 | 
			
		||||
    * The rules for line break and indent are as follow:
 | 
			
		||||
    * - Object value:
 | 
			
		||||
    *     - if empty then print {} without indent and line break
 | 
			
		||||
    *     - if not empty the print '{', line break & indent, print one value per line
 | 
			
		||||
    *       and then unindent and line break and print '}'.
 | 
			
		||||
    * - Array value:
 | 
			
		||||
    *     - if empty then print [] without indent and line break
 | 
			
		||||
    *     - if the array contains no object value, empty array or some other value types,
 | 
			
		||||
    *       and all the values fit on one lines, then print the array on a single line.
 | 
			
		||||
    *     - otherwise, it the values do not fit on one line, or the array contains
 | 
			
		||||
    *       object or non empty array, then print one value per line.
 | 
			
		||||
    *
 | 
			
		||||
    * If the Value have comments then they are outputed according to their #CommentPlacement.
 | 
			
		||||
    *
 | 
			
		||||
    * \param indentation Each level will be indented by this amount extra.
 | 
			
		||||
    * \sa Reader, Value, Value::setComment()
 | 
			
		||||
    */
 | 
			
		||||
   class JSON_API StyledStreamWriter
 | 
			
		||||
   {
 | 
			
		||||
   public:
 | 
			
		||||
      StyledStreamWriter( std::string indentation="\t" );
 | 
			
		||||
      ~StyledStreamWriter(){}
 | 
			
		||||
 | 
			
		||||
   public:
 | 
			
		||||
      /** \brief Serialize a Value in <a HREF="http://www.json.org">JSON</a> format.
 | 
			
		||||
       * \param out Stream to write to. (Can be ostringstream, e.g.)
 | 
			
		||||
       * \param root Value to serialize.
 | 
			
		||||
       * \note There is no point in deriving from Writer, since write() should not return a value.
 | 
			
		||||
       */
 | 
			
		||||
      void write( std::ostream &out, const Value &root );
 | 
			
		||||
 | 
			
		||||
   private:
 | 
			
		||||
      void writeValue( const Value &value );
 | 
			
		||||
      void writeArrayValue( const Value &value );
 | 
			
		||||
      bool isMultineArray( const Value &value );
 | 
			
		||||
      void pushValue( const std::string &value );
 | 
			
		||||
      void writeIndent();
 | 
			
		||||
      void writeWithIndent( const std::string &value );
 | 
			
		||||
      void indent();
 | 
			
		||||
      void unindent();
 | 
			
		||||
      void writeCommentBeforeValue( const Value &root );
 | 
			
		||||
      void writeCommentAfterValueOnSameLine( const Value &root );
 | 
			
		||||
      bool hasCommentForValue( const Value &value );
 | 
			
		||||
      static std::string normalizeEOL( const std::string &text );
 | 
			
		||||
 | 
			
		||||
      typedef std::vector<std::string> ChildValues;
 | 
			
		||||
 | 
			
		||||
      ChildValues childValues_;
 | 
			
		||||
      std::ostream* document_;
 | 
			
		||||
      std::string indentString_;
 | 
			
		||||
      int rightMargin_;
 | 
			
		||||
      std::string indentation_;
 | 
			
		||||
      bool addChildValues_;
 | 
			
		||||
   };
 | 
			
		||||
 | 
			
		||||
   std::string JSON_API valueToString( Int value );
 | 
			
		||||
   std::string JSON_API valueToString( UInt value );
 | 
			
		||||
   std::string JSON_API valueToString( double value );
 | 
			
		||||
   std::string JSON_API valueToString( bool value );
 | 
			
		||||
   std::string JSON_API valueToQuotedString( const char *value );
 | 
			
		||||
 | 
			
		||||
   /// \brief Output using the StyledStreamWriter.
 | 
			
		||||
   /// \see Json::operator>>()
 | 
			
		||||
   std::ostream& operator<<( std::ostream&, const Value &root );
 | 
			
		||||
 | 
			
		||||
} // namespace Json
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
#endif // JSON_WRITER_H_INCLUDED
 | 
			
		||||
| 
						 | 
				
			
			@ -1,7 +1,7 @@
 | 
			
		|||
/// \file util.cpp
 | 
			
		||||
/// Contains generic functions for managing processes and configuration.
 | 
			
		||||
/// \file procs.cpp
 | 
			
		||||
/// Contains generic functions for managing processes.
 | 
			
		||||
 | 
			
		||||
#include "util.h"
 | 
			
		||||
#include "procs.h"
 | 
			
		||||
#include <string.h>
 | 
			
		||||
#include <sys/types.h>
 | 
			
		||||
#include <signal.h>
 | 
			
		||||
| 
						 | 
				
			
			@ -16,37 +16,12 @@
 | 
			
		|||
#include <sys/types.h>
 | 
			
		||||
#include <fcntl.h>
 | 
			
		||||
#include <pwd.h>
 | 
			
		||||
#include <getopt.h>
 | 
			
		||||
#include <stdlib.h>
 | 
			
		||||
#include <stdio.h>
 | 
			
		||||
#include <fstream>
 | 
			
		||||
 | 
			
		||||
std::map<pid_t, std::string> Util::Procs::plist;
 | 
			
		||||
bool Util::Procs::handler_set = false;
 | 
			
		||||
 | 
			
		||||
/// Sets the current process' running user
 | 
			
		||||
void Util::setUser(std::string username){
 | 
			
		||||
  if (username != "root"){
 | 
			
		||||
    struct passwd * user_info = getpwnam(username.c_str());
 | 
			
		||||
    if (!user_info){
 | 
			
		||||
      #if DEBUG >= 1
 | 
			
		||||
      fprintf(stderr, "Error: could not setuid %s: could not get PID\n", username.c_str());
 | 
			
		||||
      #endif
 | 
			
		||||
      return;
 | 
			
		||||
    }else{
 | 
			
		||||
      if (setuid(user_info->pw_uid) != 0){
 | 
			
		||||
        #if DEBUG >= 1
 | 
			
		||||
        fprintf(stderr, "Error: could not setuid %s: not allowed\n", username.c_str());
 | 
			
		||||
        #endif
 | 
			
		||||
      }else{
 | 
			
		||||
        #if DEBUG >= 3
 | 
			
		||||
        fprintf(stderr, "Changed user to %s\n", username.c_str());
 | 
			
		||||
        #endif
 | 
			
		||||
      }
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Used internally to capture child signals and update plist.
 | 
			
		||||
void Util::Procs::childsig_handler(int signum){
 | 
			
		||||
  if (signum != SIGCHLD){return;}
 | 
			
		||||
| 
						 | 
				
			
			@ -263,108 +238,3 @@ std::string Util::Procs::getName(pid_t name){
 | 
			
		|||
  }
 | 
			
		||||
  return "";
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Creates a new configuration manager.
 | 
			
		||||
Util::Config::Config(){
 | 
			
		||||
  listen_port = 4242;
 | 
			
		||||
  daemon_mode = true;
 | 
			
		||||
  interface = "0.0.0.0";
 | 
			
		||||
  configfile = "/etc/ddvtech.conf";
 | 
			
		||||
  username = "root";
 | 
			
		||||
  ignore_daemon = false;
 | 
			
		||||
  ignore_interface = false;
 | 
			
		||||
  ignore_port = false;
 | 
			
		||||
  ignore_user = false;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Parses commandline arguments.
 | 
			
		||||
/// Calls exit if an unknown option is encountered, printing a help message.
 | 
			
		||||
/// confsection must be either already set or never be set at all when this function is called.
 | 
			
		||||
/// In other words: do not change confsection after calling this function.
 | 
			
		||||
void Util::Config::parseArgs(int argc, char ** argv){
 | 
			
		||||
  int opt = 0;
 | 
			
		||||
  static const char *optString = "ndvp:i:u:c:h?";
 | 
			
		||||
  static const struct option longOpts[] = {
 | 
			
		||||
    {"help",0,0,'h'},
 | 
			
		||||
    {"port",1,0,'p'},
 | 
			
		||||
    {"interface",1,0,'i'},
 | 
			
		||||
    {"username",1,0,'u'},
 | 
			
		||||
    {"no-daemon",0,0,'n'},
 | 
			
		||||
    {"daemon",0,0,'d'},
 | 
			
		||||
    {"configfile",1,0,'c'},
 | 
			
		||||
    {"version",0,0,'v'}
 | 
			
		||||
  };
 | 
			
		||||
  while ((opt = getopt_long(argc, argv, optString, longOpts, 0)) != -1){
 | 
			
		||||
    switch (opt){
 | 
			
		||||
      case 'p': listen_port = atoi(optarg); ignore_port = true; break;
 | 
			
		||||
      case 'i': interface = optarg; ignore_interface = true; break;
 | 
			
		||||
      case 'n': daemon_mode = false; ignore_daemon = true; break;
 | 
			
		||||
      case 'd': daemon_mode = true; ignore_daemon = true; break;
 | 
			
		||||
      case 'c': configfile = optarg; break;
 | 
			
		||||
      case 'u': username = optarg; ignore_user = true; break;
 | 
			
		||||
      case 'v':
 | 
			
		||||
        printf("%s\n", TOSTRING(VERSION));
 | 
			
		||||
        exit(1);
 | 
			
		||||
        break;
 | 
			
		||||
      case 'h':
 | 
			
		||||
      case '?':
 | 
			
		||||
        std::string doingdaemon = "true";
 | 
			
		||||
        if (!daemon_mode){doingdaemon = "false";}
 | 
			
		||||
        if (confsection == ""){
 | 
			
		||||
          printf("Options: -h[elp], -?, -v[ersion], -n[odaemon], -d[aemon], -p[ort] VAL, -i[nterface] VAL, -u[sername] VAL\n");
 | 
			
		||||
          printf("Defaults:\n  interface: %s\n  port: %i\n  daemon mode: %s\n  username: %s\n", interface.c_str(), listen_port, doingdaemon.c_str(), username.c_str());
 | 
			
		||||
        }else{
 | 
			
		||||
          printf("Options: -h[elp], -?, -v[ersion], -n[odaemon], -d[aemon], -p[ort] VAL, -i[nterface] VAL, -c[onfigfile] VAL, -u[sername] VAL\n");
 | 
			
		||||
          printf("Defaults:\n  interface: %s\n  port: %i\n  daemon mode: %s\n  configfile: %s\n  username: %s\n", interface.c_str(), listen_port, doingdaemon.c_str(), configfile.c_str(), username.c_str());
 | 
			
		||||
          printf("Username root means no change to UID, no matter what the UID is.\n");
 | 
			
		||||
          printf("If the configfile exists, it is always loaded first. Commandline settings then overwrite the config file.\n");
 | 
			
		||||
          printf("\nThis process takes it directives from the %s section of the configfile.\n", confsection.c_str());
 | 
			
		||||
        }
 | 
			
		||||
        printf("This is %s version %s\n", argv[0], TOSTRING(VERSION));
 | 
			
		||||
        exit(1);
 | 
			
		||||
        break;
 | 
			
		||||
    }
 | 
			
		||||
  }//commandline options parser
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Parses the configuration file at configfile, if it exists.
 | 
			
		||||
/// Assumes confsection is set.
 | 
			
		||||
void Util::Config::parseFile(){
 | 
			
		||||
  std::ifstream conf(configfile.c_str(), std::ifstream::in);
 | 
			
		||||
  std::string tmpstr;
 | 
			
		||||
  bool acc_comm = false;
 | 
			
		||||
  size_t foundeq;
 | 
			
		||||
  if (conf.fail()){
 | 
			
		||||
    #if DEBUG >= 3
 | 
			
		||||
    fprintf(stderr, "Configuration file %s not found - using build-in defaults...\n", configfile.c_str());
 | 
			
		||||
    #endif
 | 
			
		||||
  }else{
 | 
			
		||||
    while (conf.good()){
 | 
			
		||||
      getline(conf, tmpstr);
 | 
			
		||||
      if (tmpstr[0] == '['){//new section? check if we care.
 | 
			
		||||
        if (tmpstr == confsection){acc_comm = true;}else{acc_comm = false;}
 | 
			
		||||
      }else{
 | 
			
		||||
        if (!acc_comm){break;}//skip all lines in this section if we do not care about it
 | 
			
		||||
        foundeq = tmpstr.find('=');
 | 
			
		||||
        if (foundeq != std::string::npos){
 | 
			
		||||
          if ((tmpstr.substr(0, foundeq) == "port") && !ignore_port){listen_port = atoi(tmpstr.substr(foundeq+1).c_str());}
 | 
			
		||||
          if ((tmpstr.substr(0, foundeq) == "interface") && !ignore_interface){interface = tmpstr.substr(foundeq+1);}
 | 
			
		||||
          if ((tmpstr.substr(0, foundeq) == "username") && !ignore_user){username = tmpstr.substr(foundeq+1);}
 | 
			
		||||
          if ((tmpstr.substr(0, foundeq) == "daemon") && !ignore_daemon){daemon_mode = true;}
 | 
			
		||||
          if ((tmpstr.substr(0, foundeq) == "nodaemon") && !ignore_daemon){daemon_mode = false;}
 | 
			
		||||
        }//found equals sign
 | 
			
		||||
      }//section contents
 | 
			
		||||
    }//configfile line loop
 | 
			
		||||
  }//configuration
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Will turn the current process into a daemon.
 | 
			
		||||
/// Works by calling daemon(1,0):
 | 
			
		||||
/// Does not change directory to root.
 | 
			
		||||
/// Does redirect output to /dev/null
 | 
			
		||||
void Util::Daemonize(){
 | 
			
		||||
  #if DEBUG >= 3
 | 
			
		||||
  fprintf(stderr, "Going into background mode...\n");
 | 
			
		||||
  #endif
 | 
			
		||||
  daemon(1, 0);
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			@ -1,13 +1,10 @@
 | 
			
		|||
/// \file util.h
 | 
			
		||||
/// Contains generic function headers for managing processes and configuration.
 | 
			
		||||
/// \file procs.h
 | 
			
		||||
/// Contains generic function headers for managing processes.
 | 
			
		||||
 | 
			
		||||
#include <unistd.h>
 | 
			
		||||
#include <string>
 | 
			
		||||
#include <map>
 | 
			
		||||
 | 
			
		||||
#define STRINGIFY(x) #x
 | 
			
		||||
#define TOSTRING(x) STRINGIFY(x)
 | 
			
		||||
 | 
			
		||||
/// Contains utility code, not directly related to streaming media
 | 
			
		||||
namespace Util{
 | 
			
		||||
 | 
			
		||||
| 
						 | 
				
			
			@ -31,29 +28,4 @@ namespace Util{
 | 
			
		|||
      static std::string getName(pid_t name);
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  /// Will set the active user to the named username.
 | 
			
		||||
  void setUser(std::string user);
 | 
			
		||||
 | 
			
		||||
  /// Deals with parsing configuration from files or commandline options.
 | 
			
		||||
  class Config{
 | 
			
		||||
  private:
 | 
			
		||||
    bool ignore_daemon;
 | 
			
		||||
    bool ignore_interface;
 | 
			
		||||
    bool ignore_port;
 | 
			
		||||
    bool ignore_user;
 | 
			
		||||
  public:
 | 
			
		||||
    std::string confsection;
 | 
			
		||||
    std::string configfile;
 | 
			
		||||
    bool daemon_mode;
 | 
			
		||||
    std::string interface;
 | 
			
		||||
    int listen_port;
 | 
			
		||||
    std::string username;
 | 
			
		||||
    Config();
 | 
			
		||||
    void parseArgs(int argc, char ** argv);
 | 
			
		||||
    void parseFile();
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  /// Will turn the current process into a daemon.
 | 
			
		||||
  void Daemonize();
 | 
			
		||||
 | 
			
		||||
};
 | 
			
		||||
| 
						 | 
				
			
			@ -15,14 +15,8 @@
 | 
			
		|||
#endif
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
#ifndef CONFIGSECT
 | 
			
		||||
  /// Configuration file section for this server.
 | 
			
		||||
  #define CONFIGSECT None
 | 
			
		||||
  #error "No configuration file section was set!"
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#include "socket.h" //Socket library
 | 
			
		||||
#include "util.h" //utilities for process spawning and config management
 | 
			
		||||
#include "config.h" //utilities for config management
 | 
			
		||||
#include <signal.h>
 | 
			
		||||
#include <sys/types.h>
 | 
			
		||||
#include <pwd.h>
 | 
			
		||||
| 
						 | 
				
			
			@ -83,10 +77,8 @@ int main(int argc, char ** argv){
 | 
			
		|||
 | 
			
		||||
  //set and parse configuration
 | 
			
		||||
  Util::Config C;
 | 
			
		||||
  C.confsection = TOSTRING(CONFIGSECT);
 | 
			
		||||
  C.listen_port = DEFAULT_PORT;
 | 
			
		||||
  C.parseArgs(argc, argv);
 | 
			
		||||
  C.parseFile();
 | 
			
		||||
 | 
			
		||||
  //setup a new server socket, for the correct interface and port
 | 
			
		||||
  server_socket = Socket::Server(C.listen_port, C.interface);
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -3,6 +3,7 @@
 | 
			
		|||
/// Written by Jaron Vietor in 2010 for DDVTech
 | 
			
		||||
 | 
			
		||||
#include "socket.h"
 | 
			
		||||
#include <sys/stat.h>
 | 
			
		||||
#include <poll.h>
 | 
			
		||||
#include <netdb.h>
 | 
			
		||||
#include <sstream>
 | 
			
		||||
| 
						 | 
				
			
			@ -222,9 +223,10 @@ std::string Socket::Connection::getStats(std::string C){
 | 
			
		|||
}
 | 
			
		||||
 | 
			
		||||
/// Updates the downbuffer and upbuffer internal variables.
 | 
			
		||||
void Socket::Connection::spool(){
 | 
			
		||||
  iread(downbuffer);
 | 
			
		||||
/// Returns true if new data was received, false otherwise.
 | 
			
		||||
bool Socket::Connection::spool(){
 | 
			
		||||
  iwrite(upbuffer);
 | 
			
		||||
  return iread(downbuffer);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Returns a reference to the download buffer.
 | 
			
		||||
| 
						 | 
				
			
			@ -452,7 +454,7 @@ Socket::Server::Server(int port, std::string hostname, bool nonblock){
 | 
			
		|||
  sock = socket(AF_INET6, SOCK_STREAM, 0);
 | 
			
		||||
  if (sock < 0){
 | 
			
		||||
    #if DEBUG >= 1
 | 
			
		||||
    fprintf(stderr, "Could not create socket! Error: %s\n", strerror(errno));
 | 
			
		||||
    fprintf(stderr, "Could not create socket %s:%i! Error: %s\n", hostname.c_str(), port, strerror(errno));
 | 
			
		||||
    #endif
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
| 
						 | 
				
			
			@ -466,7 +468,7 @@ Socket::Server::Server(int port, std::string hostname, bool nonblock){
 | 
			
		|||
  struct sockaddr_in6 addr;
 | 
			
		||||
  addr.sin6_family = AF_INET6;
 | 
			
		||||
  addr.sin6_port = htons(port);//set port
 | 
			
		||||
  if (hostname == "0.0.0.0"){
 | 
			
		||||
  if (hostname == "0.0.0.0" || hostname.length() == 0){
 | 
			
		||||
    addr.sin6_addr = in6addr_any;
 | 
			
		||||
  }else{
 | 
			
		||||
    inet_pton(AF_INET6, hostname.c_str(), &addr.sin6_addr);//set interface, 0.0.0.0 (default) is all
 | 
			
		||||
| 
						 | 
				
			
			@ -485,14 +487,14 @@ Socket::Server::Server(int port, std::string hostname, bool nonblock){
 | 
			
		|||
    }
 | 
			
		||||
  }else{
 | 
			
		||||
    #if DEBUG >= 1
 | 
			
		||||
    fprintf(stderr, "Binding failed, retrying as IPv4... (%s)\n", strerror(errno));
 | 
			
		||||
    fprintf(stderr, "Binding %s:%i failed, retrying as IPv4... (%s)\n", hostname.c_str(), port, strerror(errno));
 | 
			
		||||
    #endif
 | 
			
		||||
    close();
 | 
			
		||||
  }
 | 
			
		||||
  sock = socket(AF_INET, SOCK_STREAM, 0);
 | 
			
		||||
  if (sock < 0){
 | 
			
		||||
    #if DEBUG >= 1
 | 
			
		||||
    fprintf(stderr, "Could not create socket! Error: %s\n", strerror(errno));
 | 
			
		||||
    fprintf(stderr, "Could not create socket %s:%i! Error: %s\n", hostname.c_str(), port, strerror(errno));
 | 
			
		||||
    #endif
 | 
			
		||||
    return;
 | 
			
		||||
  }
 | 
			
		||||
| 
						 | 
				
			
			@ -506,7 +508,7 @@ Socket::Server::Server(int port, std::string hostname, bool nonblock){
 | 
			
		|||
  struct sockaddr_in addr4;
 | 
			
		||||
  addr4.sin_family = AF_INET;
 | 
			
		||||
  addr4.sin_port = htons(port);//set port
 | 
			
		||||
  if (hostname == "0.0.0.0"){
 | 
			
		||||
  if (hostname == "0.0.0.0" || hostname.length() == 0){
 | 
			
		||||
    addr4.sin_addr.s_addr = INADDR_ANY;
 | 
			
		||||
  }else{
 | 
			
		||||
    inet_pton(AF_INET, hostname.c_str(), &addr4.sin_addr);//set interface, 0.0.0.0 (default) is all
 | 
			
		||||
| 
						 | 
				
			
			@ -525,7 +527,7 @@ Socket::Server::Server(int port, std::string hostname, bool nonblock){
 | 
			
		|||
    }
 | 
			
		||||
  }else{
 | 
			
		||||
    #if DEBUG >= 1
 | 
			
		||||
    fprintf(stderr, "IPv4 binding also failed, giving up. (%s)\n", strerror(errno));
 | 
			
		||||
    fprintf(stderr, "IPv4 binding %s:%i also failed, giving up. (%s)\n", hostname.c_str(), port, strerror(errno));
 | 
			
		||||
    #endif
 | 
			
		||||
    close();
 | 
			
		||||
    return;
 | 
			
		||||
| 
						 | 
				
			
			@ -646,3 +648,43 @@ bool Socket::Server::connected(){
 | 
			
		|||
 | 
			
		||||
/// Returns internal socket number.
 | 
			
		||||
int Socket::Server::getSocket(){return sock;}
 | 
			
		||||
 | 
			
		||||
/// Connect to a stream on the system.
 | 
			
		||||
/// Filters the streamname, removing invalid characters and
 | 
			
		||||
/// converting all letters to lowercase.
 | 
			
		||||
/// If a '?' character is found, everything following that character is deleted.
 | 
			
		||||
Socket::Connection Socket::getStream(std::string streamname){
 | 
			
		||||
  //strip anything that isn't numbers, digits or underscores
 | 
			
		||||
  for (std::string::iterator i=streamname.end()-1; i>=streamname.begin(); --i){
 | 
			
		||||
    if (*i == '?'){streamname.erase(i, streamname.end()); break;}
 | 
			
		||||
    if (!isalpha(*i) && !isdigit(*i) && *i != '_'){
 | 
			
		||||
      streamname.erase(i);
 | 
			
		||||
    }else{
 | 
			
		||||
      *i=tolower(*i);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  return Socket::Connection("/tmp/mist/stream_"+streamname);
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// Create a stream on the system.
 | 
			
		||||
/// Filters the streamname, removing invalid characters and
 | 
			
		||||
/// converting all letters to lowercase.
 | 
			
		||||
/// If a '?' character is found, everything following that character is deleted.
 | 
			
		||||
/// If the /tmp/ddvtech directory doesn't exist yet, this will create it.
 | 
			
		||||
Socket::Server Socket::makeStream(std::string streamname){
 | 
			
		||||
  //strip anything that isn't numbers, digits or underscores
 | 
			
		||||
  for (std::string::iterator i=streamname.end()-1; i>=streamname.begin(); --i){
 | 
			
		||||
    if (*i == '?'){streamname.erase(i, streamname.end()); break;}
 | 
			
		||||
    if (!isalpha(*i) && !isdigit(*i) && *i != '_'){
 | 
			
		||||
      streamname.erase(i);
 | 
			
		||||
    }else{
 | 
			
		||||
      *i=tolower(*i);
 | 
			
		||||
    }
 | 
			
		||||
  }
 | 
			
		||||
  std::string loc = "/tmp/mist/stream_"+streamname;
 | 
			
		||||
  //attempt to create the /tmp/mist directory if it doesn't exist already.
 | 
			
		||||
  //ignore errors - we catch all problems in the Socket::Server creation already
 | 
			
		||||
  mkdir("/tmp/mist", S_IRWXU | S_IRWXG | S_IRWXO);
 | 
			
		||||
  //create and return the Socket::Server
 | 
			
		||||
  return Socket::Server(loc);
 | 
			
		||||
}
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
| 
						 | 
				
			
			@ -48,7 +48,7 @@ namespace Socket{
 | 
			
		|||
      bool swrite(std::string & buffer); ///< Write call that is compatible with std::string.
 | 
			
		||||
      bool iread(std::string & buffer); ///< Incremental write call that is compatible with std::string.
 | 
			
		||||
      bool iwrite(std::string & buffer); ///< Write call that is compatible with std::string.
 | 
			
		||||
      void spool(); ///< Updates the downbuffer and upbuffer internal variables.
 | 
			
		||||
      bool spool(); ///< Updates the downbuffer and upbuffer internal variables.
 | 
			
		||||
      std::string & Received(); ///< Returns a reference to the download buffer.
 | 
			
		||||
      void Send(std::string data); ///< Appends data to the upbuffer.
 | 
			
		||||
      void close(); ///< Close connection.
 | 
			
		||||
| 
						 | 
				
			
			@ -77,4 +77,10 @@ namespace Socket{
 | 
			
		|||
      int getSocket(); ///< Returns internal socket number.
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  /// Connect to a stream on the system.
 | 
			
		||||
  Connection getStream(std::string streamname);
 | 
			
		||||
 | 
			
		||||
  /// Create a stream on the system.
 | 
			
		||||
  Server makeStream(std::string streamname);
 | 
			
		||||
  
 | 
			
		||||
};
 | 
			
		||||
| 
						 | 
				
			
			
 | 
			
		|||
							
								
								
									
										287
									
								
								util/tinythread.cpp
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										287
									
								
								util/tinythread.cpp
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,287 @@
 | 
			
		|||
/*
 | 
			
		||||
Copyright (c) 2010 Marcus Geelnard
 | 
			
		||||
 | 
			
		||||
This software is provided 'as-is', without any express or implied
 | 
			
		||||
warranty. In no event will the authors be held liable for any damages
 | 
			
		||||
arising from the use of this software.
 | 
			
		||||
 | 
			
		||||
Permission is granted to anyone to use this software for any purpose,
 | 
			
		||||
including commercial applications, and to alter it and redistribute it
 | 
			
		||||
freely, subject to the following restrictions:
 | 
			
		||||
 | 
			
		||||
    1. The origin of this software must not be misrepresented; you must not
 | 
			
		||||
    claim that you wrote the original software. If you use this software
 | 
			
		||||
    in a product, an acknowledgment in the product documentation would be
 | 
			
		||||
    appreciated but is not required.
 | 
			
		||||
 | 
			
		||||
    2. Altered source versions must be plainly marked as such, and must not be
 | 
			
		||||
    misrepresented as being the original software.
 | 
			
		||||
 | 
			
		||||
    3. This notice may not be removed or altered from any source
 | 
			
		||||
    distribution.
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
#include <exception>
 | 
			
		||||
#include "tinythread.h"
 | 
			
		||||
 | 
			
		||||
#if defined(_TTHREAD_POSIX_)
 | 
			
		||||
  #include <unistd.h>
 | 
			
		||||
  #include <map>
 | 
			
		||||
#elif defined(_TTHREAD_WIN32_)
 | 
			
		||||
  #include <process.h>
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
namespace tthread {
 | 
			
		||||
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
// condition_variable
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
// NOTE 1: The Win32 implementation of the condition_variable class is based on
 | 
			
		||||
// the corresponding implementation in GLFW, which in turn is based on a
 | 
			
		||||
// description by Douglas C. Schmidt and Irfan Pyarali:
 | 
			
		||||
// http://www.cs.wustl.edu/~schmidt/win32-cv-1.html
 | 
			
		||||
//
 | 
			
		||||
// NOTE 2: Windows Vista actually has native support for condition variables
 | 
			
		||||
// (InitializeConditionVariable, WakeConditionVariable, etc), but we want to
 | 
			
		||||
// be portable with pre-Vista Windows versions, so TinyThread++ does not use
 | 
			
		||||
// Vista condition variables.
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
  #define _CONDITION_EVENT_ONE 0
 | 
			
		||||
  #define _CONDITION_EVENT_ALL 1
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
condition_variable::condition_variable() : mWaitersCount(0)
 | 
			
		||||
{
 | 
			
		||||
  mEvents[_CONDITION_EVENT_ONE] = CreateEvent(NULL, FALSE, FALSE, NULL);
 | 
			
		||||
  mEvents[_CONDITION_EVENT_ALL] = CreateEvent(NULL, TRUE, FALSE, NULL);
 | 
			
		||||
  InitializeCriticalSection(&mWaitersCountLock);
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
condition_variable::~condition_variable()
 | 
			
		||||
{
 | 
			
		||||
  CloseHandle(mEvents[_CONDITION_EVENT_ONE]);
 | 
			
		||||
  CloseHandle(mEvents[_CONDITION_EVENT_ALL]);
 | 
			
		||||
  DeleteCriticalSection(&mWaitersCountLock);
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
void condition_variable::_wait()
 | 
			
		||||
{
 | 
			
		||||
  // Wait for either event to become signaled due to notify_one() or
 | 
			
		||||
  // notify_all() being called
 | 
			
		||||
  int result = WaitForMultipleObjects(2, mEvents, FALSE, INFINITE);
 | 
			
		||||
 | 
			
		||||
  // Check if we are the last waiter
 | 
			
		||||
  EnterCriticalSection(&mWaitersCountLock);
 | 
			
		||||
  -- mWaitersCount;
 | 
			
		||||
  bool lastWaiter = (result == (WAIT_OBJECT_0 + _CONDITION_EVENT_ALL)) &&
 | 
			
		||||
                    (mWaitersCount == 0);
 | 
			
		||||
  LeaveCriticalSection(&mWaitersCountLock);
 | 
			
		||||
 | 
			
		||||
  // If we are the last waiter to be notified to stop waiting, reset the event
 | 
			
		||||
  if(lastWaiter)
 | 
			
		||||
    ResetEvent(mEvents[_CONDITION_EVENT_ALL]);
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
void condition_variable::notify_one()
 | 
			
		||||
{
 | 
			
		||||
  // Are there any waiters?
 | 
			
		||||
  EnterCriticalSection(&mWaitersCountLock);
 | 
			
		||||
  bool haveWaiters = (mWaitersCount > 0);
 | 
			
		||||
  LeaveCriticalSection(&mWaitersCountLock);
 | 
			
		||||
 | 
			
		||||
  // If we have any waiting threads, send them a signal
 | 
			
		||||
  if(haveWaiters)
 | 
			
		||||
    SetEvent(mEvents[_CONDITION_EVENT_ONE]);
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
void condition_variable::notify_all()
 | 
			
		||||
{
 | 
			
		||||
  // Are there any waiters?
 | 
			
		||||
  EnterCriticalSection(&mWaitersCountLock);
 | 
			
		||||
  bool haveWaiters = (mWaitersCount > 0);
 | 
			
		||||
  LeaveCriticalSection(&mWaitersCountLock);
 | 
			
		||||
 | 
			
		||||
  // If we have any waiting threads, send them a signal
 | 
			
		||||
  if(haveWaiters)
 | 
			
		||||
    SetEvent(mEvents[_CONDITION_EVENT_ALL]);
 | 
			
		||||
}
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
// POSIX pthread_t to unique thread::id mapping logic.
 | 
			
		||||
// Note: Here we use a global thread safe std::map to convert instances of
 | 
			
		||||
// pthread_t to small thread identifier numbers (unique within one process).
 | 
			
		||||
// This method should be portable across different POSIX implementations.
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
 | 
			
		||||
#if defined(_TTHREAD_POSIX_)
 | 
			
		||||
static thread::id _pthread_t_to_ID(const pthread_t &aHandle)
 | 
			
		||||
{
 | 
			
		||||
  static mutex idMapLock;
 | 
			
		||||
  static std::map<pthread_t, unsigned long int> idMap;
 | 
			
		||||
  static unsigned long int idCount(1);
 | 
			
		||||
 | 
			
		||||
  lock_guard<mutex> guard(idMapLock);
 | 
			
		||||
  if(idMap.find(aHandle) == idMap.end())
 | 
			
		||||
    idMap[aHandle] = idCount ++;
 | 
			
		||||
  return thread::id(idMap[aHandle]);
 | 
			
		||||
}
 | 
			
		||||
#endif // _TTHREAD_POSIX_
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
// thread
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
 | 
			
		||||
/// Information to pass to the new thread (what to run).
 | 
			
		||||
struct _thread_start_info {
 | 
			
		||||
  void (*mFunction)(void *); ///< Pointer to the function to be executed.
 | 
			
		||||
  void * mArg;               ///< Function argument for the thread function.
 | 
			
		||||
  thread * mThread;          ///< Pointer to the thread object.
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
// Thread wrapper function.
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
unsigned WINAPI thread::wrapper_function(void * aArg)
 | 
			
		||||
#elif defined(_TTHREAD_POSIX_)
 | 
			
		||||
void * thread::wrapper_function(void * aArg)
 | 
			
		||||
#endif
 | 
			
		||||
{
 | 
			
		||||
  // Get thread startup information
 | 
			
		||||
  _thread_start_info * ti = (_thread_start_info *) aArg;
 | 
			
		||||
 | 
			
		||||
  try
 | 
			
		||||
  {
 | 
			
		||||
    // Call the actual client thread function
 | 
			
		||||
    ti->mFunction(ti->mArg);
 | 
			
		||||
  }
 | 
			
		||||
  catch(...)
 | 
			
		||||
  {
 | 
			
		||||
    // Uncaught exceptions will terminate the application (default behavior
 | 
			
		||||
    // according to the C++0x draft)
 | 
			
		||||
    std::terminate();
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  // The thread is no longer executing
 | 
			
		||||
  lock_guard<mutex> guard(ti->mThread->mDataMutex);
 | 
			
		||||
  ti->mThread->mNotAThread = true;
 | 
			
		||||
 | 
			
		||||
  // The thread is responsible for freeing the startup information
 | 
			
		||||
  delete ti;
 | 
			
		||||
 | 
			
		||||
  return 0;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
thread::thread(void (*aFunction)(void *), void * aArg)
 | 
			
		||||
{
 | 
			
		||||
  // Serialize access to this thread structure
 | 
			
		||||
  lock_guard<mutex> guard(mDataMutex);
 | 
			
		||||
 | 
			
		||||
  // Fill out the thread startup information (passed to the thread wrapper,
 | 
			
		||||
  // which will eventually free it)
 | 
			
		||||
  _thread_start_info * ti = new _thread_start_info;
 | 
			
		||||
  ti->mFunction = aFunction;
 | 
			
		||||
  ti->mArg = aArg;
 | 
			
		||||
  ti->mThread = this;
 | 
			
		||||
 | 
			
		||||
  // The thread is now alive
 | 
			
		||||
  mNotAThread = false;
 | 
			
		||||
 | 
			
		||||
  // Create the thread
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
  mHandle = (HANDLE) _beginthreadex(0, 0, wrapper_function, (void *) ti, 0, &mWin32ThreadID);
 | 
			
		||||
#elif defined(_TTHREAD_POSIX_)
 | 
			
		||||
  if(pthread_create(&mHandle, NULL, wrapper_function, (void *) ti) != 0)
 | 
			
		||||
    mHandle = 0;
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
  // Did we fail to create the thread?
 | 
			
		||||
  if(!mHandle)
 | 
			
		||||
  {
 | 
			
		||||
    mNotAThread = true;
 | 
			
		||||
    delete ti;
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
thread::~thread()
 | 
			
		||||
{
 | 
			
		||||
  if(joinable())
 | 
			
		||||
    std::terminate();
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
void thread::join()
 | 
			
		||||
{
 | 
			
		||||
  if(joinable())
 | 
			
		||||
  {
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
    WaitForSingleObject(mHandle, INFINITE);
 | 
			
		||||
#elif defined(_TTHREAD_POSIX_)
 | 
			
		||||
    pthread_join(mHandle, NULL);
 | 
			
		||||
#endif
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
bool thread::joinable() const
 | 
			
		||||
{
 | 
			
		||||
  mDataMutex.lock();
 | 
			
		||||
  bool result = !mNotAThread;
 | 
			
		||||
  mDataMutex.unlock();
 | 
			
		||||
  return result;
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
thread::id thread::get_id() const
 | 
			
		||||
{
 | 
			
		||||
  if(!joinable())
 | 
			
		||||
    return id();
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
  return id((unsigned long int) mWin32ThreadID);
 | 
			
		||||
#elif defined(_TTHREAD_POSIX_)
 | 
			
		||||
  return _pthread_t_to_ID(mHandle);
 | 
			
		||||
#endif
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
unsigned thread::hardware_concurrency()
 | 
			
		||||
{
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
  SYSTEM_INFO si;
 | 
			
		||||
  GetSystemInfo(&si);
 | 
			
		||||
  return (int) si.dwNumberOfProcessors;
 | 
			
		||||
#elif defined(_SC_NPROCESSORS_ONLN)
 | 
			
		||||
  return (int) sysconf(_SC_NPROCESSORS_ONLN);
 | 
			
		||||
#elif defined(_SC_NPROC_ONLN)
 | 
			
		||||
  return (int) sysconf(_SC_NPROC_ONLN);
 | 
			
		||||
#else
 | 
			
		||||
  // The standard requires this function to return zero if the number of
 | 
			
		||||
  // hardware cores could not be determined.
 | 
			
		||||
  return 0;
 | 
			
		||||
#endif
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
// this_thread
 | 
			
		||||
//------------------------------------------------------------------------------
 | 
			
		||||
 | 
			
		||||
thread::id this_thread::get_id()
 | 
			
		||||
{
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
  return thread::id((unsigned long int) GetCurrentThreadId());
 | 
			
		||||
#elif defined(_TTHREAD_POSIX_)
 | 
			
		||||
  return _pthread_t_to_ID(pthread_self());
 | 
			
		||||
#endif
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
							
								
								
									
										696
									
								
								util/tinythread.h
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										696
									
								
								util/tinythread.h
									
										
									
									
									
										Normal file
									
								
							| 
						 | 
				
			
			@ -0,0 +1,696 @@
 | 
			
		|||
/*
 | 
			
		||||
Copyright (c) 2010 Marcus Geelnard
 | 
			
		||||
 | 
			
		||||
This software is provided 'as-is', without any express or implied
 | 
			
		||||
warranty. In no event will the authors be held liable for any damages
 | 
			
		||||
arising from the use of this software.
 | 
			
		||||
 | 
			
		||||
Permission is granted to anyone to use this software for any purpose,
 | 
			
		||||
including commercial applications, and to alter it and redistribute it
 | 
			
		||||
freely, subject to the following restrictions:
 | 
			
		||||
 | 
			
		||||
    1. The origin of this software must not be misrepresented; you must not
 | 
			
		||||
    claim that you wrote the original software. If you use this software
 | 
			
		||||
    in a product, an acknowledgment in the product documentation would be
 | 
			
		||||
    appreciated but is not required.
 | 
			
		||||
 | 
			
		||||
    2. Altered source versions must be plainly marked as such, and must not be
 | 
			
		||||
    misrepresented as being the original software.
 | 
			
		||||
 | 
			
		||||
    3. This notice may not be removed or altered from any source
 | 
			
		||||
    distribution.
 | 
			
		||||
*/
 | 
			
		||||
 | 
			
		||||
#ifndef _TINYTHREAD_H_
 | 
			
		||||
#define _TINYTHREAD_H_
 | 
			
		||||
 | 
			
		||||
/// @file
 | 
			
		||||
/// @mainpage TinyThread++ API Reference
 | 
			
		||||
///
 | 
			
		||||
/// @section intro_sec Introduction
 | 
			
		||||
/// TinyThread++ is a minimal, portable implementation of basic threading
 | 
			
		||||
/// classes for C++.
 | 
			
		||||
///
 | 
			
		||||
/// They closely mimic the functionality and naming of the C++0x standard, and
 | 
			
		||||
/// should be easily replaceable with the corresponding std:: variants.
 | 
			
		||||
///
 | 
			
		||||
/// @section port_sec Portability
 | 
			
		||||
/// The Win32 variant uses the native Win32 API for implementing the thread
 | 
			
		||||
/// classes, while for other systems, the POSIX threads API (pthread) is used.
 | 
			
		||||
///
 | 
			
		||||
/// @section class_sec Classes
 | 
			
		||||
/// In order to mimic the threading API of the C++0x standard, subsets of
 | 
			
		||||
/// several classes are provided. The fundamental classes are:
 | 
			
		||||
/// @li tthread::thread
 | 
			
		||||
/// @li tthread::mutex
 | 
			
		||||
/// @li tthread::recursive_mutex
 | 
			
		||||
/// @li tthread::condition_variable
 | 
			
		||||
/// @li tthread::lock_guard
 | 
			
		||||
/// @li tthread::fast_mutex
 | 
			
		||||
///
 | 
			
		||||
/// @section misc_sec Miscellaneous
 | 
			
		||||
/// The following special keywords are available: #thread_local.
 | 
			
		||||
///
 | 
			
		||||
/// For more detailed information (including additional classes), browse the
 | 
			
		||||
/// different sections of this documentation. A good place to start is:
 | 
			
		||||
/// tinythread.h.
 | 
			
		||||
 | 
			
		||||
// Which platform are we on?
 | 
			
		||||
#if !defined(_TTHREAD_PLATFORM_DEFINED_)
 | 
			
		||||
  #if defined(_WIN32) || defined(__WIN32__) || defined(__WINDOWS__)
 | 
			
		||||
    #define _TTHREAD_WIN32_
 | 
			
		||||
  #else
 | 
			
		||||
    #define _TTHREAD_POSIX_
 | 
			
		||||
  #endif
 | 
			
		||||
  #define _TTHREAD_PLATFORM_DEFINED_
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
// Platform specific includes
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
  #include <windows.h>
 | 
			
		||||
#else
 | 
			
		||||
  #include <pthread.h>
 | 
			
		||||
  #include <signal.h>
 | 
			
		||||
  #include <sched.h>
 | 
			
		||||
  #include <unistd.h>
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
// Generic includes
 | 
			
		||||
#include <ostream>
 | 
			
		||||
 | 
			
		||||
/// TinyThread++ version (major number).
 | 
			
		||||
#define TINYTHREAD_VERSION_MAJOR 1
 | 
			
		||||
/// TinyThread++ version (minor number).
 | 
			
		||||
#define TINYTHREAD_VERSION_MINOR 0
 | 
			
		||||
/// TinyThread++ version (full version).
 | 
			
		||||
#define TINYTHREAD_VERSION (TINYTHREAD_VERSION_MAJOR * 100 + TINYTHREAD_VERSION_MINOR)
 | 
			
		||||
 | 
			
		||||
// Do we have a fully featured C++0x compiler?
 | 
			
		||||
#if (__cplusplus > 199711L) || (defined(__STDCXX_VERSION__) && (__STDCXX_VERSION__ >= 201001L))
 | 
			
		||||
  #define _TTHREAD_CPP0X_
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
// ...at least partial C++0x?
 | 
			
		||||
#if defined(_TTHREAD_CPP0X_) || defined(__GXX_EXPERIMENTAL_CXX0X__) || defined(__GXX_EXPERIMENTAL_CPP0X__)
 | 
			
		||||
  #define _TTHREAD_CPP0X_PARTIAL_
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
// Macro for disabling assignments of objects.
 | 
			
		||||
#ifdef _TTHREAD_CPP0X_PARTIAL_
 | 
			
		||||
  #define _TTHREAD_DISABLE_ASSIGNMENT(name) \
 | 
			
		||||
      name(const name&) = delete; \
 | 
			
		||||
      name& operator=(const name&) = delete;
 | 
			
		||||
#else
 | 
			
		||||
  #define _TTHREAD_DISABLE_ASSIGNMENT(name) \
 | 
			
		||||
      name(const name&); \
 | 
			
		||||
      name& operator=(const name&);
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
/// @def thread_local
 | 
			
		||||
/// Thread local storage keyword.
 | 
			
		||||
/// A variable that is declared with the \c thread_local keyword makes the
 | 
			
		||||
/// value of the variable local to each thread (known as thread-local storage,
 | 
			
		||||
/// or TLS). Example usage:
 | 
			
		||||
/// @code
 | 
			
		||||
/// // This variable is local to each thread.
 | 
			
		||||
/// thread_local int variable;
 | 
			
		||||
/// @endcode
 | 
			
		||||
/// @note The \c thread_local keyword is a macro that maps to the corresponding
 | 
			
		||||
/// compiler directive (e.g. \c __declspec(thread)). While the C++0x standard
 | 
			
		||||
/// allows for non-trivial types (e.g. classes with constructors and
 | 
			
		||||
/// destructors) to be declared with the \c thread_local keyword, most pre-C++0x
 | 
			
		||||
/// compilers only allow for trivial types (e.g. \c int). So, to guarantee
 | 
			
		||||
/// portable code, only use trivial types for thread local storage.
 | 
			
		||||
/// @note This directive is currently not supported on Mac OS X (it will give
 | 
			
		||||
/// a compiler error), since compile-time TLS is not supported in the Mac OS X
 | 
			
		||||
/// executable format. Also, some older versions of MinGW (before GCC 4.x) do
 | 
			
		||||
/// not support this directive.
 | 
			
		||||
/// @hideinitializer
 | 
			
		||||
 | 
			
		||||
#if !defined(_TTHREAD_CPP0X_) && !defined(thread_local)
 | 
			
		||||
 #if defined(__GNUC__) || defined(__INTEL_COMPILER) || defined(__SUNPRO_CC) || defined(__IBMCPP__)
 | 
			
		||||
  #define thread_local __thread
 | 
			
		||||
 #else
 | 
			
		||||
  #define thread_local __declspec(thread)
 | 
			
		||||
 #endif
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/// Main name space for TinyThread++.
 | 
			
		||||
/// This namespace is more or less equivalent to the \c std namespace for the
 | 
			
		||||
/// C++0x thread classes. For instance, the tthread::mutex class corresponds to
 | 
			
		||||
/// the std::mutex class.
 | 
			
		||||
namespace tthread {
 | 
			
		||||
 | 
			
		||||
/// Mutex class.
 | 
			
		||||
/// This is a mutual exclusion object for synchronizing access to shared
 | 
			
		||||
/// memory areas for several threads. The mutex is non-recursive (i.e. a
 | 
			
		||||
/// program may deadlock if the thread that owns a mutex object calls lock()
 | 
			
		||||
/// on that object).
 | 
			
		||||
/// @see recursive_mutex
 | 
			
		||||
class mutex {
 | 
			
		||||
  public:
 | 
			
		||||
    /// Constructor.
 | 
			
		||||
    mutex()
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
      : mAlreadyLocked(false)
 | 
			
		||||
#endif
 | 
			
		||||
    {
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
      InitializeCriticalSection(&mHandle);
 | 
			
		||||
#else
 | 
			
		||||
      pthread_mutex_init(&mHandle, NULL);
 | 
			
		||||
#endif
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// Destructor.
 | 
			
		||||
    ~mutex()
 | 
			
		||||
    {
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
      DeleteCriticalSection(&mHandle);
 | 
			
		||||
#else
 | 
			
		||||
      pthread_mutex_destroy(&mHandle);
 | 
			
		||||
#endif
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// Lock the mutex.
 | 
			
		||||
    /// The method will block the calling thread until a lock on the mutex can
 | 
			
		||||
    /// be obtained. The mutex remains locked until \c unlock() is called.
 | 
			
		||||
    /// @see lock_guard
 | 
			
		||||
    inline void lock()
 | 
			
		||||
    {
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
      EnterCriticalSection(&mHandle);
 | 
			
		||||
      while(mAlreadyLocked) Sleep(1000); // Simulate deadlock...
 | 
			
		||||
      mAlreadyLocked = true;
 | 
			
		||||
#else
 | 
			
		||||
      pthread_mutex_lock(&mHandle);
 | 
			
		||||
#endif
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// Try to lock the mutex.
 | 
			
		||||
    /// The method will try to lock the mutex. If it fails, the function will
 | 
			
		||||
    /// return immediately (non-blocking).
 | 
			
		||||
    /// @return \c true if the lock was acquired, or \c false if the lock could
 | 
			
		||||
    /// not be acquired.
 | 
			
		||||
    inline bool try_lock()
 | 
			
		||||
    {
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
      bool ret = (TryEnterCriticalSection(&mHandle) ? true : false);
 | 
			
		||||
      if(ret && mAlreadyLocked)
 | 
			
		||||
      {
 | 
			
		||||
        LeaveCriticalSection(&mHandle);
 | 
			
		||||
        ret = false;
 | 
			
		||||
      }
 | 
			
		||||
      return ret;
 | 
			
		||||
#else
 | 
			
		||||
      return (pthread_mutex_trylock(&mHandle) == 0) ? true : false;
 | 
			
		||||
#endif
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// Unlock the mutex.
 | 
			
		||||
    /// If any threads are waiting for the lock on this mutex, one of them will
 | 
			
		||||
    /// be unblocked.
 | 
			
		||||
    inline void unlock()
 | 
			
		||||
    {
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
      mAlreadyLocked = false;
 | 
			
		||||
      LeaveCriticalSection(&mHandle);
 | 
			
		||||
#else
 | 
			
		||||
      pthread_mutex_unlock(&mHandle);
 | 
			
		||||
#endif
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    _TTHREAD_DISABLE_ASSIGNMENT(mutex)
 | 
			
		||||
 | 
			
		||||
  private:
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
    CRITICAL_SECTION mHandle;
 | 
			
		||||
    bool mAlreadyLocked;
 | 
			
		||||
#else
 | 
			
		||||
    pthread_mutex_t mHandle;
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
    friend class condition_variable;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/// Recursive mutex class.
 | 
			
		||||
/// This is a mutual exclusion object for synchronizing access to shared
 | 
			
		||||
/// memory areas for several threads. The mutex is recursive (i.e. a thread
 | 
			
		||||
/// may lock the mutex several times, as long as it unlocks the mutex the same
 | 
			
		||||
/// number of times).
 | 
			
		||||
/// @see mutex
 | 
			
		||||
class recursive_mutex {
 | 
			
		||||
  public:
 | 
			
		||||
    /// Constructor.
 | 
			
		||||
    recursive_mutex()
 | 
			
		||||
    {
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
      InitializeCriticalSection(&mHandle);
 | 
			
		||||
#else
 | 
			
		||||
      pthread_mutexattr_t attr;
 | 
			
		||||
      pthread_mutexattr_init(&attr);
 | 
			
		||||
      pthread_mutexattr_settype(&attr, PTHREAD_MUTEX_RECURSIVE);
 | 
			
		||||
      pthread_mutex_init(&mHandle, &attr);
 | 
			
		||||
#endif
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// Destructor.
 | 
			
		||||
    ~recursive_mutex()
 | 
			
		||||
    {
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
      DeleteCriticalSection(&mHandle);
 | 
			
		||||
#else
 | 
			
		||||
      pthread_mutex_destroy(&mHandle);
 | 
			
		||||
#endif
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// Lock the mutex.
 | 
			
		||||
    /// The method will block the calling thread until a lock on the mutex can
 | 
			
		||||
    /// be obtained. The mutex remains locked until \c unlock() is called.
 | 
			
		||||
    /// @see lock_guard
 | 
			
		||||
    inline void lock()
 | 
			
		||||
    {
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
      EnterCriticalSection(&mHandle);
 | 
			
		||||
#else
 | 
			
		||||
      pthread_mutex_lock(&mHandle);
 | 
			
		||||
#endif
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// Try to lock the mutex.
 | 
			
		||||
    /// The method will try to lock the mutex. If it fails, the function will
 | 
			
		||||
    /// return immediately (non-blocking).
 | 
			
		||||
    /// @return \c true if the lock was acquired, or \c false if the lock could
 | 
			
		||||
    /// not be acquired.
 | 
			
		||||
    inline bool try_lock()
 | 
			
		||||
    {
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
      return TryEnterCriticalSection(&mHandle) ? true : false;
 | 
			
		||||
#else
 | 
			
		||||
      return (pthread_mutex_trylock(&mHandle) == 0) ? true : false;
 | 
			
		||||
#endif
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// Unlock the mutex.
 | 
			
		||||
    /// If any threads are waiting for the lock on this mutex, one of them will
 | 
			
		||||
    /// be unblocked.
 | 
			
		||||
    inline void unlock()
 | 
			
		||||
    {
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
      LeaveCriticalSection(&mHandle);
 | 
			
		||||
#else
 | 
			
		||||
      pthread_mutex_unlock(&mHandle);
 | 
			
		||||
#endif
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    _TTHREAD_DISABLE_ASSIGNMENT(recursive_mutex)
 | 
			
		||||
 | 
			
		||||
  private:
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
    CRITICAL_SECTION mHandle;
 | 
			
		||||
#else
 | 
			
		||||
    pthread_mutex_t mHandle;
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
    friend class condition_variable;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/// Lock guard class.
 | 
			
		||||
/// The constructor locks the mutex, and the destructor unlocks the mutex, so
 | 
			
		||||
/// the mutex will automatically be unlocked when the lock guard goes out of
 | 
			
		||||
/// scope. Example usage:
 | 
			
		||||
/// @code
 | 
			
		||||
/// mutex m;
 | 
			
		||||
/// int counter;
 | 
			
		||||
///
 | 
			
		||||
/// void increment()
 | 
			
		||||
/// {
 | 
			
		||||
///   lock_guard<mutex> guard(m);
 | 
			
		||||
///   ++ counter;
 | 
			
		||||
/// }
 | 
			
		||||
/// @endcode
 | 
			
		||||
 | 
			
		||||
template <class T>
 | 
			
		||||
class lock_guard {
 | 
			
		||||
  public:
 | 
			
		||||
    typedef T mutex_type;
 | 
			
		||||
 | 
			
		||||
    lock_guard() : mMutex(0) {}
 | 
			
		||||
 | 
			
		||||
    /// The constructor locks the mutex.
 | 
			
		||||
    explicit lock_guard(mutex_type &aMutex)
 | 
			
		||||
    {
 | 
			
		||||
      mMutex = &aMutex;
 | 
			
		||||
      mMutex->lock();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// The destructor unlocks the mutex.
 | 
			
		||||
    ~lock_guard()
 | 
			
		||||
    {
 | 
			
		||||
      if(mMutex)
 | 
			
		||||
        mMutex->unlock();
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
  private:
 | 
			
		||||
    mutex_type * mMutex;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/// Condition variable class.
 | 
			
		||||
/// This is a signalling object for synchronizing the execution flow for
 | 
			
		||||
/// several threads. Example usage:
 | 
			
		||||
/// @code
 | 
			
		||||
/// // Shared data and associated mutex and condition variable objects
 | 
			
		||||
/// int count;
 | 
			
		||||
/// mutex m;
 | 
			
		||||
/// condition_variable cond;
 | 
			
		||||
///
 | 
			
		||||
/// // Wait for the counter to reach a certain number
 | 
			
		||||
/// void wait_counter(int targetCount)
 | 
			
		||||
/// {
 | 
			
		||||
///   lock_guard<mutex> guard(m);
 | 
			
		||||
///   while(count < targetCount)
 | 
			
		||||
///     cond.wait(m);
 | 
			
		||||
/// }
 | 
			
		||||
///
 | 
			
		||||
/// // Increment the counter, and notify waiting threads
 | 
			
		||||
/// void increment()
 | 
			
		||||
/// {
 | 
			
		||||
///   lock_guard<mutex> guard(m);
 | 
			
		||||
///   ++ count;
 | 
			
		||||
///   cond.notify_all();
 | 
			
		||||
/// }
 | 
			
		||||
/// @endcode
 | 
			
		||||
class condition_variable {
 | 
			
		||||
  public:
 | 
			
		||||
    /// Constructor.
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
    condition_variable();
 | 
			
		||||
#else
 | 
			
		||||
    condition_variable()
 | 
			
		||||
    {
 | 
			
		||||
      pthread_cond_init(&mHandle, NULL);
 | 
			
		||||
    }
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
    /// Destructor.
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
    ~condition_variable();
 | 
			
		||||
#else
 | 
			
		||||
    ~condition_variable()
 | 
			
		||||
    {
 | 
			
		||||
      pthread_cond_destroy(&mHandle);
 | 
			
		||||
    }
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
    /// Wait for the condition.
 | 
			
		||||
    /// The function will block the calling thread until the condition variable
 | 
			
		||||
    /// is woken by \c notify_one(), \c notify_all() or a spurious wake up.
 | 
			
		||||
    /// @param[in] aMutex A mutex that will be unlocked when the wait operation
 | 
			
		||||
    ///   starts, an locked again as soon as the wait operation is finished.
 | 
			
		||||
    template <class _mutexT>
 | 
			
		||||
    inline void wait(_mutexT &aMutex)
 | 
			
		||||
    {
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
      // Increment number of waiters
 | 
			
		||||
      EnterCriticalSection(&mWaitersCountLock);
 | 
			
		||||
      ++ mWaitersCount;
 | 
			
		||||
      LeaveCriticalSection(&mWaitersCountLock);
 | 
			
		||||
 | 
			
		||||
      // Release the mutex while waiting for the condition (will decrease
 | 
			
		||||
      // the number of waiters when done)...
 | 
			
		||||
      aMutex.unlock();
 | 
			
		||||
      _wait();
 | 
			
		||||
      aMutex.lock();
 | 
			
		||||
#else
 | 
			
		||||
      pthread_cond_wait(&mHandle, &aMutex.mHandle);
 | 
			
		||||
#endif
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// Notify one thread that is waiting for the condition.
 | 
			
		||||
    /// If at least one thread is blocked waiting for this condition variable,
 | 
			
		||||
    /// one will be woken up.
 | 
			
		||||
    /// @note Only threads that started waiting prior to this call will be
 | 
			
		||||
    /// woken up.
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
    void notify_one();
 | 
			
		||||
#else
 | 
			
		||||
    inline void notify_one()
 | 
			
		||||
    {
 | 
			
		||||
      pthread_cond_signal(&mHandle);
 | 
			
		||||
    }
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
    /// Notify all threads that are waiting for the condition.
 | 
			
		||||
    /// All threads that are blocked waiting for this condition variable will
 | 
			
		||||
    /// be woken up.
 | 
			
		||||
    /// @note Only threads that started waiting prior to this call will be
 | 
			
		||||
    /// woken up.
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
    void notify_all();
 | 
			
		||||
#else
 | 
			
		||||
    inline void notify_all()
 | 
			
		||||
    {
 | 
			
		||||
      pthread_cond_broadcast(&mHandle);
 | 
			
		||||
    }
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
    _TTHREAD_DISABLE_ASSIGNMENT(condition_variable)
 | 
			
		||||
 | 
			
		||||
  private:
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
    void _wait();
 | 
			
		||||
    HANDLE mEvents[2];                  ///< Signal and broadcast event HANDLEs.
 | 
			
		||||
    unsigned int mWaitersCount;         ///< Count of the number of waiters.
 | 
			
		||||
    CRITICAL_SECTION mWaitersCountLock; ///< Serialize access to mWaitersCount.
 | 
			
		||||
#else
 | 
			
		||||
    pthread_cond_t mHandle;
 | 
			
		||||
#endif
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
/// Thread class.
 | 
			
		||||
class thread {
 | 
			
		||||
  public:
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
    typedef HANDLE native_handle_type;
 | 
			
		||||
#else
 | 
			
		||||
    typedef pthread_t native_handle_type;
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
    class id;
 | 
			
		||||
 | 
			
		||||
    /// Default constructor.
 | 
			
		||||
    /// Construct a \c thread object without an associated thread of execution
 | 
			
		||||
    /// (i.e. non-joinable).
 | 
			
		||||
    thread() : mHandle(0), mNotAThread(true)
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
    , mWin32ThreadID(0)
 | 
			
		||||
#endif
 | 
			
		||||
    {}
 | 
			
		||||
 | 
			
		||||
    /// Thread starting constructor.
 | 
			
		||||
    /// Construct a \c thread object with a new thread of execution.
 | 
			
		||||
    /// @param[in] aFunction A function pointer to a function of type:
 | 
			
		||||
    ///          <tt>void fun(void * arg)</tt>
 | 
			
		||||
    /// @param[in] aArg Argument to the thread function.
 | 
			
		||||
    /// @note This constructor is not fully compatible with the standard C++
 | 
			
		||||
    /// thread class. It is more similar to the pthread_create() (POSIX) and
 | 
			
		||||
    /// CreateThread() (Windows) functions.
 | 
			
		||||
    thread(void (*aFunction)(void *), void * aArg);
 | 
			
		||||
 | 
			
		||||
    /// Destructor.
 | 
			
		||||
    /// @note If the thread is joinable upon destruction, \c std::terminate()
 | 
			
		||||
    /// will be called, which terminates the process. It is always wise to do
 | 
			
		||||
    /// \c join() before deleting a thread object.
 | 
			
		||||
    ~thread();
 | 
			
		||||
 | 
			
		||||
    /// Wait for the thread to finish (join execution flows).
 | 
			
		||||
    void join();
 | 
			
		||||
 | 
			
		||||
    /// Check if the thread is joinable.
 | 
			
		||||
    /// A thread object is joinable if it has an associated thread of execution.
 | 
			
		||||
    bool joinable() const;
 | 
			
		||||
 | 
			
		||||
    /// Return the thread ID of a thread object.
 | 
			
		||||
    id get_id() const;
 | 
			
		||||
 | 
			
		||||
    /// Get the native handle for this thread.
 | 
			
		||||
    /// @note Under Windows, this is a \c HANDLE, and under POSIX systems, this
 | 
			
		||||
    /// is a \c pthread_t.
 | 
			
		||||
    inline native_handle_type native_handle()
 | 
			
		||||
    {
 | 
			
		||||
      return mHandle;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    /// Determine the number of threads which can possibly execute concurrently.
 | 
			
		||||
    /// This function is useful for determining the optimal number of threads to
 | 
			
		||||
    /// use for a task.
 | 
			
		||||
    /// @return The number of hardware thread contexts in the system.
 | 
			
		||||
    /// @note If this value is not defined, the function returns zero (0).
 | 
			
		||||
    static unsigned hardware_concurrency();
 | 
			
		||||
 | 
			
		||||
    _TTHREAD_DISABLE_ASSIGNMENT(thread)
 | 
			
		||||
 | 
			
		||||
  private:
 | 
			
		||||
    native_handle_type mHandle;   ///< Thread handle.
 | 
			
		||||
    mutable mutex mDataMutex;     ///< Serializer for access to the thread private data.
 | 
			
		||||
    bool mNotAThread;             ///< True if this object is not a thread of execution.
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
    unsigned int mWin32ThreadID;  ///< Unique thread ID (filled out by _beginthreadex).
 | 
			
		||||
#endif
 | 
			
		||||
 | 
			
		||||
    // This is the internal thread wrapper function.
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
    static unsigned WINAPI wrapper_function(void * aArg);
 | 
			
		||||
#else
 | 
			
		||||
    static void * wrapper_function(void * aArg);
 | 
			
		||||
#endif
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/// Thread ID.
 | 
			
		||||
/// The thread ID is a unique identifier for each thread.
 | 
			
		||||
/// @see thread::get_id()
 | 
			
		||||
class thread::id {
 | 
			
		||||
  public:
 | 
			
		||||
    /// Default constructor.
 | 
			
		||||
    /// The default constructed ID is that of thread without a thread of
 | 
			
		||||
    /// execution.
 | 
			
		||||
    id() : mId(0) {};
 | 
			
		||||
 | 
			
		||||
    id(unsigned long int aId) : mId(aId) {};
 | 
			
		||||
 | 
			
		||||
    id(const id& aId) : mId(aId.mId) {};
 | 
			
		||||
 | 
			
		||||
    inline id & operator=(const id &aId)
 | 
			
		||||
    {
 | 
			
		||||
      mId = aId.mId;
 | 
			
		||||
      return *this;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    inline friend bool operator==(const id &aId1, const id &aId2)
 | 
			
		||||
    {
 | 
			
		||||
      return (aId1.mId == aId2.mId);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    inline friend bool operator!=(const id &aId1, const id &aId2)
 | 
			
		||||
    {
 | 
			
		||||
      return (aId1.mId != aId2.mId);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    inline friend bool operator<=(const id &aId1, const id &aId2)
 | 
			
		||||
    {
 | 
			
		||||
      return (aId1.mId <= aId2.mId);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    inline friend bool operator<(const id &aId1, const id &aId2)
 | 
			
		||||
    {
 | 
			
		||||
      return (aId1.mId < aId2.mId);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    inline friend bool operator>=(const id &aId1, const id &aId2)
 | 
			
		||||
    {
 | 
			
		||||
      return (aId1.mId >= aId2.mId);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    inline friend bool operator>(const id &aId1, const id &aId2)
 | 
			
		||||
    {
 | 
			
		||||
      return (aId1.mId > aId2.mId);
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    inline friend std::ostream& operator <<(std::ostream &os, const id &obj)
 | 
			
		||||
    {
 | 
			
		||||
      os << obj.mId;
 | 
			
		||||
      return os;
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
  private:
 | 
			
		||||
    unsigned long int mId;
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
 | 
			
		||||
// Related to <ratio> - minimal to be able to support chrono.
 | 
			
		||||
typedef long long __intmax_t;
 | 
			
		||||
 | 
			
		||||
/// Minimal implementation of the \c ratio class. This class provides enough
 | 
			
		||||
/// functionality to implement some basic \c chrono classes.
 | 
			
		||||
template <__intmax_t N, __intmax_t D = 1> class ratio {
 | 
			
		||||
  public:
 | 
			
		||||
    static double _as_double() { return double(N) / double(D); }
 | 
			
		||||
};
 | 
			
		||||
 | 
			
		||||
/// Minimal implementation of the \c chrono namespace.
 | 
			
		||||
/// The \c chrono namespace provides types for specifying time intervals.
 | 
			
		||||
namespace chrono {
 | 
			
		||||
  /// Duration template class. This class provides enough functionality to
 | 
			
		||||
  /// implement \c this_thread::sleep_for().
 | 
			
		||||
  template <class _Rep, class _Period = ratio<1> > class duration {
 | 
			
		||||
    private:
 | 
			
		||||
      _Rep rep_;
 | 
			
		||||
    public:
 | 
			
		||||
      typedef _Rep rep;
 | 
			
		||||
      typedef _Period period;
 | 
			
		||||
 | 
			
		||||
      /// Construct a duration object with the given duration.
 | 
			
		||||
      template <class _Rep2>
 | 
			
		||||
        explicit duration(const _Rep2& r) : rep_(r) {};
 | 
			
		||||
 | 
			
		||||
      /// Return the value of the duration object.
 | 
			
		||||
      rep count() const
 | 
			
		||||
      {
 | 
			
		||||
        return rep_;
 | 
			
		||||
      }
 | 
			
		||||
  };
 | 
			
		||||
 | 
			
		||||
  // Standard duration types.
 | 
			
		||||
  typedef duration<__intmax_t, ratio<1, 1000000000> > nanoseconds; ///< Duration with the unit nanoseconds.
 | 
			
		||||
  typedef duration<__intmax_t, ratio<1, 1000000> > microseconds;   ///< Duration with the unit microseconds.
 | 
			
		||||
  typedef duration<__intmax_t, ratio<1, 1000> > milliseconds;      ///< Duration with the unit milliseconds.
 | 
			
		||||
  typedef duration<__intmax_t> seconds;                            ///< Duration with the unit seconds.
 | 
			
		||||
  typedef duration<__intmax_t, ratio<60> > minutes;                ///< Duration with the unit minutes.
 | 
			
		||||
  typedef duration<__intmax_t, ratio<3600> > hours;                ///< Duration with the unit hours.
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
/// The namespace \c this_thread provides methods for dealing with the
 | 
			
		||||
/// calling thread.
 | 
			
		||||
namespace this_thread {
 | 
			
		||||
  /// Return the thread ID of the calling thread.
 | 
			
		||||
  thread::id get_id();
 | 
			
		||||
 | 
			
		||||
  /// Yield execution to another thread.
 | 
			
		||||
  /// Offers the operating system the opportunity to schedule another thread
 | 
			
		||||
  /// that is ready to run on the current processor.
 | 
			
		||||
  inline void yield()
 | 
			
		||||
  {
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
    Sleep(0);
 | 
			
		||||
#else
 | 
			
		||||
    sched_yield();
 | 
			
		||||
#endif
 | 
			
		||||
  }
 | 
			
		||||
 | 
			
		||||
  /// Blocks the calling thread for a period of time.
 | 
			
		||||
  /// @param[in] aTime Minimum time to put the thread to sleep.
 | 
			
		||||
  /// Example usage:
 | 
			
		||||
  /// @code
 | 
			
		||||
  /// // Sleep for 100 milliseconds
 | 
			
		||||
  /// this_thread::sleep_for(chrono::milliseconds(100));
 | 
			
		||||
  /// @endcode
 | 
			
		||||
  /// @note Supported duration types are: nanoseconds, microseconds,
 | 
			
		||||
  /// milliseconds, seconds, minutes and hours.
 | 
			
		||||
  template <class _Rep, class _Period> void sleep_for(const chrono::duration<_Rep, _Period>& aTime)
 | 
			
		||||
  {
 | 
			
		||||
#if defined(_TTHREAD_WIN32_)
 | 
			
		||||
    Sleep(int(double(aTime.count()) * (1000.0 * _Period::_as_double()) + 0.5));
 | 
			
		||||
#else
 | 
			
		||||
    usleep(int(double(aTime.count()) * (1000000.0 * _Period::_as_double()) + 0.5));
 | 
			
		||||
#endif
 | 
			
		||||
  }
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
// Define/macro cleanup
 | 
			
		||||
#undef _TTHREAD_DISABLE_ASSIGNMENT
 | 
			
		||||
 | 
			
		||||
#endif // _TINYTHREAD_H_
 | 
			
		||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue