Skip to content

Conversation

@Quuxplusone
Copy link
Contributor

R.12 and R.13 were both written for a C-style world where the idea was
to get raw resource pointers and then quickly "give" them to RAII types.
The modern C++ approach is actually to use RAII types exclusively,
and never handle raw resource pointers even for an instant. R.13 was
kind of getting there, in that it knew make_shared existed; but we
can strengthen the rule by just saying "Please, use make_shared."

"Raw pointers are like raw meat: don't touch them with your hands."

Also update for C++17: it's no longer true that shared_ptr<T>(new T())
can ever cause a leak due to interleaved evaluations.
But it's still a nasty habit. Use make_shared.

(Simple) Warn on any explicit use of `new` and `delete`. Suggest using `make_unique` instead.

### <a name="Rr-immediate-alloc"></a>R.12: Immediately give the result of an explicit resource allocation to a manager object
### <a name="Rr-immediate-alloc"></a>R.12: Instead of manual resource allocation, use factory functions that return RAII objects
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This appears to be assuming that you're the author of the factory function in question? What if it's an old third party library? or it's a c library? Or the person next to you wrote it and you don't want to have to go and change all the places it's already called?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

(Simple) Warn on any explicit use of `new` and `delete`. Suggest using `make_unique` instead.

### <a name="Rr-immediate-alloc"></a>R.12: Immediately give the result of an explicit resource allocation to a manager object
### <a name="Rr-immediate-alloc"></a>R.12: Instead of manual resource allocation, use factory functions that return RAII objects
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The point of R.12 is exactly about calling C/POSIX/WinAPI/etc "explicit resource allocation" interfaces.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

That line is a footnote on what is shaping up to be a different rule.

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

...that is not to say it's wrong, but perhaps it needs to be more obvious that it covers the old cases of making calls to explicit APIs as the old example showed.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

"That line is a footnote on what is shaping up to be a different rule." — Eh? The text I was trying to point to is new in this pull request, and belongs to rule R.12: Instead of manual resource allocation, use factory functions that return RAII objects. Here it is:

If there is no factory wrapper for your RAII type yet, write one!

    void bad(const char *name)
    {
        FILE *f = fopen(name, "r");  // bad
        AutoFclosingFile myfile(f);
        // ...
    }

    void good(const char *name)
    {
        AutoFclosingFile myfile = my::open_file(name, "r");
        // ...
    }

Nobody should ever be calling "explicit resource allocation interfaces" directly in user code (conflating their actual business logic with manual resource management); that would violate rules F.2 and F.3 and perhaps F.1. Resources should be managed by resource-management classes.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm a fan of collapsing of the rules, the "write a wrapper" is what I always call for.

R.12 and R.13 were both written for a C-style world where the idea was
to get raw resource pointers and then quickly "give" them to RAII types.
The modern C++ approach is actually to use RAII types exclusively,
and never handle raw resource pointers even for an instant. R.13 was
kind of getting there, in that it knew `make_shared` existed; but we
can strengthen the rule by just saying "Please, use `make_shared`."

"Raw pointers are like raw meat: don't touch them with your hands."

Also update for C++17: it's no longer true that `shared_ptr<T>(new T())`
can ever cause a leak due to interleaved evaluations.
But it's still a nasty habit. Use `make_shared`.
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

None yet

Projects

None yet

Development

Successfully merging this pull request may close these issues.

5 participants