Having the construction accept an Error is probably fine, because you have to std::move it anyway, which means you know you're giving up ownership<br><div class="gmail_quote"><div dir="ltr">On Wed, May 17, 2017 at 3:08 AM Pavel Labath via Phabricator <<a href="mailto:reviews@reviews.llvm.org">reviews@reviews.llvm.org</a>> wrote:<br></div><blockquote class="gmail_quote" style="margin:0 0 0 .8ex;border-left:1px #ccc solid;padding-left:1ex">labath added a comment.<br>
<br>
In <a href="https://reviews.llvm.org/D33241#756921" rel="noreferrer" target="_blank">https://reviews.llvm.org/D33241#756921</a>, @zturner wrote:<br>
<br>
> Mostly just that implicit conversion operators are a very subtle source of bugs, and you can't even find where they're being used because it's impossible to grep for it.<br>
<br>
<br>
I agree, and that's why I made the conversions explicit. For example `Error foo() { return Status(); }` (or vice-versa) will fail to compile without an explicit cast. So you should always see that a conversion is happening as the other type in the general vicinity of the conversion. However you will not be able to easily grep for the occurences of the conversion, as you would be in the `toError` case.<br>
<br>
I did also have my doubts about whether this would be explicit enough though, so I would not be opposed to going to the separate function you propose. Do you want to beef up the explicitness on both directions of the conversion (e.g. `Status s = Status::ConsumeError(std::move(error))`) or just the one?<br>
<br>
<br>
<br>
================<br>
Comment at: source/Utility/Status.cpp:81-88<br>
+Status::operator llvm::Error() {<br>
+  if (Success())<br>
+    return llvm::Error::success();<br>
+  if (m_type == ErrorType::eErrorTypePOSIX)<br>
+    return llvm::errorCodeToError(std::error_code(m_code, std::generic_category()));<br>
+  return llvm::make_error<llvm::StringError>(AsCString(),<br>
+                                             llvm::inconvertibleErrorCode());<br>
----------------<br>
zturner wrote:<br>
> zturner wrote:<br>
> > Delete in favor of an `LLDBError` class as mentioned before.<br>
> Actually this doesn't really work, because you don't want to call `make_error<LLDBError>(S)` if `S` is a success condition.<br>
><br>
> So perhaps instead, I would at least call it something more explicit.  `llvm::Error toError() const` perhaps.<br>
Besides the success case issue, I believe having a member function (be it a conversion operator or not) allows us to have better interoperability between the two. I think of `Status` as being at the same level as llvm::Error, as it also tries to multiplex multiple error kinds into a single type, whereas having LLDBError introduces a subordinate relationship. Having a member function allows us to represent errno errors the same way as llvm does it.<br>
<br>
Instead of an LLDBError I'd propose we create more specialized error classes (UnwindError? NetworkError?) once we actually identify use cases for them. Although we can have LLDBError as a common superclass of these errors if you think it will be useful.<br>
<br>
<br>
<a href="https://reviews.llvm.org/D33241" rel="noreferrer" target="_blank">https://reviews.llvm.org/D33241</a><br>
<br>
<br>
<br>
</blockquote></div>