--- /dev/null
+/* SPDX-License-Identifier: Apache-2.0
+ * Copyright(c) 2025 Cisco Systems, Inc.
+ */
+
+#include <http/http3/qpack.h>
+#include <http/http2/hpack.h>
+
+typedef struct
+{
+  char *name;
+  uword name_len;
+  char *value;
+  uword value_len;
+} qpack_static_table_entry_t;
+
+#define name_val_token_lit(name, value)                                       \
+  (name), sizeof (name) - 1, (value), sizeof (value) - 1
+
+/* RFC9204 Appendix A */
+static qpack_static_table_entry_t qpack_static_table[] = {
+  { name_val_token_lit (":authority", "") },
+  { name_val_token_lit (":path", "/") },
+  { name_val_token_lit ("age", "0") },
+  { name_val_token_lit ("content-disposition", "") },
+  { name_val_token_lit ("content-length", "0") },
+  { name_val_token_lit ("cookie", "") },
+  { name_val_token_lit ("date", "") },
+  { name_val_token_lit ("etag", "") },
+  { name_val_token_lit ("if-modified-since", "") },
+  { name_val_token_lit ("if-none-match", "") },
+  { name_val_token_lit ("last-modified", "") },
+  { name_val_token_lit ("link", "") },
+  { name_val_token_lit ("location", "") },
+  { name_val_token_lit ("referer", "") },
+  { name_val_token_lit ("set-cookie", "") },
+  { name_val_token_lit (":method", "CONNECT") },
+  { name_val_token_lit (":metho", "DELETE") },
+  { name_val_token_lit (":method", "GET") },
+  { name_val_token_lit (":method", "HEAD") },
+  { name_val_token_lit (":method", "OPTIONS") },
+  { name_val_token_lit (":method", "POST") },
+  { name_val_token_lit (":method", "PUT") },
+  { name_val_token_lit (":scheme", "http") },
+  { name_val_token_lit (":scheme", "https") },
+  { name_val_token_lit (":status", "103") },
+  { name_val_token_lit (":status", "200") },
+  { name_val_token_lit (":status", "304") },
+  { name_val_token_lit (":status", "404") },
+  { name_val_token_lit (":status", "503") },
+  { name_val_token_lit ("accept", "*/*") },
+  { name_val_token_lit ("accept", "application/dns-message") },
+  { name_val_token_lit ("accept-encoding", "gzip, deflate, br") },
+  { name_val_token_lit ("accept-ranges", "bytes") },
+  { name_val_token_lit ("access-control-allow-headers", "cache-control") },
+  { name_val_token_lit ("access-control-allow-headers", "content-type") },
+  { name_val_token_lit ("access-control-allow-origin", "*") },
+  { name_val_token_lit ("cache-control", "max-age=0") },
+  { name_val_token_lit ("cache-control", "max-age=2592000") },
+  { name_val_token_lit ("cache-control", "max-age=604800") },
+  { name_val_token_lit ("cache-control", "no-cache") },
+  { name_val_token_lit ("cache-control", "no-store") },
+  { name_val_token_lit ("cache-control", "public, max-age=31536000") },
+  { name_val_token_lit ("content-encoding      ", "r") },
+  { name_val_token_lit ("content-encoding", "gzip") },
+  { name_val_token_lit ("content-type", "application/dns-message") },
+  { name_val_token_lit ("content-type", "application/javascript") },
+  { name_val_token_lit ("content-type", "application/json") },
+  { name_val_token_lit ("content-type", "application/x-www-form-urlencoded") },
+  { name_val_token_lit ("content-type", "image/gif") },
+  { name_val_token_lit ("content-type", "image/jpeg") },
+  { name_val_token_lit ("content-type", "image/png") },
+  { name_val_token_lit ("content-type", "text/css") },
+  { name_val_token_lit ("content-type", "text/html; charset=utf-8") },
+  { name_val_token_lit ("content-type", "text/plain") },
+  { name_val_token_lit ("content-type", "text/plain;charset=utf-8") },
+  { name_val_token_lit ("range", "bytes=0-") },
+  { name_val_token_lit ("strict-transport-security", "max-age=31536000") },
+  { name_val_token_lit ("strict-transport-security",
+                       "max-age=31536000; includesubdomains") },
+  { name_val_token_lit ("strict-transport-security",
+                       "max-age=31536000; includesubdomains; preload") },
+  { name_val_token_lit ("vary", "accept-encoding") },
+  { name_val_token_lit ("vary", "origin") },
+  { name_val_token_lit ("x-content-type-options", "nosniff") },
+  { name_val_token_lit ("x-xss-protection", "1; mode=block") },
+  { name_val_token_lit (":status", "100") },
+  { name_val_token_lit (":status", "204") },
+  { name_val_token_lit (":status", "206") },
+  { name_val_token_lit (":status", "302") },
+  { name_val_token_lit (":status", "400") },
+  { name_val_token_lit (":status", "403") },
+  { name_val_token_lit (":status", "421") },
+  { name_val_token_lit (":status", "425") },
+  { name_val_token_lit (":status", "500") },
+  { name_val_token_lit ("accept-language", "") },
+  { name_val_token_lit ("access-control-allow-credentials", "FALSE") },
+  { name_val_token_lit ("access-control-allow-credentials", "TRUE") },
+  { name_val_token_lit ("access-control-allow-headers", "*") },
+  { name_val_token_lit ("access-control-allow-methods", "get") },
+  { name_val_token_lit ("access-control-allow-methods",
+                       "get, post, options") },
+  { name_val_token_lit ("access-control-allow-methods", "options") },
+  { name_val_token_lit ("access-control-expose-headers", "content-length") },
+  { name_val_token_lit ("access-control-request-headers", "content-type") },
+  { name_val_token_lit ("access-control-request-method", "get") },
+  { name_val_token_lit ("access-control-request-method", "post") },
+  { name_val_token_lit ("alt-svc", "clear") },
+  { name_val_token_lit ("authorization", "") },
+  { name_val_token_lit (
+    "content-security-policy",
+    "script-src 'none'; object-src 'none'; base-uri 'none'") },
+  { name_val_token_lit ("early-data", "1") },
+  { name_val_token_lit ("expect-ct", "") },
+  { name_val_token_lit ("forwarded", "") },
+  { name_val_token_lit ("if-range", "") },
+  { name_val_token_lit ("origin", "") },
+  { name_val_token_lit ("purpose", "prefetch") },
+  { name_val_token_lit ("server", "") },
+  { name_val_token_lit ("timing-allow-origin", "*") },
+  { name_val_token_lit ("upgrade-insecure-requests", "1") },
+  { name_val_token_lit ("user-agent", "") },
+  { name_val_token_lit ("x-forwarded-for", "") },
+  { name_val_token_lit ("x-frame-options", "deny") },
+  { name_val_token_lit ("x-frame-options", "sameorigin") },
+};
+
+#define QPACK_STATIC_TABLE_SIZE                                               \
+  (sizeof (qpack_static_table) / sizeof (qpack_static_table[0]))
+
+STATIC_ASSERT (QPACK_STATIC_TABLE_SIZE == 99,
+              "static table must have 99 entries");
+
+static http3_error_t
+qpack_get_static_table_entry (uword index, http_token_t *name,
+                             http_token_t *value, u8 value_is_indexed)
+{
+  if (index >= QPACK_STATIC_TABLE_SIZE)
+    return HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
+
+  qpack_static_table_entry_t *e = &qpack_static_table[index];
+  name->base = e->name;
+  name->len = e->name_len;
+  if (value_is_indexed)
+    {
+      if (e->value_len == 0)
+       return HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
+      value->base = e->value;
+      value->len = e->value_len;
+    }
+
+  return HTTP3_ERROR_NO_ERROR;
+}
+
+static http3_error_t
+qpack_decode_string (u8 **src, u8 *end, u8 **buf, uword *buf_len,
+                    u8 prefix_len)
+{
+  u8 *p, is_huffman;
+  uword len;
+  int rv;
+
+  ASSERT (prefix_len >= 2 && prefix_len <= 8);
+  if (*src == end)
+    return HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
+
+  p = *src;
+  /* first bit for H flag */
+  is_huffman = (*p >> (prefix_len - 1)) & 0x01;
+
+  /* length is integer with (N-1) bit prefix */
+  len = hpack_decode_int (&p, end, prefix_len - 1);
+  if (PREDICT_FALSE (len == HPACK_INVALID_INT))
+    return HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
+
+  /* do we have everything? */
+  if (len > (end - p))
+    return HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
+
+  if (is_huffman)
+    {
+      *src = (p + len);
+      rv = hpack_decode_huffman (&p, p + len, buf, buf_len);
+      return rv ? HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED :
+                 HTTP3_ERROR_NO_ERROR;
+    }
+  else
+    {
+      /* enough space? */
+      if (len > *buf_len)
+       return HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
+
+      clib_memcpy (*buf, p, len);
+      *buf_len -= len;
+      *buf += len;
+      *src = (p + len);
+      return HTTP3_ERROR_NO_ERROR;
+    }
+}
+
+__clib_export http3_error_t
+qpack_decode_header (u8 **src, u8 *end, u8 **buf, uword *buf_len,
+                    u32 *name_len, u32 *value_len, void *decoder_ctx)
+{
+  u8 *p;
+  uword index, old_len;
+  http_token_t name, value;
+  http3_error_t rv;
+
+  ASSERT (*src < end);
+  p = *src;
+
+#define COPY_TOKEN(_token)                                                    \
+  if (_token.len > *buf_len)                                                  \
+    return HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;                            \
+  clib_memcpy (*buf, _token.base, _token.len);                                \
+  *buf_len -= _token.len;                                                     \
+  *buf += _token.len;
+
+  switch (*p >> 4)
+    {
+    case 12 ... 15:
+      /* indexed field line, static table */
+      index = hpack_decode_int (&p, end, 6);
+      if (index == HPACK_INVALID_INT)
+       return HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
+      rv = qpack_get_static_table_entry (index, &name, &value, 1);
+      if (rv != HTTP3_ERROR_NO_ERROR)
+       return rv;
+      COPY_TOKEN (name);
+      *name_len = name.len;
+      COPY_TOKEN (value);
+      *value_len = value.len;
+      break;
+    case 8 ... 11:
+      /* TODO: indexed field line, dynamic table */
+      return HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
+    case 7:
+    case 5:
+      /* literal field line with name reference, static table */
+      index = hpack_decode_int (&p, end, 4);
+      if (index == HPACK_INVALID_INT)
+       return HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
+      rv = qpack_get_static_table_entry (index, &name, &value, 0);
+      if (rv != HTTP3_ERROR_NO_ERROR)
+       return rv;
+      COPY_TOKEN (name);
+      *name_len = name.len;
+      old_len = *buf_len;
+      rv = qpack_decode_string (&p, end, buf, buf_len, 8);
+      if (rv != HTTP3_ERROR_NO_ERROR)
+       return rv;
+      *value_len = old_len - *buf_len;
+      break;
+    case 6:
+    case 4:
+      /* TODO: literal field line with name reference, dynamic table */
+      return HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
+    case 3:
+    case 2:
+      /* literal field line with literal name */
+      old_len = *buf_len;
+      rv = qpack_decode_string (&p, end, buf, buf_len, 4);
+      if (rv != HTTP3_ERROR_NO_ERROR)
+       return rv;
+      *name_len = old_len - *buf_len;
+      old_len = *buf_len;
+      rv = qpack_decode_string (&p, end, buf, buf_len, 8);
+      if (rv != HTTP3_ERROR_NO_ERROR)
+       return rv;
+      *value_len = old_len - *buf_len;
+      break;
+    case 1:
+      /* TODO: indexed field line with post-base index */
+      return HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
+    case 0:
+      /* TODO: literal field line with post-base name reference */
+      return HTTP3_ERROR_QPACK_DECOMPRESSION_FAILED;
+    default:
+      ASSERT (0);
+      break;
+    }
+
+  *src = p;
+  return HTTP3_ERROR_NO_ERROR;
+}
 
 #include <http/http_header_names.h>
 #include <http/http2/hpack.h>
 #include <http/http2/frame.h>
