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

[WIP]: Add sections on formal verification and Tamarin #12

Open
wants to merge 7 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from 6 commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
38 changes: 38 additions & 0 deletions .github/workflows/dictionary.txt
Original file line number Diff line number Diff line change
@@ -1,20 +1,26 @@
.semgrepignore
0x30
0xdea
20-ci
AFL
APIs
AppSec
arity
AST
ATL
auditability
autobuild
autocomplete
autofix
autofixes
Autofix
Autofixes
autogenerated
bitstrings
bookCollapseSection
bookFlatSection
cargo-fuzz
ciphertext
CLI
CMake
codebase
Expand All @@ -24,6 +30,9 @@ CodeQL
codeql-queries
config
Config
cryptographic
CryptoVerif
CryptoVerif's
CSV
CTFs
Ctrl
Expand All @@ -33,8 +42,11 @@ dataflow
DeepSemgrep
DevOps
derefs
Disjunction
Diffie-Hellman
dgryski
Dockerfiles
Dolev-Yao
elttam
eval
Exiv2
Expand All @@ -52,6 +64,8 @@ goroutine
GuardDog
Homebrew
hugo
IETF
indistinguishability
interprocedural
intraprocedural
ioutil-readdir-deprecated
Expand All @@ -63,6 +77,7 @@ lifecycle
LSP
lxml-in-pandas
macOS
MacOS
memcpy
Memcpy
metavariable
Expand All @@ -80,17 +95,29 @@ mindedsecurity
misconfiguration
multiline
multivalued
multiset
natively
Neovim
NIST
NixOS
Nixpkgs
NodeJsScan
nonce
nonces
Nullcon
OCaml
OWASP
Pacman
PCRE-compatible
plaintext
Pohekar
postMessage
pre-built
pre-compiled
propositionally
prover
ProVerif
ProVerif's
README
QL
repo
Expand All @@ -101,8 +128,10 @@ Rulesets
sanitization
SARIF
SARIF-file
Sapic
SAST
SDLC
se
semgrep
Semgrep
SEMGREP_SEND_METRICS
Expand All @@ -111,17 +140,26 @@ semgrep-rules
Semgrep's
semgrep-rules-manager
semgrep-rules-android-security
sequenceDiagram
Sheeraz
Shreya
src
subdirectories
subsorts
Tamarin
Tamarin's
trailofbits
Triaging
Trello
Undecorated
unhandled
unsanitized
untracked
untrusted
verifier
verifiers
Verifpal
Verifpal's
VSCode
vuln
XSS
Expand Down
2 changes: 1 addition & 1 deletion .github/workflows/markdown.yml
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@ jobs:
with:
command: config
globs: |
.github/workflows/.markdownlint.jsonc
.markdownlint.jsonc
**/*.md
# Spellcheck Markdown files using `retext` and `remark`
# Uses: a custom dictionary file
Expand Down
55 changes: 55 additions & 0 deletions .github/workflows/preview.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,55 @@
# .github/workflows/preview.yml
name: Deploy PR previews

on:
pull_request:
types:
- opened
- reopened
- synchronize
- closed

concurrency: preview-${{ github.ref }}

permissions:
pull-requests: write

# Default to bash
defaults:
run:
shell: bash

jobs:
# Build job
build-deploy:
runs-on: ubuntu-latest
env:
HUGO_VERSION: 0.122.0
steps:
- name: Install Hugo CLI
run: |
wget -O ${{ runner.temp }}/hugo.deb https://github.com/gohugoio/hugo/releases/download/v${HUGO_VERSION}/hugo_extended_${HUGO_VERSION}_linux-amd64.deb \
&& sudo dpkg -i ${{ runner.temp }}/hugo.deb
- name: Install Dart Sass Embedded
run: sudo snap install dart-sass-embedded
- name: Checkout
uses: actions/checkout@v3
with:
submodules: recursive
- name: Install Node.js dependencies
run: "[[ -f package-lock.json || -f npm-shrinkwrap.json ]] && npm ci || true"
- name: Build with Hugo
env:
# For maximum backward compatibility with Hugo modules
HUGO_ENVIRONMENT: production
HUGO_ENV: production
run: |
hugo \
--minify \
--baseURL "https://trailofbits.github.io/testing-handbook-preview/pr-preview/pr-${{ github.event.number }}/"
- name: Deploy preview
uses: rossjrw/pr-preview-action@v1
with:
source-dir: ./public/
token: ${{ secrets.TESTING_HANDBOOK_PREVIEW_REPO }}
deploy-repository: trailofbits/testing-handbook-preview
File renamed without changes.
21 changes: 21 additions & 0 deletions assets/mermaid.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,21 @@
{
"theme": "base",
"themeVariables": {
"actorBkg": "#dcefff",
"actorBorder": "#6d9eeb",
"actorTextColor": "#000000",
"actorLineColor": "#576675",
"noteBkgColor": "#e7edf4",
"noteTextColor": "#000000",
"noteBorderColor": "#d9d9d9"
},
"fontSize": "20px",
"fontFamily": "monospace",
"sequence": {
"showSequenceNumbers": true
},
"messageAlign": "center",
"wrap": true,
"useMaxWidth": false,
"noteAlign": "center"
}
8 changes: 7 additions & 1 deletion content/_index.md
Original file line number Diff line number Diff line change
Expand Up @@ -67,6 +67,12 @@ We currently cover the following tools:

- Fuzzing. _Coming soon!_

<--->

#### Formal verification

- [Tamarin]({{< relref "tamarin" >}})

{{< /columns >}}

We are working on expanding the tools we cover here. We are also planning to
Expand Down Expand Up @@ -107,4 +113,4 @@ codeql database analyze codeql.db --format=sarif-latest --output=results.sarif -
We want to actively maintain the highest possible quality and expand the content of Testing Handbook.
If you see a way to improve the Testing Handbook, please let us know! The best way to let us know is
by raising an issue directly on the [Testing Handbook GitHub page](https://github.com/trailofbits/testing-handbook).
<!-- markdown-link-check-enable -->
<!-- markdown-link-check-enable -->
137 changes: 137 additions & 0 deletions content/docs/formal-verification/_index.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,137 @@
---
weight: 2
bookFlatSection: true
title: "Formal verification"
---

# Formal verification

This section presents a number of tools that can be used to formally verify
cryptographic protocols. It currently covers the symbolic protocol verifier
Tamarin, but our long-term goal is to cover both symbolic and computational
verifiers. For each tool, we go though:
Copy link
Collaborator

Choose a reason for hiding this comment

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

though -> through


- Installation and basic use
- How to define and formally verify a new model
- Tool-specific pain points, and potential workarounds

{{< section >}}

## Who is formal verification for?

Formal verification tools allow us to formally prove security properties of cryptographic protocols. Alternatively, they
can often provide counterexamples (and sometimes real attacks) showing that a particular protocol does not guarantee the
security properties that we expect it to. To formally verify a protocol we need to describe the protocol we are
reviewing in a language understood by the tool, and we also need to describe the security properties that we would like
the protocol to have. The tool will then automatically search for a formal proof that the properties we have specified
are upheld by the protocol, and if it terminates, it will output either a proof showing that the properties hold, or a
counterexample showing how some property fails to hold.

Choose a reason for hiding this comment

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

It might be worth highlithing somehwere that Cryptoverif doesn't find attacks/counter-examples when it can't find a proof. At the same time, when if it doesn't find a proof that doesn't necesariy imply insecurity.


For this reason, formal verification tools provide great value for anyone designing a new cryptographic protocol. They
allow developers to verify that a new design meets the expected security guarantees. They allow us to experiment with
the design and compare the security properties and trade-offs between different design choices. Formal verification
tools also provide great value for anyone who is modifying or extending an already existing protocol. If the original
protocol already has a formal model, modeling an extension to the protocol is typically cheap, allowing the developer to
prove that the protocol extension is secure without having to model the entire protocol from scratch.

{{< hint danger >}}

### What is a cryptographic protocol?

You may think that only cryptographers design cryptographic protocols, and that you don't need complex tools to
understand the security properties of your relatively simple use case. However, our experience shows that it makes more
sense to take the following very broad view of the term _cryptographic protocol_, which also has implications for formal
verification.

**Anyone who is composing different cryptographic primitives (like encryption, signatures, and hashes) in a way that has
not been previously specified by a standards document like an IETF RFC or NIST standard, and has not been analyzed in a
public academic paper, is designing a new cryptographic protocol.**
Comment on lines +46 to +48

Choose a reason for hiding this comment

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

Yes! this this this


New cryptographic protocols need to be proven secure before they are deployed. Formal verification is one way of
achieving this which ensures both correctness and auditability. At Trail of Bits, we recommend that _all_ new
cryptographic protocols should be formally verified.

{{< /hint >}}

## The symbolic model or the computational model?

Tools used to formally verify cryptographic protocols typically come in one of two different flavors. Tools like
[ProVerif](https://bblanche.gitlabpages.inria.fr/proverif/), [Tamarin](https://tamarin-prover.com/), and
[Verifpal](https://verifpal.com/) analyze protocols in the _symbolic model_. This means that cryptographic primitives
are modeled as black boxes satisfying certain given equations. For example, a symmetric encryption scheme could be
modeled using two functions `encrypt` and `decrypt` satisfying the following equation:

```js
decrypt(key, encrypt(key, data)) = data
```

This allows the tool to replace any occurrence of the term `decrypt(key, encrypt(key, data))` with `data`, but it says
nothing about the strength of the encryption algorithm. In a sense, it treats the encryption as perfect, since the only
way to obtain the plaintext from the ciphertext is with knowledge of the correct key. Modeling cryptographic primitives
in this way implies a number of trade-offs that may not be immediately obvious. On the one hand, it provides clean
abstractions that allows us to specify the high-level building blocks of cryptographic protocols without having to worry
too much about how each cryptographic primitive is implemented. This also allows us to instantiate the protocol with any
primitive that satisfies the given equations and still be sure that the security proofs hold. On the other hand, it
means that we cannot reason about some things like brute-force or padding-oracle attacks in a straight-forward manner.
(If we want to do this, we need to model our primitives in a way that allow us to express these types of attacks within
the model. Depending on the security properties we are interested in, this may either add complexity, which often has
adverse effects on proving time, or may sometimes be impossible within the symbolic model.)

Symbolic verification tools typically model the network as untrusted, allowing the attacker to intercept, delay, modify,
and replay messages between participants at will. This is known as the _Dolev-Yao model_, and was first described in the
paper [_On the Security of Public-Key Protocols_](https://www.cs.huji.ac.il/~dolev/pubs/dolev-yao-ieee-01056650.pdf).
Individual tools often provide abstractions (like the `reliable-channel` builtin in Tamarin or the `passive` keyword in
Verifpal) that can be used to restrict the capabilities of the attacker in different ways.

Tools like [CryptoVerif](https://bblanche.gitlabpages.inria.fr/CryptoVerif/) analyze protocols in the _computational
model_. Here, individual messages are modeled as bitstrings, cryptographic primitives are modeled as functions on
bitstrings, and the adversary is modeled as an arbitrary probabilistic polynomial-time Turing Machine. The probability
of the attacker successfully breaking the security of a given primitive has to be provided up-front by the user. The
Comment on lines +88 to +89

Choose a reason for hiding this comment

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

I thinking this part could use same treatment as in the symbolic model("On the one hand, it provides clean
abstractions that allows us to specify the high-level building blocks of cryptographic protocols without having to worry
too much about how each cryptographic primitive is implemented"). The reasoning is: in some sense the probability of breaking a primitive isn't really a "responsiblity" of the user. That probability or advantage can be arbitary and Cryptoverif will find a proof(if possible) regardless. What the user should specify is the abstract security guarantee of the primitive which includes an advantage that can be arbitrary. So in this way cryptoverif allows to work with clean abstractions say of an AEAD and the real instanciation is given by the advantage. In the end, the advantages become important later for evaluating the concerete security. This might also be a good warning to users to carefully interpret these results.

tool can then automatically prove indistinguishability between sequences of games (up to some probability), where the
first game captures the analyzed protocol, and the final game is one where the required security property is obvious.
The output is given as a bound on the probability of a successful attack as a function of the security parameters of the
individual primitives used, and the number of protocol executions. A key benefit of the computational model is that this
Comment on lines +92 to +93

Choose a reason for hiding this comment

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

"Function of the advantages of the individual primitives" might work better here. In practice there's mainly a single security parameter for the entire protocol. AFAIK, by default Cryptoverif performs a concrete analysis rather than asympotic one, which would also justify omitting security parameters from the discussion.

mimics how cryptographers usually prove security of cryptographic protocols. This makes it easier to carry over
constructions from traditional pen-and-paper proofs to automated proofs.

The computational model is clearly more expressive than the symbolic model. However, it is also presupposes a deep
understanding of provable security and game-based proofs, which is not required when working in the symbolic model.

Ultimately, which tool to use depends as much on your background as on the protocol you are trying to analyze. If you

Choose a reason for hiding this comment

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

Should we consider framing them as complementary? I probably need to re-read again, but currently, it reads as if the tools are, to some extent, opposites.
Maybe we can point out that (as already done now) thatfor most users, especially those with little cryptographic analysis background, symbolic tools will provide the most value very fast. On the other hand, one could do everything in the computational model, but that’s painful. Power users like Inria tend to do tons of symbolism first and conclude the analysis with computational.

don't have a background in cryptography or formal verification we recommend that you start out with a simple symbolic
verification tool like Verifpal. If you struggle to model your protocol in Verifpal, or need to express more complex
security properties that are not supported by the tool, we suggest switching to a more expressive and mature symbolic
prover like ProVerif or Tamarin. Finally, if you want to bound attack probabilities or translate a game-based proof from
a paper, you need to work in the computational model with a tool like CryptoVerif.

{{< hint info >}}

### Where do I start?

[Verifpal](https://verifpal.com) is a great starting point if you have no previous exposure to formal verification. It
has a small set of built-in primitives that cover a range of use-cases, and the syntax mimics how developers typically
think about cryptographic protocols. Verifpal's intuitive syntax also makes the resulting models easier to parse and
maintain for less experienced users.

This is an important point. If you develop a formal model of a protocol that is too complex for the developers of the
protocol to maintain, then the proof becomes a snapshot in time with limited usefulness. If developers understand the
model and can update and extend it as the protocol develops, it can be built on to provide assurance for future versions
of the protocol as well.

{{< /hint >}}

{{< hint info >}}

### How do I start?

Since most cryptographic protocols have a large number of valid states, formal verification tools often struggle because
of state-space explosion. For this reason, it is generally a good idea to try to divide your protocol into smaller
components, or sub-protocols, like _registering a new device_ or _sending and receiving a single message_, and try to
prove that these individual components provide the expected security guarantees.

This approach clearly limits what the attacker can do since each sub-protocol is analyzed in a vacuum. However, it makes

Choose a reason for hiding this comment

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

This could also be a great place to highlight the complementarity of tools. In the computation model we can find composability results allowing to easily break down protocols in subprotocol withouth worrying about composition. So if such a composition result exist it's nice then to focus symbolic effort on a subprotocol without worrying either.

each model more manageable and helps avoid the issue of state-space explosion. At Trail of Bits we often use this method
to quickly model protocol components on cryptographic design reviews. It is both a useful tool to verify our
understanding of the protocol, and to identify potential weaknesses in the design.

{{< /hint >}}
Loading