From mboxrd@z Thu Jan 1 00:00:00 1970 Return-Path: X-Spam-Checker-Version: SpamAssassin 3.4.0 (2014-02-07) on aws-us-west-2-korg-lkml-1.web.codeaurora.org Received: from kanga.kvack.org (kanga.kvack.org [205.233.56.17]) by smtp.lore.kernel.org (Postfix) with ESMTP id 74DA1C3ABC3 for ; Fri, 9 May 2025 15:30:41 +0000 (UTC) Received: by kanga.kvack.org (Postfix) id A7D526B0134; Fri, 9 May 2025 11:30:40 -0400 (EDT) Received: by kanga.kvack.org (Postfix, from userid 40) id A29F66B0137; Fri, 9 May 2025 11:30:40 -0400 (EDT) X-Delivered-To: int-list-linux-mm@kvack.org Received: by kanga.kvack.org (Postfix, from userid 63042) id 8CD20280009; Fri, 9 May 2025 11:30:40 -0400 (EDT) X-Delivered-To: linux-mm@kvack.org Received: from relay.hostedemail.com (smtprelay0016.hostedemail.com [216.40.44.16]) by kanga.kvack.org (Postfix) with ESMTP id 6F0B26B0134 for ; Fri, 9 May 2025 11:30:40 -0400 (EDT) Received: from smtpin08.hostedemail.com (a10.router.float.18 [10.200.18.1]) by unirelay06.hostedemail.com (Postfix) with ESMTP id 58BAEBED1C for ; Fri, 9 May 2025 15:30:40 +0000 (UTC) X-FDA: 83423756640.08.F16B1D9 Received: from us-smtp-delivery-124.mimecast.com (us-smtp-delivery-124.mimecast.com [170.10.129.124]) by imf01.hostedemail.com (Postfix) with ESMTP id 258D64000A for ; Fri, 9 May 2025 15:30:37 +0000 (UTC) Authentication-Results: imf01.hostedemail.com; dkim=pass header.d=redhat.com header.s=mimecast20190719 header.b=X+w5Frdh; dmarc=pass (policy=quarantine) header.from=redhat.com; spf=pass (imf01.hostedemail.com: domain of dhildenb@redhat.com designates 170.10.129.124 as permitted sender) smtp.mailfrom=dhildenb@redhat.com ARC-Message-Signature: i=1; a=rsa-sha256; c=relaxed/relaxed; d=hostedemail.com; s=arc-20220608; t=1746804638; h=from:from:sender:reply-to:subject:subject:date:date: message-id:message-id:to:to:cc:cc:mime-version:mime-version: content-type:content-type: content-transfer-encoding:content-transfer-encoding:in-reply-to: references:dkim-signature; bh=/QETQBLLgywJiVkdBX35qjwYFT1cFvifg3en6u8zZzc=; b=UIJwEYg5Ygyapf9DO1XG+lTF0HIFL/ZBcovydsDQ1/EdO6YxvqGlF6iG6wSCOWhxOINXvT eGj3AeGqvAMWOtYtjpno6oKcUsUOXvEBciEmOzQrV62RG5cUtyMgxgHHne21CoMujsuePU CNUjM7sBpqqwMHtvEIP/xbXDF3tsl5E= ARC-Seal: i=1; s=arc-20220608; d=hostedemail.com; t=1746804638; a=rsa-sha256; cv=none; b=SHSw9+D+8dCdp10vOTHRHTROQXN3tDbhGw7yU9MrjLWiFvrt/vRiW8cUj/7GZkCH2Oep76 /YLSJ84N6g+rWu0zqbg8f/PQPT23jb9Ilym0ivpsj2vJgY2UYPveHErOq/y6zwlS7XHbeY hpfPKgdqpsVLbshy0n063YGT2U9AYGc= ARC-Authentication-Results: i=1; imf01.hostedemail.com; dkim=pass header.d=redhat.com header.s=mimecast20190719 header.b=X+w5Frdh; dmarc=pass (policy=quarantine) header.from=redhat.com; spf=pass (imf01.hostedemail.com: domain of dhildenb@redhat.com designates 170.10.129.124 as permitted sender) smtp.mailfrom=dhildenb@redhat.com DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=redhat.com; s=mimecast20190719; t=1746804637; h=from:from:reply-to:subject:subject:date:date:message-id:message-id: to:to:cc:cc:mime-version:mime-version:content-type:content-type: content-transfer-encoding:content-transfer-encoding; bh=/QETQBLLgywJiVkdBX35qjwYFT1cFvifg3en6u8zZzc=; b=X+w5FrdhtCOIQkKgCgX89Sti7/cxzl6/3ZjLcHXE9+IGkMrWAYsO9JHsSbLykhXa95siU8 R/5O2+7W2NGUkWtgetfR4VzhD5dccKBjsYnxnVL6sJ0oHcG7RA+9PDGxtPUy4fVgcuExIa wdmoP8Vc//KlORkkLNQVSWbVx5eCJnI= Received: from mail-wm1-f72.google.com (mail-wm1-f72.google.com [209.85.128.72]) by relay.mimecast.com with ESMTP with STARTTLS (version=TLSv1.3, cipher=TLS_AES_256_GCM_SHA384) id us-mta-315-YeGYHasUN2usBBVihp1tsQ-1; Fri, 09 May 2025 11:30:36 -0400 X-MC-Unique: YeGYHasUN2usBBVihp1tsQ-1 X-Mimecast-MFC-AGG-ID: YeGYHasUN2usBBVihp1tsQ_1746804635 Received: by mail-wm1-f72.google.com with SMTP id 5b1f17b1804b1-43d007b2c79so13775945e9.2 for ; Fri, 09 May 2025 08:30:36 -0700 (PDT) X-Google-DKIM-Signature: v=1; a=rsa-sha256; c=relaxed/relaxed; d=1e100.net; s=20230601; t=1746804635; x=1747409435; h=content-transfer-encoding:mime-version:message-id:date:subject:cc :to:from:x-gm-message-state:from:to:cc:subject:date:message-id :reply-to; bh=/QETQBLLgywJiVkdBX35qjwYFT1cFvifg3en6u8zZzc=; b=iXrAGaGF6hcesQ5d9nrPuub2gPEz6PLU9pHT36GbwgZ9wz7YYiVm6MhD/xiU87eco5 z8LtVDm94FLpmkTQu4FDHrFTofIAMu047sE/I45n39ew4h/DF+kOO71y8nfx1Y9Rv2Ps Swy24ewKEJL7WpMqNocJwax+2F0zojgWOOcPCqACt9axTqwdcP82NJrhAv3TVeqB/RZN 8SrANV9eX26CbmSf4eBc/ea2IUVqeBSkJitzQJYJjPC+axrzdbRVSDMjTQThi9kXDX9F lcOedDTAUV6LiYJuPsdJ/TY+G6HKsKrlN8AG1XhwqZgXTGZGzrKPqxebM+yOzs01eoqu FrFQ== X-Gm-Message-State: AOJu0YwMBTR41NpI8CH+QTnR1oSuE6uJ96eTsgInf3nrU6jTMJjL7Q0Y 9g0UW4a63Aikn10sIwShjGKDKLxHL2uIQ4WtYmpHNjV29j/XSd0X34K0vZyzC1XDygrtC7GMEOr ENdeo2m3p/vLgJ/oUAie/Soy4fCRazVt3e9nOUaBwWIOwZG2z X-Gm-Gg: ASbGnct8kjWDGtJbwvMwSaPKXH0yovRWMiwPzLr2qKo5fWy4F/i6wceBF8Lk+7G2sI/ W6UbCJ9PFF3+k0whAgq04eYtQ04Jb3zBLxY5mivMFdmwdbtSLXX0lVtZpThGHetw+Vw3YErRwum 9JzwxZVYyEFNeVvOvG5wlLetfBrwCl7qpO2ugWDL5GNz1saa21QmWu6na8XUidXkHfvxc1a9UZl 69qjb9zgBqQiABExtF8gU3ApW5yQxZd8Nyk2wxxNvGsQQ6/KmdBtyOjr72l4av2aLIzY3LPDBuD Z5ydalEYfZFVprXJJK91Rb0XxlV4X4zLnyka/kl5vaajphm4V/1Eve1EPmQGCl7IDFOMvsR+ X-Received: by 2002:a05:600c:6095:b0:43b:c0fa:f9cd with SMTP id 5b1f17b1804b1-442d6d0a9ebmr34901775e9.7.1746804634988; Fri, 09 May 2025 08:30:34 -0700 (PDT) X-Google-Smtp-Source: AGHT+IE+BxOB6NY2iOaO7FiGmjEOw9RMH7buuCJWOsw4fLTunHCh6Aav62kTXNQTqClatZwokO3a+Q== X-Received: by 2002:a05:600c:6095:b0:43b:c0fa:f9cd with SMTP id 5b1f17b1804b1-442d6d0a9ebmr34901495e9.7.1746804634577; Fri, 09 May 2025 08:30:34 -0700 (PDT) Received: from localhost (p200300d82f4555008267647f4209dedd.dip0.t-ipconnect.de. [2003:d8:2f45:5500:8267:647f:4209:dedd]) by smtp.gmail.com with UTF8SMTPSA id 5b1f17b1804b1-442d67d5c7bsm33791995e9.4.2025.05.09.08.30.33 (version=TLS1_3 cipher=TLS_AES_128_GCM_SHA256 bits=128/128); Fri, 09 May 2025 08:30:34 -0700 (PDT) From: David Hildenbrand To: linux-kernel@vger.kernel.org Cc: linux-mm@kvack.org, linux-kselftest@vger.kernel.org, David Hildenbrand , Andrew Morton , Shuah Khan , Lorenzo Stoakes , Ingo Molnar , Peter Xu , Dev Jain Subject: [PATCH v2] selftests/mm: add simple VM_PFNMAP tests based on mmap'ing /dev/mem Date: Fri, 9 May 2025 17:30:32 +0200 Message-ID: <20250509153033.952746-1-david@redhat.com> X-Mailer: git-send-email 2.49.0 MIME-Version: 1.0 X-Mimecast-Spam-Score: 0 X-Mimecast-MFC-PROC-ID: 1aunkmCFCBy-LUVFyDie1G1isKsEsXs1RVcNzgo-lC4_1746804635 X-Mimecast-Originator: redhat.com Content-Transfer-Encoding: 8bit content-type: text/plain; charset="US-ASCII"; x-default=true X-Rspamd-Server: rspam12 X-Rspamd-Queue-Id: 258D64000A X-Rspam-User: X-Stat-Signature: 5rkbxhxhtmdn3id53bpbxtn9tms7j3gc X-HE-Tag: 1746804637-139566 X-HE-Meta: U2FsdGVkX19wIS2iDZ3uF0Gp8erO8McDhuqJs1QSVGe6+8u+JKU9LBftXP6DHQ2TQclMRm1R+W3Br2AQioH6NNxlSbJ84L4lYK28fpkQA+CYlPEwfmdRtAP1Q7vs2kOjApOqh3NTQmcGslbyznby6hOv9j+TG28jotiCgBiCka3VkZudBiQiDRmLWdTZyVYi5SVylUtec2rea2SsNOPGOoSBQooo82EyFon0FtqfpYi84PPKfOHlQidZS9eR3LcROdXQvadJ0KuuPo7ydxiXNKQOWcf862S2yMzB6zzpp9LEPRzEQuusllFDstQpcJUDEPy4nd8g4QubwbAetcJBxlPw7iw79oUDAulN3m3SLs4tIkwhpqOX6Hxycc1fz0941Wtwln83jtFPAhxpU7f79eVVyPyO7b6NAUvhAihupCTzsWhCIpczpQC7/BucrpM2HyWUr5V0jjvq2fq7HUWeneF0g7V72Urw+U2YVWPSLMYGP503TrASInDqk9WNu0OtAJqlFRHddoY8XaS3uIK0T5OFFIER/5RR3Ltjqg+f30zLlPNWpUuQwYBZrIrJNAUQLQUVLO7LXEX2xwtiiURE6PS0nRme4gNX0juldpKCSrcKMZiUq05Fgy+pf4gdmhW5MFPxDEhcBziEf/6K8dgdxTbLbpJIcfI1edStRVqFla6z2APTJ/eB91oi7vzkBgGGFUEECWyah+g3lQcEY2i9ULFXegSqZC/QjTBGCe1teMdKoPc+erNLnKKXqSQkrhGPllK8ZVLfI3KPj2Q0sjeiTMues/y/cTsuMEqzHB1pj4tt71PCykH4LlnxuDqSslF/tXEVkmrNd0TwZipbv3GJNngzdM5Bk0p6fOiXwp/AeTvpRxmLOIqMaXpfN4Yt1TraV92KAaj4m0ZW/UQhQVcncqGoPhLe5PMJKxO2dMqUx7h0BegybGj9KWGYOnSwq3ErO512XlthcpwdOcjTE/Z phnaXwZC hNYW/J5y4auyfIL4arYuZ9AGY3kMpSgqi8q3AkTh2l+yxmV2cWMzhHMSc36Mv3bSXD5eMl8dN4fbHIPN9xn0OEAVAEHU5fAKnU9G/hfZAUolJHgYf0T+dbfn2y2vMelvv44QcDgAc9uof77bQB1Cu4KmVSieLKH52DBFZhDJWEte/BTHg1E3HvpLe+05kQT34EjhSISal37xCYAFZt064+RgGqIFFiROKXBV1XNGvHESYHb16yuWM04v8BydX1e/mMwZKMecXCa1vodUb9znRGLCX/tVVGNhSBN5eMV2Dg+JcHQbF0KsI8VAb7Qft+tsbhm190GWl8gVqXlpdvjjo6RJ2kEG+qPLX3Wf7UlSksGVhsKQEBf6bYxcV2J4ykbyGxF8nFImgDaLBXy9uCQv9JexZzTqfwTybhANCZKSMX7ZcBrTUAteOwSCjuvmZfJnq5yVjXZiJp8qk2YLj+NpKtLkfGia287UgBUgkhy2RKl/A7XvVVNBvO8JC+f+oLrTNFw4yIx4tG9bCtDpWKOKlqLakWGPq9CR6QG9LWqEVgsIzdN0QXAH69Mtc7aqALpyBQU44ALAPDTtNv+GkVZU4INo89O2bhJ1bIcYqu2UnKEfnZUlqkJwW6qd/rz3xiRj5cEggCU47ODOZd49DAL9cajFNdJOHqx4/L9gsQsE8xhP2HZIdXLMkZrTVEVYx5bo6pNJD X-Bogosity: Ham, tests=bogofilter, spamicity=0.000000, version=1.2.4 Sender: owner-linux-mm@kvack.org Precedence: bulk X-Loop: owner-majordomo@kvack.org List-ID: List-Subscribe: List-Unsubscribe: Let's test some basic functionality using /dev/mem. These tests will implicitly cover some PAT (Page Attribute Handling) handling on x86. These tests will only run when /dev/mem access to the first two pages in physical address space is possible and allowed; otherwise, the tests are skipped. On current x86-64 with PAT inside a VM, all tests pass: TAP version 13 1..6 # Starting 6 tests from 1 test cases. # RUN pfnmap.madvise_disallowed ... # OK pfnmap.madvise_disallowed ok 1 pfnmap.madvise_disallowed # RUN pfnmap.munmap_split ... # OK pfnmap.munmap_split ok 2 pfnmap.munmap_split # RUN pfnmap.mremap_fixed ... # OK pfnmap.mremap_fixed ok 3 pfnmap.mremap_fixed # RUN pfnmap.mremap_shrink ... # OK pfnmap.mremap_shrink ok 4 pfnmap.mremap_shrink # RUN pfnmap.mremap_expand ... # OK pfnmap.mremap_expand ok 5 pfnmap.mremap_expand # RUN pfnmap.fork ... # OK pfnmap.fork ok 6 pfnmap.fork # PASSED: 6 / 6 tests passed. # Totals: pass:6 fail:0 xfail:0 xpass:0 skip:0 error:0 However, we are able to trigger: [ 27.888251] x86/PAT: pfnmap:1790 freeing invalid memtype [mem 0x00000000-0x00000fff] There are probably more things worth testing in the future, such as MAP_PRIVATE handling. But this set of tests is sufficient to cover most of the things we will rework regarding PAT handling. Cc: Andrew Morton Cc: Shuah Khan Cc: Lorenzo Stoakes Cc: Ingo Molnar Cc: Peter Xu Cc: Dev Jain Signed-off-by: David Hildenbrand --- Hopefully I didn't miss any review feedback. v1 -> v2: * Rewrite using kselftest_harness, which simplifies a lot of things * Add to .gitignore and run_vmtests.sh * Register signal handler on demand * Use volatile trick to force a read (not factoring out FORCE_READ just yet) * Drop mprotect() test case * Add some more comments why we test certain things * Use NULL for mmap() first parameter instead of 0 * Smaller fixes --- tools/testing/selftests/mm/.gitignore | 1 + tools/testing/selftests/mm/Makefile | 1 + tools/testing/selftests/mm/pfnmap.c | 196 ++++++++++++++++++++++ tools/testing/selftests/mm/run_vmtests.sh | 4 + 4 files changed, 202 insertions(+) create mode 100644 tools/testing/selftests/mm/pfnmap.c diff --git a/tools/testing/selftests/mm/.gitignore b/tools/testing/selftests/mm/.gitignore index 91db34941a143..824266982aa36 100644 --- a/tools/testing/selftests/mm/.gitignore +++ b/tools/testing/selftests/mm/.gitignore @@ -20,6 +20,7 @@ mremap_test on-fault-limit transhuge-stress pagemap_ioctl +pfnmap *.tmp* protection_keys protection_keys_32 diff --git a/tools/testing/selftests/mm/Makefile b/tools/testing/selftests/mm/Makefile index ad4d6043a60f0..ae6f994d3add7 100644 --- a/tools/testing/selftests/mm/Makefile +++ b/tools/testing/selftests/mm/Makefile @@ -84,6 +84,7 @@ TEST_GEN_FILES += mremap_test TEST_GEN_FILES += mseal_test TEST_GEN_FILES += on-fault-limit TEST_GEN_FILES += pagemap_ioctl +TEST_GEN_FILES += pfnmap TEST_GEN_FILES += thuge-gen TEST_GEN_FILES += transhuge-stress TEST_GEN_FILES += uffd-stress diff --git a/tools/testing/selftests/mm/pfnmap.c b/tools/testing/selftests/mm/pfnmap.c new file mode 100644 index 0000000000000..8a9d19b6020c7 --- /dev/null +++ b/tools/testing/selftests/mm/pfnmap.c @@ -0,0 +1,196 @@ +// SPDX-License-Identifier: GPL-2.0-only +/* + * Basic VM_PFNMAP tests relying on mmap() of '/dev/mem' + * + * Copyright 2025, Red Hat, Inc. + * + * Author(s): David Hildenbrand + */ +#define _GNU_SOURCE +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include +#include + +#include "../kselftest_harness.h" +#include "vm_util.h" + +static sigjmp_buf sigjmp_buf_env; + +static void signal_handler(int sig) +{ + siglongjmp(sigjmp_buf_env, -EFAULT); +} + +static int test_read_access(char *addr, size_t size, size_t pagesize) +{ + size_t offs; + int ret; + + if (signal(SIGSEGV, signal_handler) == SIG_ERR) + return -EINVAL; + + ret = sigsetjmp(sigjmp_buf_env, 1); + if (!ret) { + for (offs = 0; offs < size; offs += pagesize) + /* Force a read that the compiler cannot optimize out. */ + *((volatile char *)(addr + offs)); + } + if (signal(SIGSEGV, signal_handler) == SIG_ERR) + return -EINVAL; + + return ret; +} + +FIXTURE(pfnmap) +{ + size_t pagesize; + int dev_mem_fd; + char *addr1; + size_t size1; + char *addr2; + size_t size2; +}; + +FIXTURE_SETUP(pfnmap) +{ + self->pagesize = getpagesize(); + + self->dev_mem_fd = open("/dev/mem", O_RDONLY); + if (self->dev_mem_fd < 0) + SKIP(return, "Cannot open '/dev/mem'\n"); + + /* We'll require the first two pages throughout our tests ... */ + self->size1 = self->pagesize * 2; + self->addr1 = mmap(NULL, self->size1, PROT_READ, MAP_SHARED, + self->dev_mem_fd, 0); + if (self->addr1 == MAP_FAILED) + SKIP(return, "Cannot mmap '/dev/mem'\n"); + + /* ... and want to be able to read from them. */ + if (test_read_access(self->addr1, self->size1, self->pagesize)) + SKIP(return, "Cannot read-access mmap'ed '/dev/mem'\n"); + + self->size2 = 0; + self->addr2 = MAP_FAILED; +} + +FIXTURE_TEARDOWN(pfnmap) +{ + if (self->addr2 != MAP_FAILED) + munmap(self->addr2, self->size2); + if (self->addr1 != MAP_FAILED) + munmap(self->addr1, self->size1); + if (self->dev_mem_fd >= 0) + close(self->dev_mem_fd); +} + +TEST_F(pfnmap, madvise_disallowed) +{ + int advices[] = { + MADV_DONTNEED, + MADV_DONTNEED_LOCKED, + MADV_FREE, + MADV_WIPEONFORK, + MADV_COLD, + MADV_PAGEOUT, + MADV_POPULATE_READ, + MADV_POPULATE_WRITE, + }; + int i; + + /* All these advices must be rejected. */ + for (i = 0; i < ARRAY_SIZE(advices); i++) { + EXPECT_LT(madvise(self->addr1, self->pagesize, advices[i]), 0); + EXPECT_EQ(errno, EINVAL); + } +} + +TEST_F(pfnmap, munmap_split) +{ + /* + * Unmap the first page. This munmap() call is not really expected to + * fail, but we might be able to trigger other internal issues. + */ + ASSERT_EQ(munmap(self->addr1, self->pagesize), 0); + + /* + * Remap the first page while the second page is still mapped. This + * makes sure that any PAT tracking on x86 will allow for mmap()'ing + * a page again while some parts of the first mmap() are still + * around. + */ + self->size2 = self->pagesize; + self->addr2 = mmap(NULL, self->pagesize, PROT_READ, MAP_SHARED, + self->dev_mem_fd, 0); + ASSERT_NE(self->addr2, MAP_FAILED); +} + +TEST_F(pfnmap, mremap_fixed) +{ + char *ret; + + /* Reserve a destination area. */ + self->size2 = self->size1; + self->addr2 = mmap(NULL, self->size2, PROT_READ, MAP_ANON | MAP_PRIVATE, + -1, 0); + ASSERT_NE(self->addr2, MAP_FAILED); + + /* mremap() over our destination. */ + ret = mremap(self->addr1, self->size1, self->size2, + MREMAP_FIXED | MREMAP_MAYMOVE, self->addr2); + ASSERT_NE(ret, MAP_FAILED); +} + +TEST_F(pfnmap, mremap_shrink) +{ + char *ret; + + /* Shrinking is expected to work. */ + ret = mremap(self->addr1, self->size1, self->size1 - self->pagesize, 0); + ASSERT_NE(ret, MAP_FAILED); +} + +TEST_F(pfnmap, mremap_expand) +{ + /* + * Growing is not expected to work, and getting it right would + * be challenging. So this test primarily serves as an early warning + * that something that probably should never work suddenly works. + */ + self->size2 = self->size1 + self->pagesize; + self->addr2 = mremap(self->addr1, self->size1, self->size2, MREMAP_MAYMOVE); + ASSERT_EQ(self->addr2, MAP_FAILED); +} + +TEST_F(pfnmap, fork) +{ + pid_t pid; + int ret; + + /* fork() a child and test if the child can access the pages. */ + pid = fork(); + ASSERT_GE(pid, 0); + + if (!pid) { + EXPECT_EQ(test_read_access(self->addr1, self->size1, + self->pagesize), 0); + exit(0); + } + + wait(&ret); + if (WIFEXITED(ret)) + ret = WEXITSTATUS(ret); + else + ret = -EINVAL; + ASSERT_EQ(ret, 0); +} + +TEST_HARNESS_MAIN diff --git a/tools/testing/selftests/mm/run_vmtests.sh b/tools/testing/selftests/mm/run_vmtests.sh index 188b125bf1f6b..dddd1dd8af145 100755 --- a/tools/testing/selftests/mm/run_vmtests.sh +++ b/tools/testing/selftests/mm/run_vmtests.sh @@ -63,6 +63,8 @@ separated by spaces: test soft dirty page bit semantics - pagemap test pagemap_scan IOCTL +- pfnmap + tests for VM_PFNMAP handling - cow test copy-on-write semantics - thp @@ -472,6 +474,8 @@ fi CATEGORY="pagemap" run_test ./pagemap_ioctl +CATEGORY="pfnmap" run_test ./pfnmap + # COW tests CATEGORY="cow" run_test ./cow -- 2.49.0