+#include <http/http3/qpack.h>
 
 #define HTTP_TEST_I(_cond, _comment, _args...)                                \
   ({                                                                          \
   return 0;
 }
 
+static int
+http_test_qpack (vlib_main_t *vm)
+{
+  vlib_cli_output (vm, "qpack_decode_header");
+
+  static http3_error_t (*_qpack_decode_header) (
+    u8 * *src, u8 * end, u8 * *buf, uword * buf_len, u32 * name_len,
+    u32 * value_len);
+
+  _qpack_decode_header =
+    vlib_get_plugin_symbol ("http_plugin.so", "qpack_decode_header");
+
+  u8 *pos, *bp, *buf = 0, *input = 0;
+  uword blen;
+  u32 name_len, value_len;
+  http3_error_t rv;
+
+#define TEST(i, e_name, e_value)                                              \
+  vec_validate (input, sizeof (i) - 2);                                       \
+  memcpy (input, i, sizeof (i) - 1);                                          \
+  pos = input;                                                                \
+  vec_validate_init_empty (buf, 63, 0);                                       \
+  bp = buf;                                                                   \
+  blen = vec_len (buf);                                                       \
+  rv = _qpack_decode_header (&pos, vec_end (input), &bp, &blen, &name_len,    \
+                            &value_len);                                     \
+  HTTP_TEST ((rv == HTTP3_ERROR_NO_ERROR && name_len == strlen (e_name) &&    \
+             value_len == strlen (e_value) &&                                \
+             !memcmp (buf, e_name, name_len) &&                              \
+             !memcmp (buf + name_len, e_value, value_len) &&                 \
+             vec_len (buf) == (blen + name_len + value_len) &&               \
+             pos == vec_end (input) && bp == buf + name_len + value_len),    \
+            "%U is decoded as '%U: %U'", format_hex_bytes, input,            \
+            vec_len (input), format_http_bytes, buf, name_len,               \
+            format_http_bytes, buf + name_len, value_len);                   \
+  vec_free (input);                                                           \
+  vec_free (buf);
+
+  /* literal field line with name reference, static table */
+  TEST ("\x51\x0B\x2F\x69\x6E\x64\x65\x78\x2E\x68\x74\x6D\x6C", ":path",
+       "/index.html");
+  /* indexed field line, static table */
+  TEST ("\xC1", ":path", "/");
+  /* literal field line with literal name */
+  TEST ("\x23\x61\x62\x63\x01\x5A", "abc", "Z");
+
+  return 0;
+}
+
 static clib_error_t *
 test_http_command_fn (vlib_main_t *vm, unformat_input_t *input,
                      vlib_cli_command_t *cmd)
        res = http_test_hpack (vm);
       else if (unformat (input, "h2-frame"))
        res = http_test_h2_frame (vm);
+      else if (unformat (input, "qpack"))
+       res = http_test_qpack (vm);
       else if (unformat (input, "all"))
        {
          if ((res = http_test_parse_authority (vm)))
            goto done;
          if ((res = http_test_h2_frame (vm)))
            goto done;
+         if ((res = http_test_qpack (vm)))
+           goto done;
        }
       else
        break;