Skip to content

Commit

Permalink
Document and, or, and not from #680 (#1032)
Browse files Browse the repository at this point in the history
Co-authored-by: Chandler Carruth <chandlerc@gmail.com>

Co-authored-by: Richard Smith <richard@metafoo.co.uk>
  • Loading branch information
jonmeow and zygoloid authored Jan 27, 2022
1 parent 2d567f5 commit 6218aff
Show file tree
Hide file tree
Showing 2 changed files with 164 additions and 0 deletions.
12 changes: 12 additions & 0 deletions docs/design/expressions/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,7 @@ SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
## Table of contents

- [Overview](#overview)
- [Operators](#operators)
- [Conversions and casts](#conversions-and-casts)

<!-- tocstop -->
Expand All @@ -29,6 +30,17 @@ fn Foo(a: i32*) -> i32 {
Here, the parameter type `i32*`, the return type `i32`, and the operand `*a` of
the `return` statement are all expressions.

## Operators

Most expressions are modeled as operators:

| Category | Operator | Syntax | Function |
| ---------- | ----------------------------- | --------- | ------------------------------------------------------------------- |
| Conversion | [`as`](as_expressions.md) | `x as T` | Converts the value `x` to the type `T`. |
| Logical | [`and`](logical_operators.md) | `x and y` | A short-circuiting logical AND: `true` if both operands are `true`. |
| Logical | [`or`](logical_operators.md) | `x or y` | A short-circuiting logical OR: `true` if either operand is `true`. |
| Logical | [`not`](logical_operators.md) | `not x` | Logical NOT: `true` if the operand is `false`. |

## Conversions and casts

When an expression appears in a context in which an expression of a specific
Expand Down
152 changes: 152 additions & 0 deletions docs/design/expressions/logical_operators.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,152 @@
# Logical operators

<!--
Part of the Carbon Language project, under the Apache License v2.0 with LLVM
Exceptions. See /LICENSE for license information.
SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
-->

<!-- toc -->

## Table of contents

- [Overview](#overview)
- [Details](#details)
- [Precedence](#precedence)
- [Associativity](#associativity)
- [Conversions](#conversions)
- [Overloading](#overloading)
- [Alternatives considered](#alternatives-considered)
- [References](#references)

<!-- tocstop -->

## Overview

Carbon provides three operators to support logical operations on `bool` values:

- `and` provides a logical AND operation.
- `x and y` evaluates to `true` if both operands are `true`.
- `or` provides a logical OR operation.
- `x or y` evaluates to `true` if either operand is `true`.
- `not` provides a logical NOT operation.
- `not x` evaluates to `true` if the operand is `false`.

`and` and `or` are infix binary operators, and use
[short-circuit evaluation](https://en.wikipedia.org/wiki/Short-circuit_evaluation).
`not` is a prefix unary operator.

## Details

### Precedence

`and` and `or` have very low precedence. When an expression appearing as the
condition of an `if` uses these operators unparenthesized, they are always the
lowest precedence operators in that expression.

These operators permit any reasonable operator that might be used to form a
`bool` value as a subexpression. In particular, comparison operators such as `<`
and `==` have higher precedence than `and` and `or`. However, the precedence of
`and` and `or` is not directly comparable with each other, so they cannot both
be used directly in an expression without parentheses.

`not` is higher precedence than `and` and `or`, but its precedence is
incomparable with most other operators, including comparison operators.

For example:

```carbon
// ✅ Valid: `and` is lower precedence than the `<` or `==` operators.
if (n + m == 3 and not n < m) {
...
}
// The above is equivalent to:
if (((n + m) == 3) and (not (n < m))) {
...
}
// ❌ Invalid: `and` and `or` precedence is incomparable.
if (cond1 and cond2 or cond3) {
...
}
// ✅ Valid: Parentheses avoid the precedence check.
if (cond1 and (cond2 or cond3)) {
...
}
// ❌ Invalid: `not` precedence is incomparable with `==`.
if (not cond1 == cond2) {
...
}
// ❌ Invalid: `not` precedence is incomparable with `==`.
if (cond1 == not cond2) {
...
}
// ✅ Valid: Parentheses avoid the precedence check.
if (cond1 == (not cond2)) {
...
}
```

### Associativity

`and` and `or` are left-associative. A `not` expression cannot be the operand of
another `not` expression; `not not b` is an error without parentheses.

```
// ✅ Valid: `and` is left-associative, and precedence is fine.
if (not a and not b and not c) {
...
}
// The above is equivalent to:
if ((not a) and ((not b) and (not c))) {
...
}
// ✅ Valid: Parentheses avoid the `not` associativity error.
if (not (not a)) {
...
}
// ❌ Invalid: `not not` associativity requires parentheses.
if (not not a) {
...
}
```

### Conversions

> TODO: This should be addressed through a standard `bool` conversion design.
The operand of `and`, `or`, or `not` is converted to a `bool` value in the same
way as the condition of an `if` statement. In particular:

- If we decide that certain values, such as pointers or integers, should not
be usable as the condition of an `if` without an explicit comparison against
null or zero, then those values will also not be usable as the operand of
`and`, `or`, or `not` without an explicit comparison.
- If an extension point is provided to determine how to branch on the truth of
a value in an `if` (such as by supplying a conversion to a `bool` type),
that extension point will also apply to `and`, `or`, and `not`.

### Overloading

The logical operators `and`, `or`, and `not` are not overloadable. As noted
above, any mechanism that allows types to customize how `if` treats them will
also customize how `and`, `or`, and `not` treats them.

## Alternatives considered

- [Use punctuation spelling for all three operators](/proposals/p0680.md#use-punctuation-spelling-for-all-three-operators)
- [Precedence of AND versus OR](/proposals/p0680.md#precedence-of-and-versus-or)
- [Precedence of NOT](/proposals/p0680.md#precedence-of-not)
- [Punctuation form of NOT](/proposals/p0680.md#punctuation-form-of-not)
- [Two forms of NOT](/proposals/p0680.md#two-forms-of-not)
- [Repeated NOT](/proposals/p0680.md#repeated-not)
- [AND and OR produce the decisive value](/proposals/p0680.md#and-and-or-produce-the-decisive-value)

## References

- Proposal
[#680: And, or, not](https://github.com/carbon-language/carbon-lang/pull/680).
- Proposal
[#702: Comparison operators](https://github.com/carbon-language/carbon-lang/pull/702).

0 comments on commit 6218aff

Please sign in to comment.