441 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
			
		
		
	
	
			441 lines
		
	
	
		
			10 KiB
		
	
	
	
		
			Java
		
	
	
	
	
	
| /*
 | |
|  * This file is part of the L2J Mobius project.
 | |
|  * 
 | |
|  * This program is free software: you can redistribute it and/or modify
 | |
|  * it under the terms of the GNU General Public License as published by
 | |
|  * the Free Software Foundation, either version 3 of the License, or
 | |
|  * (at your option) any later version.
 | |
|  * 
 | |
|  * This program is distributed in the hope that it will be useful,
 | |
|  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
|  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
 | |
|  * General Public License for more details.
 | |
|  * 
 | |
|  * You should have received a copy of the GNU General Public License
 | |
|  * along with this program. If not, see <http://www.gnu.org/licenses/>.
 | |
|  */
 | |
| package com.l2jmobius.gameserver.pathfinding.cellnodes;
 | |
| 
 | |
| import java.util.ArrayList;
 | |
| import java.util.LinkedList;
 | |
| import java.util.List;
 | |
| import java.util.ListIterator;
 | |
| import java.util.concurrent.CopyOnWriteArrayList;
 | |
| import java.util.logging.Level;
 | |
| import java.util.logging.Logger;
 | |
| 
 | |
| import com.l2jmobius.Config;
 | |
| import com.l2jmobius.gameserver.GeoData;
 | |
| import com.l2jmobius.gameserver.idfactory.IdFactory;
 | |
| import com.l2jmobius.gameserver.model.instancezone.Instance;
 | |
| import com.l2jmobius.gameserver.model.itemcontainer.Inventory;
 | |
| import com.l2jmobius.gameserver.model.items.instance.L2ItemInstance;
 | |
| import com.l2jmobius.gameserver.pathfinding.AbstractNode;
 | |
| import com.l2jmobius.gameserver.pathfinding.AbstractNodeLoc;
 | |
| import com.l2jmobius.gameserver.pathfinding.PathFinding;
 | |
| 
 | |
| /**
 | |
|  * @author Sami, DS Credits to Diamond
 | |
|  */
 | |
| public class CellPathFinding extends PathFinding
 | |
