diff --git a/typo3/sysext/core/Classes/Routing/Enhancer/VariableProcessor.php b/typo3/sysext/core/Classes/Routing/Enhancer/VariableProcessor.php
index da75878605c2018c9e1baec4164a18c8b843bf57..308f4541a4e11bced978c7ae278c49f231601eec 100644
--- a/typo3/sysext/core/Classes/Routing/Enhancer/VariableProcessor.php
+++ b/typo3/sysext/core/Classes/Routing/Enhancer/VariableProcessor.php
@@ -23,7 +23,7 @@ class VariableProcessor
 {
     protected const LEVEL_DELIMITER = '__';
     protected const ARGUMENT_SEPARATOR = '/';
-    protected const VARIABLE_PATTERN = '#\{(?P<name>[^}]+)\}#';
+    protected const VARIABLE_PATTERN = '#\{(?P<modifier>!)?(?P<name>[^}]+)\}#';
 
     /**
      * @var array
@@ -41,10 +41,11 @@ class VariableProcessor
      */
     protected function addHash(string $value): string
     {
-        if (strlen($value) < 32 && !preg_match('#[^\w]#', $value)) {
+        if (strlen($value) < 31 && !preg_match('#[^\w]#', $value)) {
             return $value;
         }
-        $hash = md5($value);
+        // removing one bit, e.g. for enforced route prefix `{!value}`
+        $hash = substr(md5($value), 0, -1);
         // Symfony Route Compiler requires first literal to be non-integer
         if ($hash[0] === (string)(int)$hash[0]) {
             $hash[0] = str_replace(
@@ -64,7 +65,7 @@ class VariableProcessor
      */
     protected function resolveHash(string $hash): string
     {
-        if (strlen($hash) < 32) {
+        if (strlen($hash) < 31) {
             return $hash;
         }
         if (!isset($this->hashes[$hash])) {
@@ -118,14 +119,13 @@ class VariableProcessor
             return $routePath;
         }
 
+        $replace = [];
         $search = array_values($matches[0]);
-        $replace = array_map(
-            function (string $name) {
-                return '{' . $name . '}';
-            },
-            $this->deflateValues($matches['name'], $namespace, $arguments)
-        );
-
+        $deflatedNames = $this->deflateValues($matches['name'], $namespace, $arguments);
+        foreach ($deflatedNames as $index => $deflatedName) {
+            $modifier = $matches['modifier'][$index] ?? '';
+            $replace[] = '{' . $modifier . $deflatedName . '}';
+        }
         return str_replace($search, $replace, $routePath);
     }
 
@@ -141,14 +141,13 @@ class VariableProcessor
             return $routePath;
         }
 
+        $replace = [];
         $search = array_values($matches[0]);
-        $replace = array_map(
-            function (string $name) {
-                return '{' . $name . '}';
-            },
-            $this->inflateValues($matches['name'], $namespace, $arguments)
-        );
-
+        $inflatedNames = $this->inflateValues($matches['name'], $namespace, $arguments);
+        foreach ($inflatedNames as $index => $inflatedName) {
+            $modifier = $matches['modifier'][$index] ?? '';
+            $replace[] = '{' . $modifier . $inflatedName . '}';
+        }
         return str_replace($search, $replace, $routePath);
     }
 
diff --git a/typo3/sysext/core/Tests/Unit/Routing/Enhancer/VariableProcessorTest.php b/typo3/sysext/core/Tests/Unit/Routing/Enhancer/VariableProcessorTest.php
index 69344431a97ae1032adb13364450ad85bc95d3f8..34ca8ed366d5f6dc7c0cf181beb186d3244a8b25 100644
--- a/typo3/sysext/core/Tests/Unit/Routing/Enhancer/VariableProcessorTest.php
+++ b/typo3/sysext/core/Tests/Unit/Routing/Enhancer/VariableProcessorTest.php
@@ -40,41 +40,93 @@ class VariableProcessorTest extends UnitTestCase
 
     public function routePathDataProvider(): array
     {
+        $plainInflatedRoutePath = '/static/{aa}/{bb}/{some_cc}/tail';
+        $enforcedInflatedRoutePath = '/static/{!aa}/{bb}/{some_cc}/tail';
+
         return [
-            'no arguments, no namespace' => [
+            'no arguments, no namespace (plain)' => [
                 null,
                 [],
+                $plainInflatedRoutePath,
                 '/static/{aa}/{bb}/{some_cc}/tail'
             ],
-            'aa -> zz, no namespace' => [
+            'no arguments, no namespace (enforced)' => [
+                null,
+                [],
+                $enforcedInflatedRoutePath,
+                '/static/{!aa}/{bb}/{some_cc}/tail'
+            ],
+            'aa -> zz, no namespace (plain)' => [
                 null,
                 ['aa' => 'zz'],
+                $plainInflatedRoutePath,
                 '/static/{zz}/{bb}/{some_cc}/tail'
             ],
-            'aa -> @any/nested, no namespace' => [
+            'aa -> zz, no namespace (enforced)' => [
+                null,
+                ['aa' => 'zz'],
+                $enforcedInflatedRoutePath,
+                '/static/{!zz}/{bb}/{some_cc}/tail'
+            ],
+            'aa -> @any/nested, no namespace (plain)' => [
                 null,
                 ['aa' => '@any/nested'],
-                '/static/{qbeced67e6b340abc67a397f6e90bb0e}/{bb}/{some_cc}/tail'
+                $plainInflatedRoutePath,
+                '/static/{qbeced67e6b340abc67a397f6e90bb0}/{bb}/{some_cc}/tail'
             ],
-            'no arguments, first' => [
+            'aa -> @any/nested, no namespace (enforced)' => [
+                null,
+                ['aa' => '@any/nested'],
+                $enforcedInflatedRoutePath,
+                '/static/{!qbeced67e6b340abc67a397f6e90bb0}/{bb}/{some_cc}/tail'
+            ],
+            'no arguments, first (plain)' => [
                 'first',
                 [],
+                $plainInflatedRoutePath,
                 '/static/{first__aa}/{first__bb}/{first__some_cc}/tail'
             ],
-            'aa -> zz, first' => [
+            'no arguments, first (enforced)' => [
+                'first',
+                [],
+                $enforcedInflatedRoutePath,
+                '/static/{!first__aa}/{first__bb}/{first__some_cc}/tail'
+            ],
+            'aa -> zz, first (plain)' => [
                 'first',
                 ['aa' => 'zz'],
+                $plainInflatedRoutePath,
                 '/static/{first__zz}/{first__bb}/{first__some_cc}/tail'
             ],
-            'aa -> any/nested, first' => [
+            'aa -> zz, first (enforced)' => [
+                'first',
+                ['aa' => 'zz'],
+                $enforcedInflatedRoutePath,
+                '/static/{!first__zz}/{first__bb}/{first__some_cc}/tail'
+            ],
+            'aa -> any/nested, first (plain)' => [
                 'first',
                 ['aa' => 'any/nested'],
+                $plainInflatedRoutePath,
                 '/static/{first__any__nested}/{first__bb}/{first__some_cc}/tail'
             ],
-            'aa -> @any/nested, first' => [
+            'aa -> any/nested, first (enforced)' => [
+                'first',
+                ['aa' => 'any/nested'],
+                $enforcedInflatedRoutePath,
+                '/static/{!first__any__nested}/{first__bb}/{first__some_cc}/tail'
+            ],
+            'aa -> @any/nested, first (plain)' => [
+                'first',
+                ['aa' => '@any/nested'],
+                $plainInflatedRoutePath,
+                '/static/{ab0ce8f9f822228b4f324ec38b9c038}/{first__bb}/{first__some_cc}/tail'
+            ],
+            'aa -> @any/nested, first (enforced)' => [
                 'first',
                 ['aa' => '@any/nested'],
-                '/static/{ab0ce8f9f822228b4f324ec38b9c0388}/{first__bb}/{first__some_cc}/tail'
+                $enforcedInflatedRoutePath,
+                '/static/{!ab0ce8f9f822228b4f324ec38b9c038}/{first__bb}/{first__some_cc}/tail'
             ],
         ];
     }
@@ -82,14 +134,14 @@ class VariableProcessorTest extends UnitTestCase
     /**
      * @param string|null $namespace
      * @param array $arguments
+     * @param string $inflatedRoutePath
      * @param string $deflatedRoutePath
      *
      * @test
      * @dataProvider routePathDataProvider
      */
-    public function isRoutePathProcessed(?string $namespace, array $arguments, string $deflatedRoutePath)
+    public function isRoutePathProcessed(?string $namespace, array $arguments, string $inflatedRoutePath, string $deflatedRoutePath)
     {
-        $inflatedRoutePath = '/static/{aa}/{bb}/{some_cc}/tail';
         self::assertSame(
             $deflatedRoutePath,
             $this->subject->deflateRoutePath($inflatedRoutePath, $namespace, $arguments)
@@ -108,15 +160,15 @@ class VariableProcessorTest extends UnitTestCase
         return [
             'no namespace, no arguments' => [
                 [],
-                ['a' => 'a', 'first__aa' => 'aa', 'first__second__aaa' => 'aaa', 'a9d66412d169b85537e11d9e49b75f9b' => '@any']
+                ['a' => 'a', 'first__aa' => 'aa', 'first__second__aaa' => 'aaa', 'a9d66412d169b85537e11d9e49b75f9' => '@any']
             ],
             'no namespace, a -> newA' => [
                 ['a' => 'newA'],
-                ['newA' => 'a', 'first__aa' => 'aa', 'first__second__aaa' => 'aaa', 'a9d66412d169b85537e11d9e49b75f9b' => '@any']
+                ['newA' => 'a', 'first__aa' => 'aa', 'first__second__aaa' => 'aaa', 'a9d66412d169b85537e11d9e49b75f9' => '@any']
             ],
             'no namespace, a -> @any/nested' => [
                 ['a' => '@any/nested'],
-                ['qbeced67e6b340abc67a397f6e90bb0e' => 'a', 'first__aa' => 'aa', 'first__second__aaa' => 'aaa', 'a9d66412d169b85537e11d9e49b75f9b' => '@any']
+                ['qbeced67e6b340abc67a397f6e90bb0' => 'a', 'first__aa' => 'aa', 'first__second__aaa' => 'aaa', 'a9d66412d169b85537e11d9e49b75f9' => '@any']
             ],
         ];
     }
@@ -167,32 +219,32 @@ class VariableProcessorTest extends UnitTestCase
             'first, no arguments' => [
                 'first',
                 [],
-                ['a' => 'a', 'first__aa' => 'aa', 'first__second__aaa' => 'aaa', 'a9d66412d169b85537e11d9e49b75f9b' => '@any']
+                ['a' => 'a', 'first__aa' => 'aa', 'first__second__aaa' => 'aaa', 'a9d66412d169b85537e11d9e49b75f9' => '@any']
             ],
             'first, aa -> newAA' => [
                 'first',
                 ['aa' => 'newAA'],
-                ['a' => 'a', 'first__newAA' => 'aa', 'first__second__aaa' => 'aaa', 'a9d66412d169b85537e11d9e49b75f9b' => '@any']
+                ['a' => 'a', 'first__newAA' => 'aa', 'first__second__aaa' => 'aaa', 'a9d66412d169b85537e11d9e49b75f9' => '@any']
             ],
             'first, second -> newSecond' => [
                 'first',
                 ['second' => 'newSecond'],
-                ['a' => 'a', 'first__aa' => 'aa', 'first__newSecond__aaa' => 'aaa', 'q7aded81f5d1607191c695720db7ab23' => '@any']
+                ['a' => 'a', 'first__aa' => 'aa', 'first__newSecond__aaa' => 'aaa', 'q7aded81f5d1607191c695720db7ab2' => '@any']
             ],
             'first, aa -> any/nested' => [
                 'first',
                 ['aa' => 'any/nested'],
-                ['a' => 'a', 'first__any__nested' => 'aa', 'first__second__aaa' => 'aaa', 'a9d66412d169b85537e11d9e49b75f9b' => '@any']
+                ['a' => 'a', 'first__any__nested' => 'aa', 'first__second__aaa' => 'aaa', 'a9d66412d169b85537e11d9e49b75f9' => '@any']
             ],
             'first, aa -> @any/nested' => [
                 'first',
                 ['aa' => '@any/nested'],
-                ['a' => 'a', 'ab0ce8f9f822228b4f324ec38b9c0388' => 'aa', 'first__second__aaa' => 'aaa', 'a9d66412d169b85537e11d9e49b75f9b' => '@any']
+                ['a' => 'a', 'ab0ce8f9f822228b4f324ec38b9c038' => 'aa', 'first__second__aaa' => 'aaa', 'a9d66412d169b85537e11d9e49b75f9' => '@any']
             ],
             'first, aa -> newAA, second => newSecond' => [
                 'first',
                 ['aa' => 'newAA', 'second' => 'newSecond'],
-                ['a' => 'a', 'first__newAA' => 'aa', 'first__newSecond__aaa' => 'aaa', 'q7aded81f5d1607191c695720db7ab23' => '@any']
+                ['a' => 'a', 'first__newAA' => 'aa', 'first__newSecond__aaa' => 'aaa', 'q7aded81f5d1607191c695720db7ab2' => '@any']
             ],
         ];
     }
@@ -268,17 +320,17 @@ class VariableProcessorTest extends UnitTestCase
             'a -> @any/nested, no namespace' => [
                 null,
                 ['a' => '@any/nested'],
-                ['qbeced67e6b340abc67a397f6e90bb0e' => 'a', 'b' => 'b', 'c' => ['d' => 'd', 'e' => 'e']]
+                ['qbeced67e6b340abc67a397f6e90bb0' => 'a', 'b' => 'b', 'c' => ['d' => 'd', 'e' => 'e']]
             ],
             'a -> newA, namespace_being_longer_than_32_characters' => [
                 'namespace_being_longer_than_32_characters',
                 ['a' => 'newA'],
-                ['qaea1f31c57b9c3e78c8205838d4563c' => 'a', 'ub5e2989b61a4964ba4e06fc6de85276' => 'b', 'oabf16f448f7b02c6ecb13d155e5a4b5' => ['d' => 'd', 'e' => 'e']]
+                ['qaea1f31c57b9c3e78c8205838d4563' => 'a', 'ub5e2989b61a4964ba4e06fc6de8527' => 'b', 'oabf16f448f7b02c6ecb13d155e5a4b' => ['d' => 'd', 'e' => 'e']]
             ],
             'a -> @any/nested, first' => [
                 'first',
                 ['a' => '@any/nested'],
-                ['ab0ce8f9f822228b4f324ec38b9c0388' => 'a', 'first__b' => 'b', 'first__c' => ['d' => 'd', 'e' => 'e']]
+                ['ab0ce8f9f822228b4f324ec38b9c038' => 'a', 'first__b' => 'b', 'first__c' => ['d' => 'd', 'e' => 'e']]
             ],
         ];
     }
diff --git a/typo3/sysext/frontend/Tests/Functional/SiteHandling/EnhancerSiteRequestTest.php b/typo3/sysext/frontend/Tests/Functional/SiteHandling/EnhancerSiteRequestTest.php
index fb84834c973767a2290d05f5ca143a3edd796ffe..1b2892cdf4ca0fdf15594c25554c737474495a3f 100644
--- a/typo3/sysext/frontend/Tests/Functional/SiteHandling/EnhancerSiteRequestTest.php
+++ b/typo3/sysext/frontend/Tests/Functional/SiteHandling/EnhancerSiteRequestTest.php
@@ -661,6 +661,11 @@ class EnhancerSiteRequestTest extends AbstractTestCase
                 'value',
                 'test',
             ],
+            'namespace[value] ? x^30' => [
+                'namespace',
+                'value',
+                str_repeat('x', 30),
+            ],
             'namespace[value] ? x^31' => [
                 'namespace',
                 'value',
@@ -676,10 +681,10 @@ class EnhancerSiteRequestTest extends AbstractTestCase
                 'value',
                 str_repeat('x', 33),
             ],
-            'namespace[value] ? 1^32 (type-cast)' => [
+            'namespace[value] ? 1^31 (type-cast)' => [
                 'namespace',
                 'value',
-                str_repeat('1', 32),
+                str_repeat('1', 31),
             ],
             // md5('namespace__@otne3') is 60360798585102000952995164024754 (numeric)
             // md5('ximaz') is 61529519452809720693702583126814 (numeric)
@@ -710,6 +715,11 @@ class EnhancerSiteRequestTest extends AbstractTestCase
                 'namespace/other',
             ],
             // namespace[any/value]
+            'namespace[any/value] ? x^30' => [
+                'namespace',
+                'any/value',
+                str_repeat('x', 30),
+            ],
             'namespace[any/value] ? x^31' => [
                 'namespace',
                 'any/value',
@@ -742,6 +752,11 @@ class EnhancerSiteRequestTest extends AbstractTestCase
                 'namespace/any/other',
             ],
             // namespace[@any/value]
+            'namespace[@any/value] ? x^30' => [
+                'namespace',
+                '@any/value',
+                str_repeat('x', 30),
+            ],
             'namespace[@any/value] ? x^31' => [
                 'namespace',
                 '@any/value',