mirror of
				https://github.com/pmmp/PocketMine-MP.git
				synced 2025-10-20 15:41:33 +00:00 
			
		
		
		
	Merge branch 'stable' into minor-next
This commit is contained in:
		| @@ -51,3 +51,16 @@ Released 4th May 2023. | |||||||
| ## Fixes | ## Fixes | ||||||
| - Fixed all types of wooden logs appearing as oak in the inventory. | - Fixed all types of wooden logs appearing as oak in the inventory. | ||||||
| - Fixed a performance issue in `BaseInventory->canAddItem()` (missing `continue` causing useless logic to run). | - Fixed a performance issue in `BaseInventory->canAddItem()` (missing `continue` causing useless logic to run). | ||||||
|  |  | ||||||
|  | # 4.20.3 | ||||||
|  | Released 6th May 2023. | ||||||
|  |  | ||||||
|  | ## Improvements | ||||||
|  | - Reduced memory usage of `RuntimeBlockMapping` from 25 MB to 9 MB. Since every thread has its own copy of the block map, this saves a substantial amount of memory. | ||||||
|  |  | ||||||
|  | ## Fixes | ||||||
|  | - Fixed players falling through blocks in spectator mode. | ||||||
|  | - Fixed timings reports getting bloated by prolific usage of `PluginManager->registerEvent()`. | ||||||
|  |   - This was caused by creating a new timings handler for each call, regardless of whether a timer already existed for the given event and callback. | ||||||
|  | - Fixed `Full Server Tick` and other records being missing from timings reports. | ||||||
|  |   - This was caused by timings handler depth not getting reset when timings was disabled and later re-enabled. | ||||||
|   | |||||||
| @@ -31,7 +31,7 @@ use function str_repeat; | |||||||
|  |  | ||||||
| final class VersionInfo{ | final class VersionInfo{ | ||||||
| 	public const NAME = "PocketMine-MP"; | 	public const NAME = "PocketMine-MP"; | ||||||
| 	public const BASE_VERSION = "4.20.3"; | 	public const BASE_VERSION = "4.20.4"; | ||||||
| 	public const IS_DEVELOPMENT_BUILD = true; | 	public const IS_DEVELOPMENT_BUILD = true; | ||||||
| 	public const BUILD_CHANNEL = "stable"; | 	public const BUILD_CHANNEL = "stable"; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -894,6 +894,12 @@ class NetworkSession{ | |||||||
| 			[ | 			[ | ||||||
| 				//TODO: dynamic flying speed! FINALLY!!!!!!!!!!!!!!!!! | 				//TODO: dynamic flying speed! FINALLY!!!!!!!!!!!!!!!!! | ||||||
| 				new AbilitiesLayer(AbilitiesLayer::LAYER_BASE, $boolAbilities, 0.05, 0.1), | 				new AbilitiesLayer(AbilitiesLayer::LAYER_BASE, $boolAbilities, 0.05, 0.1), | ||||||
|  |  | ||||||
|  | 				//TODO: HACK! In 1.19.80, the client starts falling in our faux spectator mode when it clips into a | ||||||
|  | 				//block. I have no idea why this works, since we don't actually use the real spectator mode. | ||||||
|  | 				new AbilitiesLayer(AbilitiesLayer::LAYER_SPECTATOR, [ | ||||||
|  | 					AbilitiesLayer::ABILITY_FLYING => true, | ||||||
|  | 				], null, null) | ||||||
| 			] | 			] | ||||||
| 		))); | 		))); | ||||||
| 	} | 	} | ||||||
|   | |||||||
| @@ -37,7 +37,7 @@ use pocketmine\permission\DefaultPermissions; | |||||||
| use pocketmine\permission\PermissionManager; | use pocketmine\permission\PermissionManager; | ||||||
| use pocketmine\permission\PermissionParser; | use pocketmine\permission\PermissionParser; | ||||||
| use pocketmine\Server; | use pocketmine\Server; | ||||||
| use pocketmine\timings\TimingsHandler; | use pocketmine\timings\Timings; | ||||||
| use pocketmine\utils\AssumptionFailedError; | use pocketmine\utils\AssumptionFailedError; | ||||||
| use pocketmine\utils\Utils; | use pocketmine\utils\Utils; | ||||||
| use Symfony\Component\Filesystem\Path; | use Symfony\Component\Filesystem\Path; | ||||||
| @@ -651,7 +651,7 @@ class PluginManager{ | |||||||
| 			throw new PluginException("Plugin attempted to register event handler " . $handlerName . "() to event " . $event . " while not enabled"); | 			throw new PluginException("Plugin attempted to register event handler " . $handlerName . "() to event " . $event . " while not enabled"); | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		$timings = new TimingsHandler($handlerName . "(" . (new \ReflectionClass($event))->getShortName() . ")", group: $plugin->getDescription()->getFullName()); | 		$timings = Timings::getEventHandlerTimings($event, $handlerName, $plugin->getDescription()->getFullName()); | ||||||
|  |  | ||||||
| 		$registeredListener = new RegisteredListener($handler, $priority, $plugin, $handleCancelled, $timings); | 		$registeredListener = new RegisteredListener($handler, $priority, $plugin, $handleCancelled, $timings); | ||||||
| 		HandlerListManager::global()->getListFor($event)->register($registeredListener); | 		HandlerListManager::global()->getListFor($event)->register($registeredListener); | ||||||
|   | |||||||
| @@ -166,6 +166,8 @@ abstract class Timings{ | |||||||
|  |  | ||||||
| 	/** @var TimingsHandler[] */ | 	/** @var TimingsHandler[] */ | ||||||
| 	private static array $events = []; | 	private static array $events = []; | ||||||
|  | 	/** @var TimingsHandler[][] */ | ||||||
|  | 	private static array $eventHandlers = []; | ||||||
|  |  | ||||||
| 	public static function init() : void{ | 	public static function init() : void{ | ||||||
| 		if(self::$initialized){ | 		if(self::$initialized){ | ||||||
| @@ -336,4 +338,16 @@ abstract class Timings{ | |||||||
|  |  | ||||||
| 		return self::$events[$eventClass]; | 		return self::$events[$eventClass]; | ||||||
| 	} | 	} | ||||||
|  |  | ||||||
|  | 	/** | ||||||
|  | 	 * @phpstan-template TEvent of Event | ||||||
|  | 	 * @phpstan-param class-string<TEvent> $event | ||||||
|  | 	 */ | ||||||
|  | 	public static function getEventHandlerTimings(string $event, string $handlerName, string $group) : TimingsHandler{ | ||||||
|  | 		if(!isset(self::$eventHandlers[$event][$handlerName])){ | ||||||
|  | 			self::$eventHandlers[$event][$handlerName] = new TimingsHandler($handlerName . "(" . self::shortenCoreClassName($event, "pocketmine\\event\\") . ")", group: $group); | ||||||
|  | 		} | ||||||
|  |  | ||||||
|  | 		return self::$eventHandlers[$event][$handlerName]; | ||||||
|  | 	} | ||||||
| } | } | ||||||
|   | |||||||
| @@ -96,7 +96,7 @@ class TimingsHandler{ | |||||||
| 	} | 	} | ||||||
|  |  | ||||||
| 	public static function reload() : void{ | 	public static function reload() : void{ | ||||||
| 		TimingsRecord::clearRecords(); | 		TimingsRecord::reset(); | ||||||
| 		if(self::$enabled){ | 		if(self::$enabled){ | ||||||
| 			self::$timingStart = hrtime(true); | 			self::$timingStart = hrtime(true); | ||||||
| 		} | 		} | ||||||
| @@ -204,8 +204,9 @@ class TimingsHandler{ | |||||||
| 	/** | 	/** | ||||||
| 	 * @internal | 	 * @internal | ||||||
| 	 */ | 	 */ | ||||||
| 	public function destroyCycles() : void{ | 	public function reset() : void{ | ||||||
| 		$this->rootRecord = null; | 		$this->rootRecord = null; | ||||||
| 		$this->recordsByParent = []; | 		$this->recordsByParent = []; | ||||||
|  | 		$this->timingDepth = 0; | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|   | |||||||
| @@ -42,9 +42,12 @@ final class TimingsRecord{ | |||||||
|  |  | ||||||
| 	private static ?self $currentRecord = null; | 	private static ?self $currentRecord = null; | ||||||
|  |  | ||||||
| 	public static function clearRecords() : void{ | 	/** | ||||||
|  | 	 * @internal | ||||||
|  | 	 */ | ||||||
|  | 	public static function reset() : void{ | ||||||
| 		foreach(self::$records as $record){ | 		foreach(self::$records as $record){ | ||||||
| 			$record->handler->destroyCycles(); | 			$record->handler->reset(); | ||||||
| 		} | 		} | ||||||
| 		self::$records = []; | 		self::$records = []; | ||||||
| 		self::$currentRecord = null; | 		self::$currentRecord = null; | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user