Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Casting 0 to extern "C" fn is permitted #8730

Closed
MicahChalmer opened this issue Aug 24, 2013 · 8 comments
Closed

Casting 0 to extern "C" fn is permitted #8730

MicahChalmer opened this issue Aug 24, 2013 · 8 comments
Assignees
Labels
A-type-system Area: Type system C-enhancement Category: An issue proposing an enhancement or a PR with one.

Comments

@MicahChalmer
Copy link
Contributor

Now that #8666 is in place, there is not a good way to make a NULL extern "C" fn. You can do it, but you have to specify the whole type of the function pointer again with as:

0 as extern "C" fn(args...) -> ReturnType

using std::ptr::null() will not compile, and gives:

hello_nullfuncptrs.rs:4:8: 4:25 error: mismatched types: expected `extern "C" fn(int, int) -> int` but found `*<V0>` (expected extern fn but found *-ptr)

Some C APIs really do need to have NULL passed in for function pointers. Right now, using as like the code above is the only way to create such a null pointer; For those cases where you need to pass a NULL into a C API, it would be nice if std::ptr::null() would work for that, without having to repeat the whole type (including all arguments and return type) of the function pointer in order to do it.

UPDATE: Title amended by @nikomatsakis to reflect the real issue at hand.

@MicahChalmer
Copy link
Contributor Author

This was previously both about this and #8728, but edited to be about the inference issue only; #8728 is for the safety hole.

@nikomatsakis
Copy link
Contributor

The correct type for a nullable extern pointer is Option<extern "C" fn>, and you can pass null with None.

@nikomatsakis
Copy link
Contributor

In light of that, I'm not sure if there is still a bug here or not?

@MicahChalmer
Copy link
Contributor Author

Ah! That does work, and makes sense. I didn't know you could use Option that way and it would still match the C signature of just a pointer--Option is an enum after all, and the current version of the FFI reference specifically says there are no guarantees about the layout of an enum. But as far as usability goes it's great.

All I would ask is for this to be documented the FFI tutorial and in the reference manual. I wouldn't have expected it to be there yet, since nothing from #8666 is documented yet, which I assume is a different issue altogether, or at least in progress.

(Given this, am I expected to close the bug here? Or should only people with commit access be doing that?)

@nikomatsakis
Copy link
Contributor

Do not close the bug, 0 as extern "C" fn should not type check. I will edit the title and so forth. And I agree that this use of Option should be documented, both with respect to the fact that the enum layout in this case is guaranteed and to give examples of how it is done.

@ghost ghost assigned nikomatsakis Aug 25, 2013
@nikomatsakis
Copy link
Contributor

oh, nm, dup of #8728

@olson-dan
Copy link

olson-dan commented Aug 13, 2018

The correct type for a nullable extern pointer is Option<extern "C" fn>, and you can pass null with None.

How does one deal with this in the case of APIs that are not correctly wrapped in an option? Like this one:
https://docs.rs/libz-sys/1.0.20/libz_sys/struct.z_stream.html

Edit:
I found that std::mem:zeroed() works, so I guess that this is an ok workaround.

@BatmanAoD
Copy link
Member

@nikomatsakis Is it still true that None should automatically convert to null when passed to an external function, or did that change? It doesn't seem to work for me, so I don't know if that feature was removed before 1.0 or if I'm misunderstanding how to use it.

Guiguiprim added a commit to Guiguiprim/qmetaobject-rs that referenced this issue Jul 11, 2019
Guiguiprim added a commit to Guiguiprim/qmetaobject-rs that referenced this issue Jul 12, 2019
Guiguiprim added a commit to Guiguiprim/qmetaobject-rs that referenced this issue Jul 12, 2019
Guiguiprim added a commit to Guiguiprim/qmetaobject-rs that referenced this issue Jul 12, 2019
Guiguiprim added a commit to Guiguiprim/qmetaobject-rs that referenced this issue Jul 12, 2019
ogoffart pushed a commit to woboq/qmetaobject-rs that referenced this issue Jul 12, 2019
flip1995 pushed a commit to flip1995/rust that referenced this issue May 5, 2022
[FP] identity_op in front of if

fix rust-lang#8724

changelog: FP: [`identity_op`]: is now allowed in front of if statements, blocks and other expressions where the suggestion would be invalid.

Resolved simular problems with blocks, mathces, and loops.
identity_op always does NOT suggest reducing `0 + if b { 1 } else { 2 } + 3` into  `if b { 1 } else { 2 } + 3` even in the case that the expression is in `f(expr)` or `let x = expr;` for now.
flip1995 pushed a commit to flip1995/rust that referenced this issue Jun 4, 2022
…swij,xFrednet

`identity_op`: add parenthesis to suggestions where required

changelog: [`identity_op`]: add parenthesis to suggestions where required

Follow up to rust-lang#8730, wraps the cases we can't lint as-is in parenthesis rather than ignoring them

Catches a couple new FPs with mixed operator precedences and `as` casts

```rust
// such as
0 + { a } * 2;
0 + a as usize;
```

The suggestions are now applied using `span_lint_and_sugg` rather than appearing in just the message and have a `run-rustfix` test
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
A-type-system Area: Type system C-enhancement Category: An issue proposing an enhancement or a PR with one.
Projects
None yet
Development

No branches or pull requests

4 participants