Skip to content

Commit

Permalink
Tweak wording in the macros guide
Browse files Browse the repository at this point in the history
  • Loading branch information
kmcallister committed Feb 25, 2015
1 parent 1804242 commit df08657
Showing 1 changed file with 19 additions and 10 deletions.
29 changes: 19 additions & 10 deletions src/doc/trpl/macros.md
Original file line number Diff line number Diff line change
Expand Up @@ -189,14 +189,12 @@ shorthand for a data type could be valid as either an expression or a pattern.

## Repetition

The repetition behavior can seem somewhat magical, especially when multiple
names are bound at multiple nested levels of repetition. The two rules to keep
in mind are:
The repetition operator follows two principal rules:

1. the behavior of `$(...)*` is to walk through one "layer" of repetitions, for
all of the `$name`s it contains, in lockstep, and
1. `$(...)*` walks through one "layer" of repetitions, for all of the `$name`s
it contains, in lockstep, and
2. each `$name` must be under at least as many `$(...)*`s as it was matched
against. If it is under more, it'll be duplicated, as appropriate.
against. If it is under more, it'll be duplicated, as appropriate.

This baroque macro illustrates the duplication of variables from outer
repetition levels.
Expand Down Expand Up @@ -226,6 +224,10 @@ That's most of the matcher syntax. These examples use `$(...)*`, which is a
more" match. Both forms optionally include a separator, which can be any token
except `+` or `*`.

This system is based on
"[Macro-by-Example](http://www.cs.indiana.edu/ftp/techreports/TR206.pdf)"
(PDF link).

# Hygiene

Some languages implement macros using simple text substitution, which leads to
Expand Down Expand Up @@ -273,19 +275,26 @@ macro, using [a GNU C extension] to emulate Rust's expression blocks.
})
```

This looks reasonable, but watch what happens in this example:
Here's a simple use case that goes terribly wrong:

```text
const char *state = "reticulating splines";
LOG(state);
LOG(state)
```

The program will likely segfault, after it tries to execute
This expands to

```text
printf("log(%d): %s\n", state, state);
const char *state = "reticulating splines";
int state = get_log_state();
if (state > 0) {
printf("log(%d): %s\n", state, state);
}
```

The second variable named `state` shadows the first one. This is a problem
because the print statement should refer to both of them.

The equivalent Rust macro has the desired behavior.

```rust
Expand Down

0 comments on commit df08657

Please sign in to comment.