2012-08-15 17 views
10

Estoy usando la biblioteca de códigos de error C++ 11 system_error para crear una clase de error personalizada para una biblioteca que estoy creando. He hecho esto antes con boost::error_code, pero no puedo hacerlo funcionar con std::error_code. Estoy usando GCC 4.6.Comparando un código de error enum con std :: error_code

Básicamente, me he presentado todo el código repetitivo para crear una clase de error, un error_category, y las rutinas de conversión del espacio de nombres de ETS para convertir mis enumeraciones personalizados en un std::error_code objeto:

namespace mylib 
{ 
    namespace errc { 

     enum my_error 
     { 
      failed = 0 
     }; 

     inline const char* error_message(int c) 
     { 
      static const char* err_msg[] = 
      { 
       "Failed", 
      }; 

      assert(c < sizeof(err_msg)/sizeof(err_msg[0])); 
      return err_msg[c]; 
     } 

     class my_error_category : public std::error_category 
     { 
      public: 

      my_error_category() 
      { } 

      std::string message(int c) const 
      { 
       return error_message(c); 
      } 

      const char* name() const { return "My Error Category"; } 

      const static error_category& get() 
      { 
       const static my_error_category category_const; 
       return category_const; 
      } 
     }; 

    } // end namespace errc 
} // end namespace mylib 

namespace std { 

inline error_code make_error_code(mylib::errc::my_error e) 
{ 
    return error_code(static_cast<int>(e), mylib::errc::my_error_category::get()); 
} 

template<> 
struct is_error_code_enum<mylib::errc::my_error> 
    : std::true_type 
{ }; 


El problema es, no parece conversión implícita entre mis errores de código y enumeraciones std::error_code objetos que se van a trabajar, por lo que no puede, por ejemplo, tratar de comparar una instancia de std::error_code con literales de enumeración:

int main() 
{ 
    std::error_code ec1 = std::make_error_code(mylib::errc::failed); // works 
    std::error_code ec2 = mylib::errc::failed; // doesn't compile 
    bool result = (ec2 == mylib::errc::failed); // doesn't compile 
} 

La expresión ec2 == mylib::errc::failed no compilará - Tengo que decir ec2 == std::make_error_code(mylib::errc::failed).

El error del compilador emite es:

In file included from test6.cc:3:0: 
/usr/include/c++/4.6/system_error: In constructor ‘std::error_code::error_code(_ErrorCodeEnum, typename std::enable_if<std::is_error_code_enum<_ErrorCodeEnum>::value>::type*) [with _ErrorCodeEnum = mylib::errc::my_error, typename std::enable_if<std::is_error_code_enum<_ErrorCodeEnum>::value>::type = void]’: 
test6.cc:70:37: instantiated from here 
/usr/include/c++/4.6/system_error:127:9: error: cannot convert ‘mylib::errc::my_error’ to ‘std::errc’ for argument ‘1’ to ‘std::error_code std::make_error_code(std::errc)’ 

Y he aquí un Ideone link.

Entonces, ¿por qué no funciona? ¿Necesito un código repetitivo para habilitar mylib::errc::my_error enumeraciones para ser implícitamente convertible a std::error_code? Pensé que la especialización de std::make_error_code se ocupa de eso?

Respuesta

Cuestiones relacionadas