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

MACRO: expand attribute proc macros #7194

Merged
merged 2 commits into from
Oct 18, 2021
Merged

MACRO: expand attribute proc macros #7194

merged 2 commits into from
Oct 18, 2021

Conversation

vlad20012
Copy link
Member

@vlad20012 vlad20012 commented Apr 30, 2021

Issue #6908

Despite the fact that this is a draft, it is useable.

What is done:

  • We can distinguish procedural macro attributes from other attributes (including tool attributes, #![register_attr()] and #![register_tool()])
  • An item under procedural macro attribute is considered !existsAfterExpansion (see Introduce existsAfterExpansion, use it instead of isEnabledByCfg #7187), so it feels like a cfg-disabled code (or like a regular function-like macro call body)
  • When a procedural macro is expanded, IDE analysis works with expanded elements (instead of original elements under a macro attribute)
  • There is a concept of hardcoded "not a macro" macros (better names are welcome). The idea is that there are procedural macros that feel like regular attributes and should be simply ignored by the IDE. For example, tokio::main and tokio::test. We want to allow proc macro authors to specify "not a macro" status in Cargo.toml metadata
  • Navigation takes into account range mapping
  • Find usages works in items under procedural macro attributes

TODO (maybe in next PRs):

  • Highlighting (RsMacroExpansionHighlightingPass)
  • Fix existsAfterExpansion vs isEnabledByCfg - an element can be wrongly considered cfg-disabled when it is a part of procedural macro and vise versa
  • Support attribute macros on declarative macro definitions and regular macro calls
  • Clean doPrepareCustomDeriveMacroCallBody/doPrepareAttributeProcMacroCallBody
  • Support "Expand recursively" action for attribute macros
  • Correctly resolve references inside attribute macros
  • Support range mapping for an attribute value
  • (!) Store CustomAttributes in defMap, pass it to getProcMacroAttributeRaw
  • Fill in the hardcoded macros list
  • Allow specifying not-a-macro status in cargo metadata
  • Measure how item text storage impacts memory footprint
  • Measure how heavy existsAfterExpansion and ProcMacroAttribute.getProcMacroAttribute implementations impact the performance
  • Check $crate & hygiene
  • Investigate why find usages doesn't work in web-sys crate
  • "not-a-macro" works only for some kinds of items (functions, structs, etc). Doublecheck that ProcMacroAttribute.getProcMacroAttribute takes this fact into account
  • Add test for proc macro body preparation (undecoration)
  • Show generated items in the structure view
  • Check completion in proc macros

changelog: Provide initial support for attribute procedural macros. Now the plugin can expand such procedural macro calls and take into account all expanded items in type inference and name resolution. Note, it’s only an initial implementation, so it may work in an unexpected way in some cases. The feature is disabled by default for now. To turn it on, you should enable Use experimental name resolution engine option in Preferences | Languages & Frameworks | Rust settings and enable org.rust.cargo.evaluate.build.scripts and org.rust.macros.proc experimental features. Don’t forget to reload a project structure after enabling the corresponding features via Refresh Cargo Projects action on Cargo tool window. See tracking issue for more details and the current status of procedural macros support

@vlad20012 vlad20012 force-pushed the proc-macro-attribute branch from 0d14d6a to c7cfd14 Compare April 30, 2021 19:52
@vlad20012 vlad20012 changed the title MACRO: support attribute proc macro expansion MACRO: expand attribute proc macros Apr 30, 2021
@vlad20012 vlad20012 force-pushed the proc-macro-attribute branch from c7cfd14 to d46b87b Compare April 30, 2021 20:47
@matklad
Copy link
Member

matklad commented May 16, 2021

cc @jonas-schievink :0)

bors bot added a commit that referenced this pull request Jun 9, 2021
6745: ANN: add quick fix to add function parameter from function call r=dima74 a=Kobzol

This PR adds a quick fix that lets the user add new function parameters to a function from a function call, using the new Change signature refactoring. So far it only works for function and method calls and not tuple struct constructors. The behaviour of the fix is heavily inspired by a similar fix in the Kotlin plugin.
![add-parameters](https://user-images.githubusercontent.com/4539057/106059839-964ed400-60f3-11eb-973b-983495cc76fd.gif)

There are two TODOs in the code, which I'm not sure how to resolve:
- How to name parameters? `p0`, `p1`, etc.? Or continue counting from the existing parameter count? Also the fix should probably make sure that it doesn't generate collisions in parameter binding names.
- How to get the correct type reference from the argument expression type? That is especially important for import to work. I tried to use a type reference fragment, that sort-of works, but it imports the type with a wrong path. I need to somehow get from `Ty` to `RsTypeReference` that exists in the context of the target function.

Related issue: #6744
Fixes: #3582

changelog: Add quick fix to add a parameter to a function from a function call.

7229: STUB: stub macro definitions and stub macro calls in code blocks if the block is stubbed  r=vlad20012 a=vlad20012

We need this because `RsItemsOwner.itemsAndMacros` should
behave identically on stubs and on AST. Otherwise, it leads to
weird flaky bugs. This fixes the regression in #7194

Also, macro definitions should be available in symbol search as well as other
items. So they should be stubbed and indexed, even local ones

changelog: FEATURE: Add local macro definitions to `symbol search`. FIX: Fix possible flaky false positives around local macro calls


Co-authored-by: Jakub Beránek <[email protected]>
Co-authored-by: vlad20012 <[email protected]>
@vlad20012 vlad20012 self-assigned this Jul 20, 2021
@vlad20012 vlad20012 force-pushed the proc-macro-attribute branch 7 times, most recently from ac2a0d5 to c6cebd8 Compare August 9, 2021 07:49
@vlad20012 vlad20012 marked this pull request as ready for review August 9, 2021 11:27
@vlad20012 vlad20012 requested review from dima74 and Undin August 9, 2021 11:27
@vlad20012 vlad20012 force-pushed the proc-macro-attribute branch 2 times, most recently from d3b2171 to 4c892a8 Compare August 10, 2021 17:07
bors bot added a commit that referenced this pull request Aug 30, 2021
7751: GRAM&ANN: parse `unsafe` keyword in modules and annotate it as error r=vlad20012 a=Undin

These changes allow parsing mod items with `unsafe` keyword to be consistent with rustc (see rust-lang/rust#75857) and annotate `unsafe` keyword as an error in such cases on the semantic level.

![image](https://user-images.githubusercontent.com/2539310/131228287-fc3bfcbb-efd2-4f57-9466-a1d196477b3e.png)


Note, the corresponding change in rustc was made mainly to provide token tree to proc macros where `unsafe` keyword should be handled by proc macro itself. And in case of such proc macro, `unsafe` keyword shouldn't be annotated as an error. But after these changes, `unsafe` will be annotated as an error by the plugin. But it doesn't look like a new issue, since previously it was annotated as parser error.

For example, for the following code
```rust
//- lib.rs
use proc_macro::TokenStream;
use std::str::FromStr;

#[proc_macro_attribute]
pub fn my_attr(_attr: TokenStream, _item: TokenStream) -> TokenStream {
    return TokenStream::from_str("pub fn foo() { println!(\"Hello from macro!\")}").unwrap()
}

//- main.rs
#[rust_project::my_attr]
unsafe mod foo {}

fn main() {
    foo();
}
```

the plugin produces the corresponding annotations
| Before | After |
| - | - |
| ![image](https://user-images.githubusercontent.com/2539310/131228207-b9b0ca35-c6dc-4ea3-b91e-0b894cde1d41.png) | ![image](https://user-images.githubusercontent.com/2539310/131228220-4ef37560-ffee-447c-bde1-a87fa4cb44b6.png) |


Proper support for error annotations of such cases should be implemented separately, not only for `unsafe` keyword in modules and requires changes from #7194




changelog: Parse `unsafe` keyword in modules and provide error annotation for it


Co-authored-by: Arseniy Pendryak <[email protected]>
bors bot added a commit that referenced this pull request Aug 30, 2021
7751: GRAM&ANN: parse `unsafe` keyword in modules and annotate it as error r=vlad20012 a=Undin

These changes allow parsing mod items with `unsafe` keyword to be consistent with rustc (see rust-lang/rust#75857) and annotate `unsafe` keyword as an error in such cases on the semantic level.

![image](https://user-images.githubusercontent.com/2539310/131228287-fc3bfcbb-efd2-4f57-9466-a1d196477b3e.png)


Note, the corresponding change in rustc was made mainly to provide token tree to proc macros where `unsafe` keyword should be handled by proc macro itself. And in case of such proc macro, `unsafe` keyword shouldn't be annotated as an error. But after these changes, `unsafe` will be annotated as an error by the plugin. But it doesn't look like a new issue, since previously it was annotated as parser error.

For example, for the following code
```rust
//- lib.rs
use proc_macro::TokenStream;
use std::str::FromStr;

#[proc_macro_attribute]
pub fn my_attr(_attr: TokenStream, _item: TokenStream) -> TokenStream {
    return TokenStream::from_str("pub fn foo() { println!(\"Hello from macro!\")}").unwrap()
}

//- main.rs
#[rust_project::my_attr]
unsafe mod foo {}

fn main() {
    foo();
}
```

the plugin produces the corresponding annotations
| Before | After |
| - | - |
| ![image](https://user-images.githubusercontent.com/2539310/131228207-b9b0ca35-c6dc-4ea3-b91e-0b894cde1d41.png) | ![image](https://user-images.githubusercontent.com/2539310/131228220-4ef37560-ffee-447c-bde1-a87fa4cb44b6.png) |


Proper support for error annotations of such cases should be implemented separately, not only for `unsafe` keyword in modules and requires changes from #7194




changelog: Parse `unsafe` keyword in modules and provide error annotation for it


Co-authored-by: Arseniy Pendryak <[email protected]>
bors bot added a commit that referenced this pull request Aug 30, 2021
7751: GRAM&ANN: parse `unsafe` keyword in modules and annotate it as error r=vlad20012 a=Undin

These changes allow parsing mod items with `unsafe` keyword to be consistent with rustc (see rust-lang/rust#75857) and annotate `unsafe` keyword as an error in such cases on the semantic level.

![image](https://user-images.githubusercontent.com/2539310/131228287-fc3bfcbb-efd2-4f57-9466-a1d196477b3e.png)


Note, the corresponding change in rustc was made mainly to provide token tree to proc macros where `unsafe` keyword should be handled by proc macro itself. And in case of such proc macro, `unsafe` keyword shouldn't be annotated as an error. But after these changes, `unsafe` will be annotated as an error by the plugin. But it doesn't look like a new issue, since previously it was annotated as parser error.

For example, for the following code
```rust
//- lib.rs
use proc_macro::TokenStream;
use std::str::FromStr;

#[proc_macro_attribute]
pub fn my_attr(_attr: TokenStream, _item: TokenStream) -> TokenStream {
    return TokenStream::from_str("pub fn foo() { println!(\"Hello from macro!\")}").unwrap()
}

//- main.rs
#[rust_project::my_attr]
unsafe mod foo {}

fn main() {
    foo();
}
```

the plugin produces the corresponding annotations
| Before | After |
| - | - |
| ![image](https://user-images.githubusercontent.com/2539310/131228207-b9b0ca35-c6dc-4ea3-b91e-0b894cde1d41.png) | ![image](https://user-images.githubusercontent.com/2539310/131228220-4ef37560-ffee-447c-bde1-a87fa4cb44b6.png) |


Proper support for error annotations of such cases should be implemented separately, not only for `unsafe` keyword in modules and requires changes from #7194




changelog: Parse `unsafe` keyword in modules and provide error annotation for it


Co-authored-by: Arseniy Pendryak <[email protected]>
@vlad20012 vlad20012 force-pushed the proc-macro-attribute branch from 4c892a8 to d78367f Compare August 31, 2021 15:31
bors bot added a commit that referenced this pull request Sep 15, 2021
7818: PERF: Expand macros in parallel during CrateDefMap building r=vlad20012 a=dima74

Now we expand macros in new resolve using following pipeline:
* Resolve all macros (single thread)
* Expand macros in batches (multithread). Usually [macro cache](#6212) is used here
* Add expanded elements from all macros to DefMap (single thread)

Most performance impact will be on crates with lots of procedural macros (such as [`web-sys`](https://github.com/rustwasm/wasm-bindgen/tree/master/crates/web-sys)) or when we expand macros first time (without cache).

Here is performance comparison on my 8-core machine with code from #7194 (note that currently we use at most 4 processes to expand procedural macros):

project | macro cache? | single-thread | multi-thread | difference
------- | ------------ | ------------- | ------------ | ----------
web-sys |           no |          ~35s |         ~14s |      ~2.5x
web-sys |          yes |        2920ms |       2620ms |       ~10%
cargo   |           no |        3750ms |       3090ms |       ~17%
cargo   |          yes |        1540ms |       1500ms |        ~0%

changelog: Optimize macro expansion — now they are expanded in parallel

7851: Fix `Generate Parser` run configuration r=vlad20012 a=vlad20012

The configuration was broken in #7828 because of TOML plugin source removal

Co-authored-by: Dmitry Murzin <[email protected]>
Co-authored-by: vlad20012 <[email protected]>
bors bot added a commit that referenced this pull request Sep 15, 2021
7818: PERF: Expand macros in parallel during CrateDefMap building r=vlad20012 a=dima74

Now we expand macros in new resolve using following pipeline:
* Resolve all macros (single thread)
* Expand macros in batches (multithread). Usually [macro cache](#6212) is used here
* Add expanded elements from all macros to DefMap (single thread)

Most performance impact will be on crates with lots of procedural macros (such as [`web-sys`](https://github.com/rustwasm/wasm-bindgen/tree/master/crates/web-sys)) or when we expand macros first time (without cache).

Here is performance comparison on my 8-core machine with code from #7194 (note that currently we use at most 4 processes to expand procedural macros):

project | macro cache? | single-thread | multi-thread | difference
------- | ------------ | ------------- | ------------ | ----------
web-sys |           no |          ~35s |         ~14s |      ~2.5x
web-sys |          yes |        2920ms |       2620ms |       ~10%
cargo   |           no |        3750ms |       3090ms |       ~17%
cargo   |          yes |        1540ms |       1500ms |        ~0%

changelog: Optimize macro expansion — now they are expanded in parallel

Co-authored-by: Dmitry Murzin <[email protected]>
@vlad20012 vlad20012 force-pushed the proc-macro-attribute branch 2 times, most recently from 4f0891c to bc83fc6 Compare September 17, 2021 13:20
Copy link
Member

@Undin Undin left a comment

Choose a reason for hiding this comment

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

Sorry, for long review but code is quite hard 😅

It works in general but some notes and questions:

  • Some tests fail
  • I get Low Memory notification if I run IDE with default -Xmx768m. I see that you haven't checked memory consumption yet, so maybe it's worth adding a note in the tracking issue about it?
  • I get errors with nightly toolchain (rustc 1.56.0-nightly (b7404c898 2021-09-03)) although everything is ok with stable one. Yet another change in API?

Also, it would be great if @dima74 checks changes in name resolution code

Comment on lines 154 to 160
val props = attr.attr.resolveToProcMacroWithoutPsiUnchecked(checkIsMacroAttr = false)?.props
if (props != null && props.treatAsBuiltinAttr && RsProcMacroPsiUtil.canFallbackItem(owner)) {
None
} else {
attr
}
Copy link
Member

Choose a reason for hiding this comment

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

Is it true that we return some Attr instance even if we failed to resolve proc macro?

Copy link
Member Author

Choose a reason for hiding this comment

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

Absolutely

fn <FUNCTION>main</FUNCTION>() {
<ATTRIBUTE>#!</ATTRIBUTE><ATTRIBUTE>[crate_type = <STRING>"lib"</STRING></ATTRIBUTE><ATTRIBUTE>]</ATTRIBUTE>
}
""")
Copy link
Member

Choose a reason for hiding this comment

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

I'd add more tests with non trivial proc macro

Copy link
Member

@Undin Undin left a comment

Choose a reason for hiding this comment

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

Still receive Low Memory notification on Rocket project with -Xmx768m
Note, -Xmx1024m is enough for me for this project

@dima74 dima74 mentioned this pull request Sep 24, 2021
18 tasks
@vlad20012 vlad20012 force-pushed the proc-macro-attribute branch 3 times, most recently from 77baa8e to 682d772 Compare October 4, 2021 15:18
@Undin Undin force-pushed the proc-macro-attribute branch from 682d772 to 7e00146 Compare October 18, 2021 08:54
Copy link
Member

@Undin Undin left a comment

Choose a reason for hiding this comment

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

bors r+

@vlad20012
Copy link
Member Author

bors r-

@bors
Copy link
Contributor

bors bot commented Oct 18, 2021

Canceled.

@vlad20012 vlad20012 force-pushed the proc-macro-attribute branch from 7e00146 to 1974938 Compare October 18, 2021 14:51
@vlad20012
Copy link
Member Author

bors r=undin

@bors
Copy link
Contributor

bors bot commented Oct 18, 2021

Build succeeded:

@bors bors bot merged commit de9a704 into master Oct 18, 2021
@bors bors bot deleted the proc-macro-attribute branch October 18, 2021 18:07
@github-actions github-actions bot added this to the v158 milestone Oct 18, 2021
@mili-l mili-l self-assigned this Oct 22, 2021
bors bot added a commit that referenced this pull request Nov 1, 2021
7725: MACRO&RES: support arbitrary items expanded from custom derive r=dima74 a=vlad20012

Depends on #7194

changelog: Take into account all kinds of items generated by custom derive procedural macros. Previously only `impl` blocks were accounted. Please, note that procedural macro expansion is experimental. If you want to try it, enable `org.rust.cargo.evaluate.build.scripts` and `org.rust.macros.proc` experimental features


7908: INT: use Template builder in DestructureIntention and remove RsMultipleVariableRenamer r=dima74 a=Kobzol

`RsMultipleVariableRenamer` was a hack and since the template builder was upgraded, it's no longer needed.

Related: #5650

7959: REF: handle name collisions in RsIntroduceConstantHandler r=dima74 a=Kobzol

This PR adds basic handling of name collisions in `RsIntroduceConstantHandler`.

`@dima74` Is there a better way to get all bindings/names at a given place?

Fixes: #7957

changelog: Handle name collisions in Introduce Constant refactoring.

Co-authored-by: vlad20012 <[email protected]>
Co-authored-by: Jakub Beránek <[email protected]>
bors bot added a commit that referenced this pull request Nov 2, 2021
6929: ANN: Provide lifetime name check r=mchernyavsky a=Stzx

changelog:

Fixed: #6927


7719: ANN: support E0451 for shorthand struct field literals r=mchernyavsky a=Kobzol

I noticed that E0451 (struct field is private) wasn't being applied for shorthand field literals. This PR fixes it.

changelog: Support the E0451 compiler error for struct field literals using shorthand init.

7725: MACRO&RES: support arbitrary items expanded from custom derive r=dima74 a=vlad20012

Depends on #7194

changelog: Take into account all kinds of items generated by custom derive procedural macros. Previously only `impl` blocks were accounted. Please, note that procedural macro expansion is experimental. If you want to try it, enable `org.rust.cargo.evaluate.build.scripts` and `org.rust.macros.proc` experimental features


Co-authored-by: Stzx <[email protected]>
Co-authored-by: Jakub Beránek <[email protected]>
Co-authored-by: vlad20012 <[email protected]>
bors bot added a commit that referenced this pull request Nov 2, 2021
7719: ANN: support E0451 for shorthand struct field literals r=mchernyavsky a=Kobzol

I noticed that E0451 (struct field is private) wasn't being applied for shorthand field literals. This PR fixes it.

changelog: Support the E0451 compiler error for struct field literals using shorthand init.

7725: MACRO&RES: support arbitrary items expanded from custom derive r=dima74 a=vlad20012

Depends on #7194

changelog: Take into account all kinds of items generated by custom derive procedural macros. Previously only `impl` blocks were accounted. Please, note that procedural macro expansion is experimental. If you want to try it, enable `org.rust.cargo.evaluate.build.scripts` and `org.rust.macros.proc` experimental features


7770: "Add remaining patterns" quickfix: add patterns even if no `match` body r=mchernyavsky a=t-kameyama

Fixes #7759

```kotlin
enum E { A, B, C }

fn test(e: E, i: i32) {
    match e

    match i

    match {}
}
```

<img width="332" alt="スクリーンショット 2021-09-01 18 59 40" src="https://user-images.githubusercontent.com/1121855/131653263-f8601fb2-5db2-41ea-b876-6ed749d58eb1.png">

↓

<img width="342" alt="スクリーンショット 2021-09-01 19 00 00" src="https://user-images.githubusercontent.com/1121855/131653280-9bfc2eab-6638-46e5-87a6-b865025c47ce.png">


7775: Introduce variable: place new variable in lambda block when target expression is in lambda without braces r=mchernyavsky a=t-kameyama

Fixes #7754

8030: INSP: Add "simplify if with constant condition" inspection r=mchernyavsky a=dima74

We have `RedundantElseInspection` which works for irrefutable patterns and always true `if`-conditions. This PR adds new inspection which works for both always true and always false `if`-conditions. And `RedundantElseInspection` now checks only irrefutable patterns. 

![](https://user-images.githubusercontent.com/6505554/139441929-2b6aea03-1973-47a8-a45c-6f166022a803.gif)

changelog: Add "simplify if with constant condition" inspection


Co-authored-by: Jakub Beránek <[email protected]>
Co-authored-by: vlad20012 <[email protected]>
Co-authored-by: Toshiaki Kameyama <[email protected]>
Co-authored-by: Dmitry Murzin <[email protected]>
bors bot added a commit that referenced this pull request Nov 2, 2021
7725: MACRO&RES: support arbitrary items expanded from custom derive r=dima74 a=vlad20012

Depends on #7194

changelog: Take into account all kinds of items generated by custom derive procedural macros. Previously only `impl` blocks were accounted. Please, note that procedural macro expansion is experimental. If you want to try it, enable `org.rust.cargo.evaluate.build.scripts` and `org.rust.macros.proc` experimental features


Co-authored-by: vlad20012 <[email protected]>
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Projects
None yet
Development

Successfully merging this pull request may close these issues.

4 participants