diff --git a/fabric-resource-loader-v0/src/client/java/net/fabricmc/fabric/mixin/resource/loader/client/ClientDataPackManagerMixin.java b/fabric-resource-loader-v0/src/client/java/net/fabricmc/fabric/mixin/resource/loader/client/ClientDataPackManagerMixin.java
new file mode 100644
index 000000000..28166019c
--- /dev/null
+++ b/fabric-resource-loader-v0/src/client/java/net/fabricmc/fabric/mixin/resource/loader/client/ClientDataPackManagerMixin.java
@@ -0,0 +1,55 @@
+/*
+ * Copyright (c) 2016, 2017, 2018, 2019 FabricMC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package net.fabricmc.fabric.mixin.resource.loader.client;
+
+import java.util.List;
+
+import com.llamalad7.mixinextras.injector.ModifyReturnValue;
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import org.spongepowered.asm.mixin.Mixin;
+import org.spongepowered.asm.mixin.Unique;
+import org.spongepowered.asm.mixin.injection.At;
+import org.spongepowered.asm.mixin.injection.Redirect;
+
+import net.minecraft.client.resource.ClientDataPackManager;
+import net.minecraft.registry.VersionedIdentifier;
+import net.minecraft.resource.ResourcePackManager;
+
+import net.fabricmc.fabric.impl.resource.loader.ModResourcePackCreator;
+import net.fabricmc.fabric.impl.resource.loader.ModResourcePackUtil;
+
+@Mixin(ClientDataPackManager.class)
+public class ClientDataPackManagerMixin {
+	@Unique
+	private static final Logger LOGGER = LoggerFactory.getLogger("ClientDataPackManagerMixin");
+
+	@Redirect(method = "<init>", at = @At(value = "INVOKE", target = "Lnet/minecraft/resource/VanillaDataPackProvider;createClientManager()Lnet/minecraft/resource/ResourcePackManager;"))
+	public ResourcePackManager createClientManager() {
+		return ModResourcePackUtil.createClientManager();
+	}
+
+	@ModifyReturnValue(method = "getCommonKnownPacks", at = @At("RETURN"))
+	List<VersionedIdentifier> getCommonKnownPacksReturn(List<VersionedIdentifier> original) {
+		if (original.size() > ModResourcePackCreator.MAX_KNOWN_PACKS) {
+			LOGGER.warn("Too many knownPacks: Found {}; max {}", original.size(), ModResourcePackCreator.MAX_KNOWN_PACKS);
+			return original.subList(0, ModResourcePackCreator.MAX_KNOWN_PACKS);
+		}
+
+		return original;
+	}
+}
diff --git a/fabric-resource-loader-v0/src/client/resources/fabric-resource-loader-v0.client.mixins.json b/fabric-resource-loader-v0/src/client/resources/fabric-resource-loader-v0.client.mixins.json
index 8e7b6d57d..7a91d9fb1 100644
--- a/fabric-resource-loader-v0/src/client/resources/fabric-resource-loader-v0.client.mixins.json
+++ b/fabric-resource-loader-v0/src/client/resources/fabric-resource-loader-v0.client.mixins.json
@@ -9,7 +9,8 @@
     "KeyedResourceReloadListenerClientMixin",
     "ResourcePackOrganizerMixin",
     "VanillaResourcePackProviderMixin",
-    "GameOptionsWriteVisitorMixin"
+    "GameOptionsWriteVisitorMixin",
+    "ClientDataPackManagerMixin"
   ],
   "injectors": {
     "defaultRequire": 1
diff --git a/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/impl/resource/loader/FabricOriginalKnownPacksGetter.java b/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/impl/resource/loader/FabricOriginalKnownPacksGetter.java
new file mode 100644
index 000000000..cfef14cc1
--- /dev/null
+++ b/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/impl/resource/loader/FabricOriginalKnownPacksGetter.java
@@ -0,0 +1,28 @@
+/*
+ * Copyright (c) 2016, 2017, 2018, 2019 FabricMC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package net.fabricmc.fabric.impl.resource.loader;
+
+import java.util.List;
+
+import net.minecraft.registry.VersionedIdentifier;
+
+public interface FabricOriginalKnownPacksGetter {
+	/**
+	 * @return the data packs known at server start
+	 */
+	List<VersionedIdentifier> fabric_getOriginalKnownPacks();
+}
diff --git a/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/impl/resource/loader/ModNioResourcePack.java b/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/impl/resource/loader/ModNioResourcePack.java
index 00aa0480b..75c604f93 100644
--- a/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/impl/resource/loader/ModNioResourcePack.java
+++ b/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/impl/resource/loader/ModNioResourcePack.java
@@ -42,6 +42,7 @@ import org.jetbrains.annotations.Nullable;
 import org.slf4j.Logger;
 import org.slf4j.LoggerFactory;
 
