diff --git a/final/dry-run/concpar21final01/concpar21final01.md b/final/dry-run/concpar21final01/concpar21final01.md
new file mode 100644
index 0000000000000000000000000000000000000000..74b069af80945a10983dbad141c57213d574ce16
--- /dev/null
+++ b/final/dry-run/concpar21final01/concpar21final01.md
@@ -0,0 +1,55 @@
+# Problem 1: Futures
+
+## Setup
+
+Use the following commands to make a fresh clone of your repository:
+
+```
+git clone -b concpar21final01 git@gitlab.epfl.ch:lamp/student-repositories-s22/cs206-GASPAR.git concpar21final01
+```
+
+If you have issues with the IDE, try [reimporting the
+build](https://gitlab.epfl.ch/lamp/cs206/-/blob/master/labs/example-lab.md#troubleshooting),
+if you still have problems, use `compile` in sbt instead.
+
+## Useful links
+
+  * [The API documentation of the Scala standard library](https://www.scala-lang.org/files/archive/api/2.13.4)
+  * [The API documentation of the Java standard library](https://docs.oracle.com/en/java/javase/15/docs/api/index.html)
+  * [The API documentation of the Play framework](https://www.playframework.com/documentation/2.8.x/api/scala/index.html)
+
+## Exercise
+
+In this exercise, your task is to implement a leaderboard webpage for a GitLab-based exam. This leaderboard is live in the sense that it is constructed on the fly by extracting grades directly from GitLab pipelines: it is always up to date.
+
+This exercise uses the Play framework, a popular web application framework.
+Play is entirely asynchronous and uses futures for concurrency.
+
+In `src/main/scala/f1/MyComponents.scala`, we define a minimal Play application to display the leaderboard (you do not need to modify this file).
+
+You can start this application using `sbt run` and open the leaderboard in a web browser at [http://localhost:9000/](http://localhost:9000/). After having completed this exercise, you should see a populated leaderboard as shown in this screenshot:
+
+![leaderboard.png](leaderboard.png "leaderboard.png")
+
+In this exercise, your task is to implement the `leaderboard()` method in `src/main/scala/f1/F1.scala` which asynchronously retrieves and sorts student grades.
+Grades should be sorted such that maximum grades appear at the head of the list, like in the screenshot above.
+Your implementation should use the following two methods to get the list of students and the grade of a particular student:
+
+```scala
+/** Retrieve a student's grade using GitLab's API */
+def getGrade(sciper: Int): Future[Option[Grade]]
+
+/** Retrieve the list of enrolled students from IS-academia */
+def getScipers(): Future[List[Int]]
+```
+
+These methods have mock implementations that return made-up values after a short delay (simulating a network call).
+Your implementation should be asynchronous (it is forbidden to use the `Await.result` method).
+Furthermore, given the large number of students, calls to the GitLab API should be made in parallel such that the overall request is completed in about 1 second.
+
+*Hint:* this exercise can be solved without writing any recursive functions! You are allowed to use every function defined on [Future][1] and [List][2], as well as functions defined on their companion objects ([Future][3], [List][4]).
+
+[1]: https://www.scala-lang.org/api/2.13.4/scala/concurrent/Future.html
+[2]: https://www.scala-lang.org/api/2.13.4/scala/collection/immutable/List.html
+[3]: https://www.scala-lang.org/api/2.13.4/scala/concurrent/Future$.html
+[4]: https://www.scala-lang.org/api/2.13.4/scala/collection/immutable/List$.html
diff --git a/final/dry-run/concpar21final01/leaderboard.png b/final/dry-run/concpar21final01/leaderboard.png
new file mode 100644
index 0000000000000000000000000000000000000000..345da3a6dfb917967344ef7546d41755623336e3
Binary files /dev/null and b/final/dry-run/concpar21final01/leaderboard.png differ
diff --git a/final/dry-run/concpar21final02/concpar21final02.md b/final/dry-run/concpar21final02/concpar21final02.md
new file mode 100644
index 0000000000000000000000000000000000000000..f5498d50b390aaa351e5f3e8bf6544252d89b717
--- /dev/null
+++ b/final/dry-run/concpar21final02/concpar21final02.md
@@ -0,0 +1,85 @@
+# Problem 2: Actors
+
+## Setup
+
+Use the following commands to make a fresh clone of your repository:
+
+```
+git clone -b concpar21final02 git@gitlab.epfl.ch:lamp/student-repositories-s22/cs206-GASPAR.git concpar21final02
+```
+
+If you have issues with the IDE, try [reimporting the
+build](https://gitlab.epfl.ch/lamp/cs206/-/blob/master/labs/example-lab.md#troubleshooting),
+if you still have problems, use `compile` in sbt instead.
+
+## Useful links
+
+  * [A guide to the Scala parallel collections](https://docs.scala-lang.org/overviews/parallel-collections/overview.html)
+  * [The API documentation of the Scala parallel collections](https://www.javadoc.io/doc/org.scala-lang.modules/scala-parallel-collections_2.13/latest/scala/collection/index.html)
+  * [The API documentation of the Scala standard library](https://www.scala-lang.org/files/archive/api/2.13.4)
+  * [The API documentation of the Java standard library](https://docs.oracle.com/en/java/javase/15/docs/api/index.html)
+
+## Exercise
+
+Your task is to implement a couple of actors in Discord like service. You will only focus on two actors in the system: (1) the `NotificationService` and (2) and a `DiscordChannel`.
+
+A user will be able to post messages on the channel and retrieve the messages from the channel.
+A user also has the ability to (un)register from the notifications.
+Before it can receive requests, each channel is initialized with a notification service.
+When a message is posted to the channel, it tells its notification service to notify the users.
+
+The following diagram shows how the communication between actors works. Requests are within `()` and responses are within `[]`.
+
+```none
+           (Post)                                     [Active]
+       (GetLastPosts)       ┏━━━━━━━━━━━━━━━━━━┓  [AlreadyActive]  ┏━━━━━━━━━┓
+    ┌──────────────────────>┃                  ┃──────────────────►┃         ┃
+    │                       ┃  DiscordChannel  ┃                   ┃ System  ┃
+    │                       ┃                  ┃◄──────────────────┨         ┃
+    │          [Posts]      ┗━┯━━━━━━━━━━━━━━━┯┛      (Init)       ┗━━━━━━━━━┛
+ ┏━━┷━━━┓    [NotActive]      │               │
+ ┃      ┃◄────────────────────┘               │
+ ┃ User ┃                                     │
+ ┃      ┃◄────────────────┐                   │
+ ┗━━┯━━━┛  [Registered]   │                   │ (NotifyAll)
+    │     [Notification]  │                   │
+    │                     │                   │
+    │                ┏━━━━┷━━━━━━━━━━━━━━━━┓  │
+    └───────────────►┃                     ┃  │
+     (Register)      ┃ NotificationService ┃◄─┘
+    (UnRegister)     ┃                     ┃
+                     ┗━━━━━━━━━━━━━━━━━━━━━┛
+```
+
+Your tasks in the exercise will be to:
+
+TASK 1: Complete the implementation of the `NotificationService` by implementing method `receive` to handle messages from `NotificationService.Protocol`:
+
+```scala
+def receive: Receive = ???
+```
+TASK 2: Complete the implementation of the `DiscordChannel` by implementing methods `nonActive` and `active` to handle messages from `DiscordChannel.Protocol`:
+
+```scala
+def nonActive: Receive = ???
+def active(notificationService: ActorRef): Receive = ???
+```
+
+### NotificationService protocol
+
+* __Register__: Registers the user for notifications and responds with a `Registered(true)`.
+* __UnRegister__: Un-registers the user for notifications and responds with a `Registered(false)`.
+* __NotifyAll__: Sends a `Notification` to all registered users.
+
+
+### DiscordService protocol
+
+The channel can be in one of two states: _non-active_ or _active_.
+
+* __Init__: When _non-active_, responds with `Active` and the state becomes _active_. Otherwise responds `AlreadyActive`.
+* __Post__: When _active_, stores the message and sends `NotifyAll` to its `NotificationService`. Otherwise responds `NotActive`.
+* __GetLastPosts__: When _active_, responds with `Posts` containing the latest messages (ordered from most to least recent). Otherwise responds `NotActive`.
+
+### Running the code
+
+Apart from the grading tests, we include a `@main def debug` method, where you can write your own tests to help you debug your implementation. This method is __not__ graded. It can be executed using `run` within SBT.
diff --git a/final/dry-run/concpar21final03/concpar21final03.md b/final/dry-run/concpar21final03/concpar21final03.md
new file mode 100644
index 0000000000000000000000000000000000000000..0954e94b4dad3ea21afa3f5d27ab8c5dd209caa1
--- /dev/null
+++ b/final/dry-run/concpar21final03/concpar21final03.md
@@ -0,0 +1,161 @@
+# Problem 3: Concurrency
+
+## Setup
+
+Use the following commands to make a fresh clone of your repository:
+
+```
+git clone -b concpar21final03 git@gitlab.epfl.ch:lamp/student-repositories-s22/cs206-GASPAR.git concpar21final03
+```
+
+If you have issues with the IDE, try [reimporting the
+build](https://gitlab.epfl.ch/lamp/cs206/-/blob/master/labs/example-lab.md#troubleshooting),
+if you still have problems, use `compile` in sbt instead.
+
+## Useful links
+
+  * [The API documentation of the Scala standard library](https://www.scala-lang.org/files/archive/api/2.13.4)
+  * [The API documentation of the Java standard library](https://docs.oracle.com/en/java/javase/15/docs/api/index.html)
+
+## Problem description
+### Preliminary: File handling
+
+In this exercise, we will work with files using the following trait to handle
+all file operations:
+```scala
+type FileName = String
+trait FileSystem:
+  /** Create a new file named `file` with the passed `content`. */
+  def createFile(file: File, content: String): Unit
+  /** If `file` exists, return its content, otherwise crash. */
+  def readFile(file: File): String
+  /** If `file` exists, delete it, otherwise crash. */
+  def deleteFile(file: File): Unit
+```
+Note that to make testing easier, the actual implementation of `FileSystem` we will use
+won't actually create files on disks, instead it will simply use a `Map` to
+represent files and their content in memory, but that's an implementation detail
+that won't affect how this exercise should be solved.
+
+### The update distribution problem
+
+You work at a game company on the popular online game EPFNiteâ„¢. Your job is
+to distribute game updates to the players from the update server represented by
+the following class:
+```scala
+class UpdateServer:
+  def fetchUpdate(): Option[String]
+  def newUpdate(newName: FileName, newContent: String): Unit
+```
+The requirements of the update server are as follows:
+- When a player starts his game, it connects to the update server which starts a
+  new thread and run `fetchUpdate()` which should return the content of the latest game update
+  if one is available.
+- When a new version of the game is available, the developers call
+  `newUpdate` with the name of the update file and its content.
+- Storage space is limited on the server, so when a new update is stored on the
+  server, old ones must be deleted.
+
+Based on these requirements, you come up with the following implementation:
+
+```scala
+class UpdateServer(fs: FileSystem):
+  @volatile private var updateFile: Option[FileName] = None
+
+  def fetchUpdate(): Option[String] =
+    updateFile.map(fs.readFile)
+
+  def newUpdate(newName: FileName, newContent: String): Unit =
+    val oldFile = updateFile
+    fs.createFile(newName, newContent)
+    updateFile = Some(newName)
+    oldFile.foreach(fs.deleteFile)
+```
+Unfortunately, it turns out that reading a file is not an atomic operation: if
+you delete a file while another thread is reading it, your program crashes.
+Theoretically, you could solve this using locks to make sure `deleteFile`
+is never called at the same time as `readFile`, but this solution isn't good
+enough for EPFNite: it's important that a player is never blocked from playing
+the game because `fetchUpdate` is waiting for a lock to become available.
+
+Thankfully, there is one property of the problem which we can take advantage of:
+a call to `fetchUpdate` which happens *after* `updateFile = Some(newName)` will
+read the new file and not the old one, so all we need to do is to wait until all
+calls to `fetchUpdate` which were started *before* we mutated `updateFile` have
+finished before calling `deleteFile`. It turns out that there exists one
+mechanism to do this efficiently: **RCU** (Read-copy-update) which you will implement in the next
+section.
+
+## Implementation
+### Part 1: Complete the `ThreadMap` implementation
+
+To implement RCU we will need a thread-safe way to associate a value to a
+thread: this is the job of `ThreadMap` (defined in `ThreadMap.scala`) which we
+implement with a `Map` whose keys are instances of `Thread`
+(`Thread.currentThread` can be used to retrieve the instance for the current
+thread).
+
+Instead of the usual `forall` method on collections, `ThreadMap` has a
+`waitForall` method which will **block** until all entries of the map return
+true for the predicate. For example given `val m: ThreadMap[Int]`, if one thread
+runs:
+```scala
+m.setCurrentThread(1)
+```
+and another thread runs:
+```scala
+m.waitForall(_ < 0)
+```
+Then the second thread will be blocked, but if the first thread then runs:
+```scala
+m.setCurrentThread(-1)
+```
+The second thread will be immediately unblocked.
+
+Your first task is to **implement all methods in `ThreadMap.scala` whose body is
+currently `???`.** Once you're done, the "Part 1" test will pass.
+
+### Part 2: Complete the `RCU` implementation
+
+#### What is RCU ?
+
+The RCU API defines three methods:
+
+```scala
+class RCU:
+  def startRead(): Unit
+  def stopRead(): Unit
+  def waitForOldReads(): Unit
+```
+
+It has the following contract:
+- The first two methods are meant to be used by threads that read shared data:
+  they must must call `startRead` *before* reading shared data, then call
+  `stopRead` once they're done reading.
+- `waitForOldReaders()` can be called from any thread: this is a blocking method
+  that only returns when all reads started *before* the call to
+  `waitForOldReaders()` are stopped (new reads may have started since then).
+
+#### Implementing RCU
+
+To implement RCU we need a way to differentiate reads started before a call to
+`waitForOldReaders()` from those started after. To do so we will use *version
+numbers*: the RCU version (starting at 0) is stored in `latestVersion`, and for
+each *active* reader thread, we remember the value of `latestVersion` *at the time
+startRead() was called* in `readersVersion`. Implementing `waitForOldReaders` is
+then easy:
+1. Increment the RCU version.
+2. Wait until there's no active reader associated with a previous version.
+
+**Implement `waitForOldReaders` in `RCU.scala`**, for simplicity you can assume
+that `waitForOldReaders` will never be called from multiple threads at once.
+Once you're done, the "Part 2" test will pass.
+
+### Part 3: Using RCU in `UpdateServer`
+
+Finally, its time to put our RCU implementation to good use: **complete the
+implementation of `fetchUpdate` and `newUpdate`** (defined in `UpdateServer.scala`) by adding calls to
+`rcu.startRead()`, `rcu.stopRead()` and `rcu.waitForOldReaders()` where they
+need to be to allow multiple calls to `fetchUpdate` and at most one call to
+`newUpdate` to be run concurrently. Once you're done, the "Part 3" test will
+pass.