| {
 | |
| 	private static final Logger _log = Logger.getLogger(CellPathFinding.class.getName());
 | |
| 	private BufferInfo[] _allBuffers;
 | |
| 	private int _findSuccess = 0;
 | |
| 	private int _findFails = 0;
 | |
| 	private int _postFilterUses = 0;
 | |
| 	private int _postFilterPlayableUses = 0;
 | |
| 	private int _postFilterPasses = 0;
 | |
| 	private long _postFilterElapsed = 0;
 | |
| 	
 | |
| 	private List<L2ItemInstance> _debugItems = null;
 | |
| 	
 | |
| 	public static CellPathFinding getInstance()
 | |
| 	{
 | |
| 		return SingletonHolder._instance;
 | |
| 	}
 | |
| 	
 | |
| 	protected CellPathFinding()
 | |
| 	{
 | |
| 		try
 | |
| 		{
 | |
| 			final String[] array = Config.PATHFIND_BUFFERS.split(";");
 | |
| 			
 | |
| 			_allBuffers = new BufferInfo[array.length];
 | |
| 			
 | |
| 			String buf;
 | |
| 			String[] args;
 | |
| 			for (int i = 0; i < array.length; i++)
 | |
| 			{
 | |
| 				buf = array[i];
 | |
| 				args = buf.split("x");
 | |
| 				if (args.length != 2)
 | |
| 				{
 | |
| 					throw new Exception("Invalid buffer definition: " + buf);
 | |
| 				}
 | |
| 				
 | |
| 				_allBuffers[i] = new BufferInfo(Integer.parseInt(args[0]), Integer.parseInt(args[1]));
 | |
| 			}
 | |
| 		}
 | |
| 		catch (Exception e)
 | |
| 		{
 | |
| 			_log.log(Level.WARNING, "CellPathFinding: Problem during buffer init: " + e.getMessage(), e);
 | |
| 			throw new Error("CellPathFinding: load aborted");
 | |
| 		}
 | |
| 	}
 | |
| 	
 | |
| 	@Override
 | |
| 	public boolean pathNodesExist(short regionoffset)
 | |
| 	{
 | |
| 		return false;
 | |
| 	}
 | |
| 	
 | |
| 	@Override
 | |
| 	public List<AbstractNodeLoc> findPath(int x, int y, int z, int tx, int ty, int tz, Instance instance, boolean playable)
 | |
| 	{
 | |
| 		final int gx = GeoData.getInstance().getGeoX(x);
 | |
| 		final int gy = GeoData.getInstance().getGeoY(y);
 | |
| 		if (!GeoData.getInstance().hasGeo(x, y))
 | |
| 		{
 | |
| 			return null;
 | |
| 		}
 | |
| 		final int gz = GeoData.getInstance().getHeight(x, y, z);
 | |
| 		final int gtx = GeoData.getInstance().getGeoX(tx);
 | |
| 		final int gty = GeoData.getInstance().getGeoY(ty);
 | |
| 		if (!GeoData.getInstance().hasGeo(tx, ty))
 | |
| 		{
 | |
| 			return null;
 | |
| 		}
 | |
| 		final int gtz = GeoData.getInstance().getHeight(tx, ty, tz);
 | |
| 		final CellNodeBuffer buffer = alloc(64 + (2 * Math.max(Math.abs(gx - gtx), Math.abs(gy - gty))), playable);
 | |
| 		if (buffer == null)
 | |
| 		{
 | |
| 			return null;
 | |
| 		}
 | |
| 		
 | |
| 		final boolean debug = playable && Config.DEBUG_PATH;
 | |
| 		
 | |
| 		if (debug)
 | |
| 		{
 | |
| 			if (_debugItems == null)
 | |
| 			{
 | |
| 				_debugItems = new CopyOnWriteArrayList<>();
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				for (L2ItemInstance item : _debugItems)
 | |
| 				{
 | |
| 					if (item == null)
 | |
| 					{
 | |
| 						continue;
 | |
| 					}
 | |
| 					item.decayMe();
 | |
| 				}
 | |
| 				
 | |
| 				_debugItems.clear();
 | |
| 			}
 | |
| 		}
 | |
| 		
 | |
| 		List<AbstractNodeLoc> path = null;
 | |
| 		try
 | |
| 		{
 | |
| 			final CellNode result = buffer.findPath(gx, gy, gz, gtx, gty, gtz);
 | |
| 			
 | |
| 			if (debug)
 | |
| 			{
 | |
| 				for (CellNode n : buffer.debugPath())
 | |
| 				{
 | |
| 					if (n.getCost() < 0)
 | |
| 					{
 | |
| 						dropDebugItem(1831, (int) (-n.getCost() * 10), n.getLoc());
 | |
| 					}
 | |
| 					else
 | |
| 					{
 | |
| 						// known nodes
 | |
| 						dropDebugItem(Inventory.ADENA_ID, (int) (n.getCost() * 10), n.getLoc());
 | |
| 					}
 | |
| 				}
 | |
| 			}
 | |
| 			
 | |
| 			if (result == null)
 | |
| 			{
 | |
| 				_findFails++;
 | |
| 				return null;
 | |
| 			}
 | |
| 			
 | |
| 			path = constructPath(result);
 | |
| 		}
 | |
| 		catch (Exception e)
 | |
| 		{
 | |
| 			_log.log(Level.WARNING, "", e);
 | |
| 			return null;
 | |
| 		}
 | |
| 		finally
 | |
| 		{
 | |
| 			buffer.free();
 | |
| 		}
 | |
| 		
 | |
| 		if ((path.size() < 3) || (Config.MAX_POSTFILTER_PASSES <= 0))
 | |
| 		{
 | |
| 			_findSuccess++;
 | |
| 			return path;
 | |
| 		}
 | |
| 		
 | |
| 		final long timeStamp = System.currentTimeMillis();
 | |
| 		_postFilterUses++;
 | |
| 		if (playable)
 | |
| 		{
 | |
| 			_postFilterPlayableUses++;
 | |
| 		}
 | |
| 		
 | |
| 		int currentX, currentY, currentZ;
 | |
| 		ListIterator<AbstractNodeLoc> middlePoint;
 | |
| 		boolean remove;
 | |
| 		int pass = 0;
 | |
| 		do
 | |
| 		{
 | |
| 			pass++;
 | |
| 			_postFilterPasses++;
 | |
| 			
 | |
| 			remove = false;
 | |
| 			middlePoint = path.listIterator();
 | |
| 			currentX = x;
 | |
| 			currentY = y;
 | |
| 			currentZ = z;
 | |
| 			
 | |
| 			while (middlePoint.hasNext())
 | |
| 			{
 | |
| 				final AbstractNodeLoc locMiddle = middlePoint.next();
 | |
| 				if (!middlePoint.hasNext())
 | |
| 				{
 | |
| 					break;
 | |
| 				}
 | |
| 				
 | |
| 				final AbstractNodeLoc locEnd = path.get(middlePoint.nextIndex());
 | |
| 				if (GeoData.getInstance().canMove(currentX, currentY, currentZ, locEnd.getX(), locEnd.getY(), locEnd.getZ(), instance))
 | |
| 				{
 | |
| 					middlePoint.remove();
 | |
| 					remove = true;
 | |
| 					if (debug)
 | |
| 					{
 | |
| 						dropDebugItem(735, 1, locMiddle);
 | |
| 					}
 | |
| 				}
 | |
| 				else
 | |
| 				{
 | |
| 					currentX = locMiddle.getX();
 | |
| 					currentY = locMiddle.getY();
 | |
| 					currentZ = locMiddle.getZ();
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		// only one postfilter pass for AI
 | |
| 		while (playable && remove && (path.size() > 2) && (pass < Config.MAX_POSTFILTER_PASSES));
 | |
| 		
 | |
| 		if (debug)
 | |
| 		{
 | |
| 			path.forEach(n -> dropDebugItem(65, 1, n));
 | |
| 		}
 | |
| 		
 | |
| 		_findSuccess++;
 | |
| 		_postFilterElapsed += System.currentTimeMillis() - timeStamp;
 | |
| 		return path;
 | |
| 	}
 | |
| 	
 | |
| 	private List<AbstractNodeLoc> constructPath(AbstractNode<NodeLoc> node)
 | |
| 	{
 | |
| 		final LinkedList<AbstractNodeLoc> path = new LinkedList<>();
 | |
| 		int previousDirectionX = Integer.MIN_VALUE;
 | |
| 		int previousDirectionY = Integer.MIN_VALUE;
 | |
| 		int directionX, directionY;
 | |
| 		
 | |
| 		while (node.getParent() != null)
 | |
| 		{
 | |
| 			if (!Config.ADVANCED_DIAGONAL_STRATEGY && (node.getParent().getParent() != null))
 | |
| 			{
 | |
| 				final int tmpX = node.getLoc().getNodeX() - node.getParent().getParent().getLoc().getNodeX();
 | |
| 				final int tmpY = node.getLoc().getNodeY() - node.getParent().getParent().getLoc().getNodeY();
 | |
| 				if (Math.abs(tmpX) == Math.abs(tmpY))
 | |
| 				{
 | |
| 					directionX = tmpX;
 | |
| 					directionY = tmpY;
 | |
| 				}
 | |
| 				else
 | |
| 				{
 | |
| 					directionX = node.getLoc().getNodeX() - node.getParent().getLoc().getNodeX();
 | |
| 					directionY = node.getLoc().getNodeY() - node.getParent().getLoc().getNodeY();
 | |
| 				}
 | |
| 			}
 | |
| 			else
 | |
| 			{
 | |
| 				directionX = node.getLoc().getNodeX() - node.getParent().getLoc().getNodeX();
 | |
| 				directionY = node.getLoc().getNodeY() - node.getParent().getLoc().getNodeY();
 | |
| 			}
 | |
| 			
 | |
| 			// only add a new route point if moving direction changes
 | |
| 			if ((directionX != previousDirectionX) || (directionY != previousDirectionY))
 | |
| 			{
 | |
| 				previousDirectionX = directionX;
 | |
| 				previousDirectionY = directionY;
 | |
| 				
 | |
| 				path.addFirst(node.getLoc());
 | |
| 				node.setLoc(null);
 | |
| 			}
 | |
| 			
 | |
| 			node = node.getParent();
 | |
| 		}
 | |
| 		
 | |
| 		return path;
 | |
| 	}
 | |
| 	
 | |
| 	private CellNodeBuffer alloc(int size, boolean playable)
 | |
| 	{
 | |
| 		CellNodeBuffer current = null;
 | |
| 		for (BufferInfo i : _allBuffers)
 | |
| 		{
 | |
| 			if (i.mapSize >= size)
 | |
| 			{
 | |
| 				for (CellNodeBuffer buf : i.bufs)
 | |
| 				{
 | |
| 					if (buf.lock())
 | |
| 					{
 | |
| 						i.uses++;
 | |
| 						if (playable)
 | |
| 						{
 | |
| 							i.playableUses++;
 | |
| 						}
 | |
| 						i.elapsed += buf.getElapsedTime();
 | |
| 						current = buf;
 | |
| 						break;
 | |
| 					}
 | |
| 				}
 | |
| 				if (current != null)
 | |
| 				{
 | |
| 					break;
 | |
| 				}
 | |
| 				
 | |
| 				// not found, allocate temporary buffer
 | |
| 				current = new CellNodeBuffer(i.mapSize);
 | |
| 				current.lock();
 | |
| 				if (i.bufs.size() < i.count)
 | |
| 				{
 | |
| 					i.bufs.add(current);
 | |
| 					i.uses++;
 | |
| 					if (playable)
 | |
| 					{
 | |
| 						i.playableUses++;
 | |
| 					}
 | |
| 					break;
 | |
| 				}
 | |
| 				
 | |
| 				i.overflows++;
 | |
| 				if (playable)
 | |
| 				{
 | |
| 					i.playableOverflows++;
 | |
| 					// System.err.println("Overflow, size requested: " + size + " playable:"+playable);
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		
 | |
| 		return current;
 | |
| 	}
 | |
| 	
 | |
| 	private void dropDebugItem(int itemId, int num, AbstractNodeLoc loc)
 | |
| 	{
 | |
| 		final L2ItemInstance item = new L2ItemInstance(IdFactory.getInstance().getNextId(), itemId);
 | |
| 		item.setCount(num);
 | |
| 		item.spawnMe(loc.getX(), loc.getY(), loc.getZ());
 | |
| 		_debugItems.add(item);
 | |
| 	}
 | |
| 	
 | |
| 	private static final class BufferInfo
 | |
| 	{
 | |
| 		final int mapSize;
 | |
| 		final int count;
 | |
| 		ArrayList<CellNodeBuffer> bufs;
 | |
| 		int uses = 0;
 | |
| 		int playableUses = 0;
 | |
| 		int overflows = 0;
 | |
| 		int playableOverflows = 0;
 | |
| 		long elapsed = 0;
 | |
| 		
 | |
| 		public BufferInfo(int size, int cnt)
 | |
| 		{
 | |
| 			mapSize = size;
 | |
| 			count = cnt;
 | |
| 			bufs = new ArrayList<>(count);
 | |
| 		}
 | |
| 		
 | |
| 		@Override
 | |
| 		public String toString()
 | |
| 		{
 | |
| 			final StringBuilder sb = new StringBuilder(100);
 | |
| 			sb.append(mapSize);
 | |
| 			sb.append("x");
 | |
| 			sb.append(mapSize);
 | |
| 			sb.append(" num:");
 | |
| 			sb.append(bufs.size());
 | |
| 			sb.append("/");
 | |
| 			sb.append(count);
 | |
| 			sb.append(" uses:");
 | |
| 			sb.append(uses);
 | |
| 			sb.append("/");
 | |
| 			sb.append(playableUses);
 | |
| 			if (uses > 0)
 | |
| 			{
 | |
| 				sb.append(" total/avg(ms):");
 | |
| 				sb.append(elapsed);
 | |
| 				sb.append("/");
 | |
| 				sb.append(String.format("%1.2f", (double) elapsed / uses));
 | |
| 			}
 | |
| 			
 | |
| 			sb.append(" ovf:");
 | |
| 			sb.append(overflows);
 | |
| 			sb.append("/");
 | |
| 			sb.append(playableOverflows);
 | |
| 			
 | |
| 			return sb.toString();
 | |
| 		}
 | |
| 	}
 | |
| 	
 | |
| 	@Override
 | |
| 	public String[] getStat()
 | |
| 	{
 | |
| 		final String[] result = new String[_allBuffers.length + 1];
 | |
| 		for (int i = 0; i < _allBuffers.length; i++)
 | |
| 		{
 | |
| 			result[i] = _allBuffers[i].toString();
 | |
| 		}
 | |
| 		
 | |
| 		final StringBuilder sb = new StringBuilder(128);
 | |
| 		sb.append("LOS postfilter uses:");
 | |
| 		sb.append(_postFilterUses);
 | |
| 		sb.append("/");
 | |
| 		sb.append(_postFilterPlayableUses);
 | |
| 		if (_postFilterUses > 0)
 | |
| 		{
 | |
| 			sb.append(" total/avg(ms):");
 | |
| 			sb.append(_postFilterElapsed);
 | |
| 			sb.append("/");
 | |
| 			sb.append(String.format("%1.2f", (double) _postFilterElapsed / _postFilterUses));
 | |
| 			sb.append(" passes total/avg:");
 | |
| 			sb.append(_postFilterPasses);
 | |
| 			sb.append("/");
 | |
| 			sb.append(String.format("%1.1f", (double) _postFilterPasses / _postFilterUses));
 | |
| 			sb.append(Config.EOL);
 | |
| 		}
 | |
| 		sb.append("Pathfind success/fail:");
 | |
| 		sb.append(_findSuccess);
 | |
| 		sb.append("/");
 | |
| 		sb.append(_findFails);
 | |
| 		result[result.length - 1] = sb.toString();
 | |
| 		
 | |
| 		return result;
 | |
| 	}
 | |
| 	
 | |
| 	private static class SingletonHolder
 | |
| 	{
 | |
| 		protected static final CellPathFinding _instance = new CellPathFinding();
 | |
| 	}
 | |
| }
 | 