+import net.minecraft.registry.VersionedIdentifier;
 import net.minecraft.resource.AbstractFileResourcePack;
 import net.minecraft.resource.InputSupplier;
 import net.minecraft.resource.ResourcePack;
@@ -106,7 +107,7 @@ public class ModNioResourcePack implements ResourcePack, ModResourcePack {
 				packId,
 				displayName,
 				ModResourcePackCreator.RESOURCE_PACK_SOURCE,
-				Optional.empty()
+				Optional.of(new VersionedIdentifier(ModResourcePackCreator.FABRIC, packId, mod.getMetadata().getVersion().getFriendlyString()))
 		);
 		ModNioResourcePack ret = new ModNioResourcePack(packId, mod, paths, type, activationType, modBundled, metadata);
 
diff --git a/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/impl/resource/loader/ModResourcePackCreator.java b/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/impl/resource/loader/ModResourcePackCreator.java
index 36374d785..303ed305b 100644
--- a/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/impl/resource/loader/ModResourcePackCreator.java
+++ b/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/impl/resource/loader/ModResourcePackCreator.java
@@ -68,12 +68,23 @@ public class ModResourcePackCreator implements ResourcePackProvider {
 		}
 	};
 	public static final ModResourcePackCreator CLIENT_RESOURCE_PACK_PROVIDER = new ModResourcePackCreator(ResourceType.CLIENT_RESOURCES);
-	private static final ResourcePackPosition ACTIVATION_INFO = new ResourcePackPosition(true, ResourcePackProfile.InsertionPosition.TOP, false);
+	/**
+	 * The maximum ammount of known data packs requested from the client, including vanilla data packs.
+	 */
+	public static final int MAX_KNOWN_PACKS = Integer.getInteger("fabric-resource-loader-v0:maxKnownPacks", 1024);
 
 	private final ResourceType type;
+	private final ResourcePackPosition activationInfo;
+	private final boolean forClientDataPackManager;
 
 	public ModResourcePackCreator(ResourceType type) {
+		this(type, false);
+	}
+
+	protected ModResourcePackCreator(ResourceType type, boolean forClientDataPackManager) {
 		this.type = type;
+		this.activationInfo = new ResourcePackPosition(!forClientDataPackManager, ResourcePackProfile.InsertionPosition.TOP, false);
+		this.forClientDataPackManager = forClientDataPackManager;
 	}
 
 	/**
@@ -106,7 +117,7 @@ public class ModResourcePackCreator implements ResourcePackProvider {
 				metadata,
 				new PlaceholderResourcePack.Factory(this.type, metadata),
 				this.type,
-				ACTIVATION_INFO
+				this.activationInfo
 		));
 
 		// Build a list of mod resource packs.
@@ -131,11 +142,14 @@ public class ModResourcePackCreator implements ResourcePackProvider {
 					pack.getInfo(),
 					new ModResourcePackFactory(pack),
 					this.type,
-					ACTIVATION_INFO
+					this.activationInfo
 			);
 
 			if (profile != null) {
-				((FabricResourcePackProfile) profile).fabric_setParentsPredicate(parents);
+				if (!forClientDataPackManager) {
+					((FabricResourcePackProfile) profile).fabric_setParentsPredicate(parents);
+				}
+
 				consumer.accept(profile);
 			}
 		}
diff --git a/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/impl/resource/loader/ModResourcePackUtil.java b/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/impl/resource/loader/ModResourcePackUtil.java
index f7a50f831..7a3231716 100644
--- a/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/impl/resource/loader/ModResourcePackUtil.java
+++ b/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/impl/resource/loader/ModResourcePackUtil.java
@@ -43,11 +43,14 @@ import net.minecraft.SharedConstants;
 import net.minecraft.resource.DataConfiguration;
 import net.minecraft.resource.DataPackSettings;
 import net.minecraft.resource.ResourcePack;
+import net.minecraft.resource.ResourcePackManager;
 import net.minecraft.resource.ResourcePackProfile;
 import net.minecraft.resource.ResourceType;
+import net.minecraft.resource.VanillaDataPackProvider;
 import net.minecraft.resource.featuretoggle.FeatureFlags;
 import net.minecraft.resource.metadata.PackResourceMetadata;
 import net.minecraft.text.Text;
+import net.minecraft.util.path.SymlinkFinder;
 
 import net.fabricmc.fabric.api.resource.ModResourcePack;
 import net.fabricmc.fabric.api.resource.ResourcePackActivationType;
@@ -233,4 +236,12 @@ public final class ModResourcePackUtil {
 
 		return new DataPackSettings(enabled, disabled);
 	}
+
+	/**
+	 * Creates the ResousePackManager used by the ClientDataPackManager and replaces
+	 * {@code VanillaDataPackProvider.createClientManager} used by vanilla.
+	 */
+	public static ResourcePackManager createClientManager() {
+		return new ResourcePackManager(new VanillaDataPackProvider(new SymlinkFinder((path) -> true)), new ModResourcePackCreator(ResourceType.SERVER_DATA, true));
+	}
 }
diff --git a/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/impl/resource/loader/ResourceManagerHelperImpl.java b/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/impl/resource/loader/ResourceManagerHelperImpl.java
index fe51ceace..d3521ab16 100644
--- a/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/impl/resource/loader/ResourceManagerHelperImpl.java
+++ b/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/impl/resource/loader/ResourceManagerHelperImpl.java
@@ -25,7 +25,6 @@ import java.util.Iterator;
 import java.util.LinkedHashSet;
 import java.util.List;
 import java.util.Map;
-import java.util.Optional;
 import java.util.Set;
 import java.util.function.Consumer;
 
@@ -119,7 +118,7 @@ public class ResourceManagerHelperImpl implements ResourceManagerHelper {
 						entry.getRight().getId(),
 						entry.getLeft(),
 						new BuiltinModResourcePackSource(pack.getFabricModMetadata().getName()),
-						Optional.empty()
+						entry.getRight().getKnownPackInfo()
 				);
 				ResourcePackPosition info2 = new ResourcePackPosition(
 						pack.getActivationType() == ResourcePackActivationType.ALWAYS_ENABLED,
diff --git a/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/mixin/resource/loader/MinecraftServerMixin.java b/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/mixin/resource/loader/MinecraftServerMixin.java
index 07c5c139c..d5c060764 100644
--- a/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/mixin/resource/loader/MinecraftServerMixin.java
+++ b/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/mixin/resource/loader/MinecraftServerMixin.java
@@ -16,22 +16,41 @@
 
 package net.fabricmc.fabric.mixin.resource.loader;
 
+import java.net.Proxy;
 import java.util.List;
 
+import com.mojang.datafixers.DataFixer;
 import org.spongepowered.asm.mixin.Mixin;
+import org.spongepowered.asm.mixin.Unique;
 import org.spongepowered.asm.mixin.injection.At;
+import org.spongepowered.asm.mixin.injection.Inject;
 import org.spongepowered.asm.mixin.injection.Redirect;
+import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
 
+import net.minecraft.registry.VersionedIdentifier;
 import net.minecraft.resource.ResourcePack;
 import net.minecraft.resource.ResourcePackManager;
 import net.minecraft.resource.ResourcePackProfile;
 import net.minecraft.server.MinecraftServer;
+import net.minecraft.server.SaveLoader;
+import net.minecraft.server.WorldGenerationProgressListenerFactory;
+import net.minecraft.util.ApiServices;
+import net.minecraft.world.level.storage.LevelStorage;
 
 import net.fabricmc.fabric.impl.resource.loader.BuiltinModResourcePackSource;
+import net.fabricmc.fabric.impl.resource.loader.FabricOriginalKnownPacksGetter;
 import net.fabricmc.fabric.impl.resource.loader.ModNioResourcePack;
 
 @Mixin(MinecraftServer.class)
-public class MinecraftServerMixin {
+public class MinecraftServerMixin implements FabricOriginalKnownPacksGetter {
+	@Unique
+	private List<VersionedIdentifier> fabric_originalKnownPacks;
+
+	@Inject(method = "<init>", at = @At("TAIL"))
+	private void init(Thread serverThread, LevelStorage.Session session, ResourcePackManager dataPackManager, SaveLoader saveLoader, Proxy proxy, DataFixer dataFixer, ApiServices apiServices, WorldGenerationProgressListenerFactory worldGenerationProgressListenerFactory, CallbackInfo ci) {
+		this.fabric_originalKnownPacks = saveLoader.resourceManager().streamResourcePacks().flatMap(pack -> pack.getInfo().knownPackInfo().stream()).toList();
+	}
+
 	@Redirect(method = "loadDataPacks(Lnet/minecraft/resource/ResourcePackManager;Lnet/minecraft/resource/DataConfiguration;ZZ)Lnet/minecraft/resource/DataConfiguration;", at = @At(value = "INVOKE", target = "Ljava/util/List;contains(Ljava/lang/Object;)Z"))
 	private static boolean onCheckDisabled(List<String> list, Object o, ResourcePackManager resourcePackManager) {
 		String profileId = (String) o;
@@ -52,4 +71,9 @@ public class MinecraftServerMixin {
 
 		return false;
 	}
+
+	@Override
+	public List<VersionedIdentifier> fabric_getOriginalKnownPacks() {
+		return this.fabric_originalKnownPacks;
+	}
 }
diff --git a/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/mixin/resource/loader/RegistryLoaderMixin.java b/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/mixin/resource/loader/RegistryLoaderMixin.java
deleted file mode 100644
index 1e254bb4d..000000000
--- a/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/mixin/resource/loader/RegistryLoaderMixin.java
+++ /dev/null
@@ -1,51 +0,0 @@
-/*
- * Copyright (c) 2016, 2017, 2018, 2019 FabricMC
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- *     http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
-
-package net.fabricmc.fabric.mixin.resource.loader;
-
-import java.util.Optional;
-
-import com.llamalad7.mixinextras.injector.ModifyExpressionValue;
-import com.llamalad7.mixinextras.sugar.Local;
-import com.mojang.serialization.Lifecycle;
-import org.spongepowered.asm.mixin.Mixin;
-import org.spongepowered.asm.mixin.Unique;
-import org.spongepowered.asm.mixin.injection.At;
-
-import net.minecraft.registry.RegistryLoader;
-import net.minecraft.registry.entry.RegistryEntryInfo;
-import net.minecraft.resource.Resource;
-
-import net.fabricmc.fabric.api.resource.ModResourcePack;
-
-@Mixin(RegistryLoader.class)
-public class RegistryLoaderMixin {
-	@Unique
-	private static final RegistryEntryInfo MOD_PROVIDED_INFO = new RegistryEntryInfo(Optional.empty(), Lifecycle.stable());
-
-	// On the server side, loading mod-provided DRM entries should not show experiments screen.
-	// While the lifecycle is set to experimental on the client side (a de-sync),
-	// there is no good way to fix this without breaking protocol; the lifecycle seems to be unused on
-	// the client side anyway.
-	@ModifyExpressionValue(method = "loadFromResource(Lnet/minecraft/resource/ResourceManager;Lnet/minecraft/registry/RegistryOps$RegistryInfoGetter;Lnet/minecraft/registry/MutableRegistry;Lcom/mojang/serialization/Decoder;Ljava/util/Map;)V", at = @At(value = "INVOKE", target = "Ljava/util/function/Function;apply(Ljava/lang/Object;)Ljava/lang/Object;"))
-	private static Object markModProvidedAsStable(Object original, @Local Resource resource) {
-		if (original instanceof RegistryEntryInfo info && info.knownPackInfo().isEmpty() && resource.getPack() instanceof ModResourcePack) {
-			return MOD_PROVIDED_INFO;
-		}
-
-		return original;
-	}
-}
diff --git a/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/mixin/resource/loader/SelectKnownPacksC2SPacketMixin.java b/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/mixin/resource/loader/SelectKnownPacksC2SPacketMixin.java
new file mode 100644
index 000000000..e7af53191
--- /dev/null
+++ b/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/mixin/resource/loader/SelectKnownPacksC2SPacketMixin.java
@@ -0,0 +1,33 @@
+/*
+ * Copyright (c) 2016, 2017, 2018, 2019 FabricMC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package net.fabricmc.fabric.mixin.resource.loader;
+
+import org.spongepowered.asm.mixin.Mixin;
+import org.spongepowered.asm.mixin.injection.At;
+import org.spongepowered.asm.mixin.injection.ModifyArg;
+
+import net.minecraft.network.packet.c2s.config.SelectKnownPacksC2SPacket;
+
+import net.fabricmc.fabric.impl.resource.loader.ModResourcePackCreator;
+
+@Mixin(SelectKnownPacksC2SPacket.class)
+public class SelectKnownPacksC2SPacketMixin {
+	@ModifyArg(method = "<clinit>", at = @At(value = "INVOKE", target = "Lnet/minecraft/network/codec/PacketCodecs;toList(I)Lnet/minecraft/network/codec/PacketCodec$ResultFunction;"))
+	private static int setMaxKnownPacks(int constant) {
+		return ModResourcePackCreator.MAX_KNOWN_PACKS;
+	}
+}
diff --git a/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/mixin/resource/loader/ServerConfigurationNetworkHandlerMixin.java b/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/mixin/resource/loader/ServerConfigurationNetworkHandlerMixin.java
new file mode 100644
index 000000000..842f63405
--- /dev/null
+++ b/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/mixin/resource/loader/ServerConfigurationNetworkHandlerMixin.java
@@ -0,0 +1,49 @@
+/*
+ * Copyright (c) 2016, 2017, 2018, 2019 FabricMC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package net.fabricmc.fabric.mixin.resource.loader;
+
+import java.util.List;
+
+import org.spongepowered.asm.mixin.Mixin;
+import org.spongepowered.asm.mixin.injection.At;
+import org.spongepowered.asm.mixin.injection.ModifyArg;
+
+import net.minecraft.network.ClientConnection;
+import net.minecraft.registry.VersionedIdentifier;
+import net.minecraft.server.MinecraftServer;
+import net.minecraft.server.network.ConnectedClientData;
+import net.minecraft.server.network.ServerCommonNetworkHandler;
+import net.minecraft.server.network.ServerConfigurationNetworkHandler;
+
+import net.fabricmc.fabric.impl.resource.loader.FabricOriginalKnownPacksGetter;
+
+@Mixin(ServerConfigurationNetworkHandler.class)
+public abstract class ServerConfigurationNetworkHandlerMixin extends ServerCommonNetworkHandler {
+	public ServerConfigurationNetworkHandlerMixin(MinecraftServer server, ClientConnection connection, ConnectedClientData clientData) {
+		super(server, connection, clientData);
+	}
+
+	/**
+	 * Only use packs that were enabled at server start and are enabled now. This avoids a descync when packs have been
+	 * enabled or disabled before the client joins. Since the server registry contents aren't reloaded, we don't want
+	 * the client to use the new data pack data.
+	 */
+	@ModifyArg(method = "sendConfigurations", at = @At(value = "INVOKE", target = "Lnet/minecraft/server/network/SynchronizeRegistriesTask;<init>(Ljava/util/List;Lnet/minecraft/registry/CombinedDynamicRegistries;)V", ordinal = 0))
+	public List<VersionedIdentifier> filterKnownPacks(List<VersionedIdentifier> currentKnownPacks) {
+		return ((FabricOriginalKnownPacksGetter) this.server).fabric_getOriginalKnownPacks().stream().filter(currentKnownPacks::contains).toList();
+	}
+}
diff --git a/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/mixin/resource/loader/SynchronizeRegistriesTaskMixin.java b/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/mixin/resource/loader/SynchronizeRegistriesTaskMixin.java
new file mode 100644
index 000000000..506ff01aa
--- /dev/null
+++ b/fabric-resource-loader-v0/src/main/java/net/fabricmc/fabric/mixin/resource/loader/SynchronizeRegistriesTaskMixin.java
@@ -0,0 +1,73 @@
+/*
+ * Copyright (c) 2016, 2017, 2018, 2019 FabricMC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package net.fabricmc.fabric.mixin.resource.loader;
+
+import java.util.HashSet;
+import java.util.List;
+import java.util.Set;
+import java.util.function.Consumer;
+
+import org.slf4j.Logger;
+import org.slf4j.LoggerFactory;
+import org.spongepowered.asm.mixin.Final;
+import org.spongepowered.asm.mixin.Mixin;
+import org.spongepowered.asm.mixin.Shadow;
+import org.spongepowered.asm.mixin.Unique;
+import org.spongepowered.asm.mixin.injection.At;
+import org.spongepowered.asm.mixin.injection.Inject;
+import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
+
+import net.minecraft.network.packet.Packet;
+import net.minecraft.network.packet.s2c.config.SelectKnownPacksS2CPacket;
+import net.minecraft.registry.VersionedIdentifier;
+import net.minecraft.server.network.SynchronizeRegistriesTask;
+
+import net.fabricmc.fabric.impl.resource.loader.ModResourcePackCreator;
+
+@Mixin(SynchronizeRegistriesTask.class)
+public abstract class SynchronizeRegistriesTaskMixin {
+	@Unique
+	private static final Logger LOGGER = LoggerFactory.getLogger("SynchronizeRegistriesTaskMixin");
+	@Shadow
+	@Final
+	private List<VersionedIdentifier> knownPacks;
+
+	@Shadow
+	protected abstract void syncRegistryAndTags(Consumer<Packet<?>> sender, Set<VersionedIdentifier> commonKnownPacks);
+
+	@Inject(method = "onSelectKnownPacks", at = @At("HEAD"), cancellable = true)
+	public void onSelectKnownPacks(List<VersionedIdentifier> clientKnownPacks, Consumer<Packet<?>> sender, CallbackInfo ci) {
+		if (new HashSet<>(this.knownPacks).containsAll(clientKnownPacks)) {
+			this.syncRegistryAndTags(sender, Set.copyOf(clientKnownPacks));
+			ci.cancel();
+		}
+	}
+
+	@Inject(method = "syncRegistryAndTags", at = @At("HEAD"))
+	public void syncRegistryAndTags(Consumer<Packet<?>> sender, Set<VersionedIdentifier> commonKnownPacks, CallbackInfo ci) {
+		LOGGER.debug("Syncronizing registries with common known packs: {}", commonKnownPacks);
+	}
+
+	@Inject(method = "sendPacket", at = @At("HEAD"), cancellable = true)
+	private void sendPacket(Consumer<Packet<?>> sender, CallbackInfo ci) {
+		if (this.knownPacks.size() > ModResourcePackCreator.MAX_KNOWN_PACKS) {
+			LOGGER.warn("Too many knownPacks: Found {}; max {}", this.knownPacks.size(), ModResourcePackCreator.MAX_KNOWN_PACKS);
+			sender.accept(new SelectKnownPacksS2CPacket(this.knownPacks.subList(0, ModResourcePackCreator.MAX_KNOWN_PACKS)));
+			ci.cancel();
+		}
+	}
+}
diff --git a/fabric-resource-loader-v0/src/main/resources/fabric-resource-loader-v0.mixins.json b/fabric-resource-loader-v0/src/main/resources/fabric-resource-loader-v0.mixins.json
index 10f9d5e5d..04c6747b7 100644
--- a/fabric-resource-loader-v0/src/main/resources/fabric-resource-loader-v0.mixins.json
+++ b/fabric-resource-loader-v0/src/main/resources/fabric-resource-loader-v0.mixins.json
@@ -7,12 +7,14 @@
     "KeyedResourceReloadListenerMixin",
     "LifecycledResourceManagerImplMixin",
     "MinecraftServerMixin",
-    "RegistryLoaderMixin",
     "ResourceMixin",
     "ResourcePackManagerMixin",
     "ResourcePackProfileMixin",
+    "SelectKnownPacksC2SPacketMixin",
+    "ServerConfigurationNetworkHandlerMixin",
     "ServerPropertiesHandlerMixin",
     "SimpleResourceReloadMixin",
+    "SynchronizeRegistriesTaskMixin",
     "TestServerMixin"
   ],
   "server": [
diff --git a/fabric-resource-loader-v0/src/testmod/java/net/fabricmc/fabric/test/mixin/resource/loader/SynchronizeRegistriesTaskMixin.java b/fabric-resource-loader-v0/src/testmod/java/net/fabricmc/fabric/test/mixin/resource/loader/SynchronizeRegistriesTaskMixin.java
new file mode 100644
index 000000000..bd6b2e4f4
--- /dev/null
+++ b/fabric-resource-loader-v0/src/testmod/java/net/fabricmc/fabric/test/mixin/resource/loader/SynchronizeRegistriesTaskMixin.java
@@ -0,0 +1,50 @@
+/*
+ * Copyright (c) 2016, 2017, 2018, 2019 FabricMC
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ *     http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+package net.fabricmc.fabric.test.mixin.resource.loader;
+
+import java.util.List;
+import java.util.Set;
+import java.util.function.Consumer;
+
+import org.spongepowered.asm.mixin.Final;
+import org.spongepowered.asm.mixin.Mixin;
+import org.spongepowered.asm.mixin.Shadow;
+import org.spongepowered.asm.mixin.injection.At;
+import org.spongepowered.asm.mixin.injection.Inject;
+import org.spongepowered.asm.mixin.injection.callback.CallbackInfo;
+
+import net.minecraft.network.packet.Packet;
+import net.minecraft.registry.VersionedIdentifier;
+import net.minecraft.server.network.SynchronizeRegistriesTask;
+
+import net.fabricmc.fabric.test.resource.loader.BuiltinResourcePackTestMod;
+
+@Mixin(SynchronizeRegistriesTask.class)
+public class SynchronizeRegistriesTaskMixin {
+	@Shadow
+	@Final
+	private List<VersionedIdentifier> knownPacks;
+
+	@Inject(method = "syncRegistryAndTags", at = @At("HEAD"))
+	public void syncRegistryAndTags(Consumer<Packet<?>> sender, Set<VersionedIdentifier> commonKnownPacks, CallbackInfo ci) {
+		BuiltinResourcePackTestMod.LOGGER.info("Syncronizing registries with common known packs: {}", commonKnownPacks);
+
+		if (!commonKnownPacks.containsAll(this.knownPacks)) {
+			BuiltinResourcePackTestMod.LOGGER.error("(Ignore when not local client) Not all server mod data packs known to client. Missing: {}", this.knownPacks.stream().filter(knownPack -> !commonKnownPacks.contains(knownPack)).toList());
+		}
+	}
+}
diff --git a/fabric-resource-loader-v0/src/testmod/java/net/fabricmc/fabric/test/resource/loader/BuiltinResourcePackTestMod.java b/fabric-resource-loader-v0/src/testmod/java/net/fabricmc/fabric/test/resource/loader/BuiltinResourcePackTestMod.java
index a7257c1b2..95a003726 100644
--- a/fabric-resource-loader-v0/src/testmod/java/net/fabricmc/fabric/test/resource/loader/BuiltinResourcePackTestMod.java
+++ b/fabric-resource-loader-v0/src/testmod/java/net/fabricmc/fabric/test/resource/loader/BuiltinResourcePackTestMod.java
@@ -30,7 +30,7 @@ import net.fabricmc.loader.api.FabricLoader;
 public class BuiltinResourcePackTestMod implements ModInitializer {
 	public static final String MODID = "fabric-resource-loader-v0-testmod";
 
-	private static final Logger LOGGER = LoggerFactory.getLogger(BuiltinResourcePackTestMod.class);
+	public static final Logger LOGGER = LoggerFactory.getLogger(BuiltinResourcePackTestMod.class);
 
 	@Override
 	public void onInitialize() {
diff --git a/fabric-resource-loader-v0/src/testmod/resources/assets/fabric-resource-loader-v0-testmod/textures/entity/wolf/green.png b/fabric-resource-loader-v0/src/testmod/resources/assets/fabric-resource-loader-v0-testmod/textures/entity/wolf/green.png
new file mode 100644
index 000000000..9ad0758ce
Binary files /dev/null and b/fabric-resource-loader-v0/src/testmod/resources/assets/fabric-resource-loader-v0-testmod/textures/entity/wolf/green.png differ
diff --git a/fabric-resource-loader-v0/src/testmod/resources/assets/fabric-resource-loader-v0-testmod/textures/entity/wolf/pink.png b/fabric-resource-loader-v0/src/testmod/resources/assets/fabric-resource-loader-v0-testmod/textures/entity/wolf/pink.png
new file mode 100644
index 000000000..d76d6f734
Binary files /dev/null and b/fabric-resource-loader-v0/src/testmod/resources/assets/fabric-resource-loader-v0-testmod/textures/entity/wolf/pink.png differ
diff --git a/fabric-resource-loader-v0/src/testmod/resources/data/minecraft/wolf_variant/woods.json b/fabric-resource-loader-v0/src/testmod/resources/data/minecraft/wolf_variant/woods.json
new file mode 100644
index 000000000..de53dddfc
--- /dev/null
+++ b/fabric-resource-loader-v0/src/testmod/resources/data/minecraft/wolf_variant/woods.json
@@ -0,0 +1,6 @@
+{
+  "wild_texture": "fabric-resource-loader-v0-testmod:entity/wolf/green",
+  "angry_texture": "fabric-resource-loader-v0-testmod:entity/wolf/green",
+  "tame_texture": "fabric-resource-loader-v0-testmod:entity/wolf/green",
+  "biomes": "minecraft:forest"
+}
diff --git a/fabric-resource-loader-v0/src/testmod/resources/fabric-resource-loader-v0-testmod.mixins.json b/fabric-resource-loader-v0/src/testmod/resources/fabric-resource-loader-v0-testmod.mixins.json
new file mode 100644
index 000000000..73e4f1c42
--- /dev/null
+++ b/fabric-resource-loader-v0/src/testmod/resources/fabric-resource-loader-v0-testmod.mixins.json
@@ -0,0 +1,13 @@
+{
+  "required": true,
+  "package": "net.fabricmc.fabric.test.mixin.resource.loader",
+  "compatibilityLevel": "JAVA_17",
+  "mixins": [
+    "SynchronizeRegistriesTaskMixin"
+  ],
+  "server": [
+  ],
+  "injectors": {
+    "defaultRequire": 1
+  }
+}
diff --git a/fabric-resource-loader-v0/src/testmod/resources/fabric.mod.json b/fabric-resource-loader-v0/src/testmod/resources/fabric.mod.json
index 3739e7234..a8456fa2b 100644
--- a/fabric-resource-loader-v0/src/testmod/resources/fabric.mod.json
+++ b/fabric-resource-loader-v0/src/testmod/resources/fabric.mod.json
@@ -17,5 +17,8 @@
     "server": [
       "net.fabricmc.fabric.test.resource.loader.LanguageTestMod"
     ]
-  }
+  },
+  "mixins": [
+    "fabric-resource-loader-v0-testmod.mixins.json"
+  ]
 }
diff --git a/fabric-resource-loader-v0/src/testmod/resources/resourcepacks/test2/data/minecraft/wolf_variant/woods.json b/fabric-resource-loader-v0/src/testmod/resources/resourcepacks/test2/data/minecraft/wolf_variant/woods.json
new file mode 100644
index 000000000..093c17db3
--- /dev/null
+++ b/fabric-resource-loader-v0/src/testmod/resources/resourcepacks/test2/data/minecraft/wolf_variant/woods.json
@@ -0,0 +1,6 @@
+{
+  "wild_texture": "fabric-resource-loader-v0-testmod:entity/wolf/pink",
+  "angry_texture": "fabric-resource-loader-v0-testmod:entity/wolf/pink",
+  "tame_texture": "fabric-resource-loader-v0-testmod:entity/wolf/pink",
+  "biomes": "minecraft:forest"
+}