diff --git a/src/action/action_patrol.cpp b/src/action/action_patrol.cpp
index 60d827500..5b17a2085 100644
--- a/src/action/action_patrol.cpp
+++ b/src/action/action_patrol.cpp
@@ -75,10 +75,12 @@ void HandleActionPatrol(Unit* unit)
 		//
 		// Swap the points.
 		//
-		tmp = (int)unit->Orders[0].Arg1;
-		unit->Orders[0].Arg1 = (void*)((unit->Orders[0].X << 16) | unit->Orders[0].Y);
-		unit->Orders[0].X = tmp >> 16;
-		unit->Orders[0].Y = tmp & 0xFFFF;
+		tmp = unit->Orders[0].Arg1.Patrol.X;
+		unit->Orders[0].Arg1.Patrol.X = unit->Orders[0].X;
+		unit->Orders[0].X = tmp;
+		tmp = unit->Orders[0].Arg1.Patrol.Y;
+		unit->Orders[0].Arg1.Patrol.Y = unit->Orders[0].Y;
+		unit->Orders[0].Y = tmp;
 
 		NewResetPath(unit);
 	}
diff --git a/src/action/action_research.cpp b/src/action/action_research.cpp
index cbae3c770..d63a60668 100644
--- a/src/action/action_research.cpp
+++ b/src/action/action_research.cpp
@@ -64,7 +64,7 @@ void HandleActionResearch(Unit* unit)
 	const Upgrade* upgrade;
 
 	if (!unit->SubAction) { // first entry
-		upgrade = unit->Data.Research.Upgrade = unit->Orders[0].Arg1;
+		upgrade = unit->Data.Research.Upgrade = unit->Orders[0].Arg1.Upgrade;
 #if 0
 		// FIXME: I want to support both, but with network we need this check
 		//  but if want combined upgrades this is worse
@@ -72,7 +72,7 @@ void HandleActionResearch(Unit* unit)
 		//
 		// Check if an other building has already started?
 		//
-		if (unit->Player->UpgradeTimers.Upgrades[upgrade-Upgrades]) {
+		if (unit->Player->UpgradeTimers.Upgrades[upgrade - Upgrades]) {
 			DebugPrint("Two researches running\n");
 			PlayerAddCosts(unit->Player, upgrade->Costs);
 
diff --git a/src/action/action_resource.cpp b/src/action/action_resource.cpp
index 5e550466d..753c16a45 100644
--- a/src/action/action_resource.cpp
+++ b/src/action/action_resource.cpp
@@ -289,7 +289,7 @@ static void LoseResource(Unit* unit, const Unit* source)
 		//
 		// Remember were it mined, so it can look around for another resource.
 		//
-		unit->Orders[0].Arg1 = (void*)((unit->X << 16) | unit->Y);
+		unit->Orders[0].Arg1.ResourcePos = (unit->X << 16) | unit->Y;
 		unit->Orders[0].Goal = depot;
 		RefsIncrease(depot);
 		NewResetPath(unit);
@@ -495,7 +495,7 @@ static int StopGathering(Unit* unit)
 
 	// Store resource position.
 	// FIXME: is this the best way?
-	unit->Orders[0].Arg1 = (void*)((unit->X << 16) | unit->Y);
+	unit->Orders[0].Arg1.ResourcePos = (unit->X << 16) | unit->Y;
 
 #ifdef DEBUG
 	if (!unit->ResourcesHeld) {
@@ -636,12 +636,12 @@ static int WaitInDepot(Unit* unit)
 	Assert(depot);
 	// Could be destroyed, but then we couldn't be in?
 
-	if (unit->Orders[0].Arg1 == (void*)-1) {
+	if (unit->Orders[0].Arg1.ResourcePos == -1) {
 		x = unit->X;
 		y = unit->Y;
 	} else {
-		x = (int)unit->Orders[0].Arg1 >> 16;
-		y = (int)unit->Orders[0].Arg1 & 0xFFFF;
+		x = unit->Orders[0].Arg1.ResourcePos >> 16;
+		y = unit->Orders[0].Arg1.ResourcePos & 0xFFFF;
 	}
 	// Range hardcoded. don't stray too far though
 	if (resinfo->TerrainHarvester) {
@@ -684,6 +684,10 @@ static int WaitInDepot(Unit* unit)
 void ResourceGiveUp(Unit* unit)
 {
 	DebugPrint("Unit %d gave up on resource gathering.\n" _C_ unit->Slot);
+	if (unit->Orders[0].Goal) {
+		RefsDecrease(unit->Orders->Goal);
+	}
+	memset(unit->Orders, 0, sizeof(*unit->Orders));
 	unit->Orders[0].Action = UnitActionStill;
 	unit->Wait = 1;
 	unit->Reset = 1;
@@ -695,10 +699,6 @@ void ResourceGiveUp(Unit* unit)
 		unit->ResourcesHeld = 0;
 		unit->CurrentResource = 0;
 	}
-	if (unit->Orders[0].Goal) {
-		RefsDecrease(unit->Orders->Goal);
-		unit->Orders[0].Goal = NoUnitP;
-	}
 }
 
 /**
diff --git a/src/action/action_returngoods.cpp b/src/action/action_returngoods.cpp
index e79f99baf..574c3476f 100644
--- a/src/action/action_returngoods.cpp
+++ b/src/action/action_returngoods.cpp
@@ -36,6 +36,7 @@
 
 #include <stdio.h>
 #include <stdlib.h>
+#include <string.h>
 
 #include "stratagus.h"
 #include "video.h"
@@ -73,12 +74,20 @@ void HandleActionReturnGoods(Unit* unit)
 				unit->Type->ResInfo[unit->CurrentResource]->LoseResources)) {
 		DebugPrint("Unit can't return resources, it doesn't carry any.\n");
 		NotifyPlayer(unit->Player, NotifyYellow, unit->X, unit->Y, "No Resources to Return.");
+
+		if (unit->Orders[0].Goal) { // Depot (if not destroyed)
+			RefsDecrease(unit->Orders[0].Goal);
+		}
+		memset(unit->Orders, 0, sizeof(*unit->Orders));
 		unit->Orders[0].Action = UnitActionStill;
 		return;
 	}
+	// If depot was destroyed.
+	// Search for an another one.
 	if (!unit->Orders[0].Goal) {
 		if (!(destu = FindDeposit(unit, unit->X, unit->Y, 1000,
 				unit->CurrentResource))) {
+			memset(unit->Orders, 0, sizeof(*unit->Orders));
 			unit->Orders[0].Action = UnitActionStill;
 			return;
 		}
@@ -89,9 +98,9 @@ void HandleActionReturnGoods(Unit* unit)
 	unit->Orders[0].Action = UnitActionResource;
 	// Somewhere on the way the loaded worker could have change Arg1
 	// Bummer, go get the closest resource to the depot
-	unit->Orders[0].Arg1 = (void*)-1;
+	unit->Orders[0].Arg1.ResourcePos = -1;
 	NewResetPath(unit);
-	unit->SubAction = 70;
+	unit->SubAction = 70; // FIXME : Define value.
 	unit->Wait = 1;
 	return;
 }
diff --git a/src/action/action_spellcast.cpp b/src/action/action_spellcast.cpp
index afc4cf0a4..763202f88 100644
--- a/src/action/action_spellcast.cpp
+++ b/src/action/action_spellcast.cpp
@@ -133,9 +133,9 @@ static void SpellMoveToTarget(Unit* unit)
 			// FIXME: buildings could have directions
 			UnitHeadingFromDeltaXY(unit,
 				unit->Orders[0].X +
-					((SpellType*)unit->Orders[0].Arg1)->Range - unit->X,
+					unit->Orders[0].Arg1.Spell->Range - unit->X,
 				unit->Orders[0].Y +
-					((SpellType*)unit->Orders[0].Arg1)->Range - unit->Y);
+					unit->Orders[0].Arg1.Spell->Range - unit->Y);
 		}
 		unit->SubAction++; // cast the spell
 		return;
@@ -145,7 +145,7 @@ static void SpellMoveToTarget(Unit* unit)
 		// just as close as possible, since the spell is centered
 		// on the caster anyway.
 		//
-		if ((spell = unit->Orders[0].Arg1)->Target == TargetSelf) {
+		if ((spell = unit->Orders[0].Arg1.Spell)->Target == TargetSelf) {
 			DebugPrint("Increase range for spellcast.");
 			unit->Orders->Range++;
 		} else {
@@ -179,7 +179,7 @@ void HandleActionSpellCast(Unit* unit)
 			//
 			// Check if we can cast the spell.
 			//
-			spell = unit->Orders[0].Arg1;
+			spell = unit->Orders[0].Arg1.Spell;
 			if (!CanCastSpell(unit, spell, unit->Orders[0].Goal,
 					unit->Orders[0].X, unit->Orders[0].Y)) {
 
@@ -214,7 +214,7 @@ void HandleActionSpellCast(Unit* unit)
 			unit->SubAction = 1;
 			// FALL THROUGH
 		case 1:                         // Move to the target.
-			if ((spell = unit->Orders[0].Arg1)->Range != INFINITE_RANGE) {
+			if ((spell = unit->Orders[0].Arg1.Spell)->Range != INFINITE_RANGE) {
 				SpellMoveToTarget(unit);
 				break;
 			} else {
@@ -230,7 +230,7 @@ void HandleActionSpellCast(Unit* unit)
 					if (unit->Orders[0].Goal && !UnitVisibleAsGoal(unit->Orders->Goal, unit->Player)) {
 						unit->ReCast = 0;
 					} else {
-						spell = unit->Orders[0].Arg1;
+						spell = unit->Orders[0].Arg1.Spell;
 						unit->ReCast = SpellCast(unit, spell, unit->Orders[0].Goal,
 							unit->Orders[0].X, unit->Orders[0].Y);
 					}
@@ -243,7 +243,7 @@ void HandleActionSpellCast(Unit* unit)
 				if (unit->Orders[0].Goal && !UnitVisibleAsGoal(unit->Orders->Goal, unit->Player)) {
 					unit->ReCast = 0;
 				} else {
-					spell = unit->Orders[0].Arg1;
+					spell = unit->Orders[0].Arg1.Spell;
 					unit->ReCast = SpellCast(unit, spell, unit->Orders[0].Goal,
 						unit->Orders[0].X, unit->Orders[0].Y);
 				}
diff --git a/src/action/command.cpp b/src/action/command.cpp
index 2c31aa5ff..2a5405b14 100644
--- a/src/action/command.cpp
+++ b/src/action/command.cpp
@@ -130,6 +130,7 @@ static void RemoveOrder(Unit* unit, int order)
 {
 	int i;
 	
+	Assert(0 <= order && order < unit->OrderCount);
 	i = order;
 	while(i < unit->OrderCount - 1) {
 		unit->Orders[i] = unit->Orders[i + 1];
@@ -139,11 +140,10 @@ static void RemoveOrder(Unit* unit, int order)
 	if (unit->OrderCount > 1) {
 		--unit->OrderCount;
 	} else {
+		Assert(i == 0);
+		memset(unit->Orders + i, 0, sizeof(*unit->Orders));
 		unit->Orders[i].Action = UnitActionStill;
 		unit->Orders[i].X = unit->Orders[i].Y = -1;
-		unit->SubAction = 0;
-		unit->Orders[i].Type = NULL;
-		unit->Orders[i].Arg1 = NULL;
 	}
 }
 
@@ -159,10 +159,9 @@ static void ClearSavedAction(Unit* unit)
 {
 	ReleaseOrder(&unit->SavedOrder);
 
+	memset(&unit->SavedOrder, 0, sizeof(unit->SavedOrder));
 	unit->SavedOrder.Action = UnitActionStill; // clear saved action
 	unit->SavedOrder.X = unit->SavedOrder.Y = -1;
-	unit->SavedOrder.Type = NULL;
-	unit->SavedOrder.Arg1 = NULL;
 }
 
 /*----------------------------------------------------------------------------
@@ -181,11 +180,11 @@ void CommandStopUnit(Unit* unit)
 	// Ignore that the unit could be removed.
 
 	order = GetNextOrder(unit, FlushCommands); // Flush them.
+	Assert(order);
+	memset(order, 0, sizeof(*order));
+
 	order->Action = UnitActionStill;
 	order->X = order->Y = -1;
-	order->Goal = NoUnitP;
-	order->Type = NULL;
-	order->Arg1 = NULL;
 	ReleaseOrder(&unit->SavedOrder);
 	ReleaseOrder(&unit->NewOrder);
 	unit->SavedOrder = unit->NewOrder = *order;
@@ -267,11 +266,9 @@ void CommandStandGround(Unit* unit, int flush)
 	} else if (!(order = GetNextOrder(unit, flush))) {
 		return;
 	}
+	memset(order, 0, sizeof(*order));
 	order->Action = UnitActionStandGround;
 	order->X = order->Y = -1;
-	order->Goal = NoUnitP;
-	order->Type = NULL;
-	order->Arg1 = NULL;
 	ClearSavedAction(unit);
 }
 
@@ -297,6 +294,7 @@ void CommandFollow(Unit* unit, Unit* dest, int flush)
 		} else if (!(order = GetNextOrder(unit, flush))) {
 			return;
 		}
+		memset(order, 0, sizeof(*order));
 
 		order->Action = UnitActionFollow;
 		//
@@ -307,16 +305,12 @@ void CommandFollow(Unit* unit, Unit* dest, int flush)
 		if (dest->Destroyed) {
 			order->X = dest->X + dest->Type->TileWidth / 2;
 			order->Y = dest->Y + dest->Type->TileHeight / 2;
-			order->Goal = NoUnitP;
-			order->Range = 0;
 		} else {
 			order->X = order->Y = -1;
 			order->Goal = dest;
 			RefsIncrease(dest);
 			order->Range = 1;
 		}
-		order->Type = NULL;
-		order->Arg1 = NULL;
 	}
 	ClearSavedAction(unit);
 }
@@ -346,14 +340,12 @@ void CommandMove(Unit* unit, int x, int y, int flush)
 		} else if (!(order = GetNextOrder(unit, flush))) {
 			return;
 		}
+		memset(order, 0, sizeof(*order));
 
 		order->Action = UnitActionMove;
 		order->Goal = NoUnitP;
 		order->X = x;
 		order->Y = y;
-		order->Range = 0;
-		order->Type = NULL;
-		order->Arg1 = NULL;
 	}
 	ClearSavedAction(unit);
 }
@@ -382,6 +374,7 @@ void CommandRepair(Unit* unit, int x, int y, Unit* dest, int flush)
 		} else if (!(order = GetNextOrder(unit, flush))) {
 			return;
 		}
+		memset(order, 0, sizeof(*order));
 
 		order->Action = UnitActionRepair;
 		//
@@ -393,12 +386,8 @@ void CommandRepair(Unit* unit, int x, int y, Unit* dest, int flush)
 			if (dest->Destroyed) {
 				order->X = dest->X + dest->Type->TileWidth / 2;
 				order->Y = dest->Y + dest->Type->TileHeight / 2;
-				order->Goal = NoUnitP;
-				order->Range = 0;
-				order->Width = order->Height = 0;
 			} else {
 				order->X = order->Y = -1;
-				order->Width = order->Height = 0;
 				order->Goal = dest;
 				RefsIncrease(dest);
 				order->Range = unit->Type->RepairRange;
@@ -406,11 +395,7 @@ void CommandRepair(Unit* unit, int x, int y, Unit* dest, int flush)
 		} else {
 			order->X = x;
 			order->Y = y;
-			order->Goal = NoUnitP;
-			order->Range = 0;
 		}
-		order->Type = NULL;
-		order->Arg1 = NULL;
 	}
 	ClearSavedAction(unit);
 }
@@ -441,6 +426,7 @@ void CommandAttack(Unit* unit, int x, int y, Unit* attack, int flush)
 		} else if (!(order = GetNextOrder(unit, flush))) {
 			return;
 		}
+		memset(order, 0, sizeof(*order));
 
 		order->Action = UnitActionAttack;
 		if (attack) {
@@ -452,8 +438,6 @@ void CommandAttack(Unit* unit, int x, int y, Unit* attack, int flush)
 			if (attack->Destroyed) {
 				order->X = attack->X + attack->Type->TileWidth / 2;
 				order->Y = attack->Y + attack->Type->TileHeight / 2;
-				order->Goal = NoUnitP;
-				order->Range = 0;
 			} else {
 				// Removed, Dying handled by action routine.
 				order->X = order->Y = -1;
@@ -468,16 +452,12 @@ void CommandAttack(Unit* unit, int x, int y, Unit* attack, int flush)
 			order->Y = y;
 			order->Range = unit->Stats->AttackRange;
 			order->MinRange = unit->Type->MinAttackRange;
-			order->Goal = NoUnitP;
 		} else {
 			order->X = x;
 			order->Y = y;
 			order->Range = 0;
 			order->Goal = NoUnitP;
 		}
-		order->Type = NULL;
-		order->Arg1 = NULL;
-
 	}
 	ClearSavedAction(unit);
 }
@@ -507,15 +487,13 @@ void CommandAttackGround(Unit* unit, int x, int y, int flush)
 		} else if (!(order = GetNextOrder(unit, flush))) {
 			return;
 		}
+		memset(order, 0, sizeof(*order));
 
 		order->Action = UnitActionAttackGround;
 		order->X = x;
 		order->Y = y;
 		order->Range = unit->Stats->AttackRange;
 		order->MinRange = unit->Type->MinAttackRange;
-		order->Goal = NoUnitP;
-		order->Type = NULL;
-		order->Arg1 = NULL;
 
 		DebugPrint("FIXME this next\n");
 	}
@@ -549,16 +527,14 @@ void CommandPatrolUnit(Unit* unit, int x, int y, int flush)
 		} else if (!(order = GetNextOrder(unit, flush))) {
 			return;
 		}
+		memset(order, 0, sizeof(*order));
 
 		order->Action = UnitActionPatrol;
-		order->Goal = NoUnitP;
 		order->X = x;
 		order->Y = y;
-		order->Range = 0;
-		order->Type = NULL;
 		Assert(!(unit->X & ~0xFFFF) && !(unit->Y & ~0xFFFF));
-		// FIXME: BUG-ALERT: encode source into arg1 as two 16 bit values!
-		order->Arg1 = (void*)((unit->X << 16) | unit->Y);
+		order->Arg1.Patrol.X = unit->X;
+		order->Arg1.Patrol.Y = unit->Y;
 	}
 	ClearSavedAction(unit);
 }
@@ -594,14 +570,13 @@ void CommandBoard(Unit* unit, Unit* dest, int flush)
 		} else if (!(order = GetNextOrder(unit, flush))) {
 			return;
 		}
+		memset(order, 0, sizeof(*order));
 
 		order->Action = UnitActionBoard;
 		order->X = order->Y = -1;
 		order->Goal = dest;
 		RefsIncrease(dest);
 		order->Range = 1;
-		order->Type = NULL;
-		order->Arg1 = NULL;
 	}
 	ClearSavedAction(unit);
 }
@@ -626,6 +601,7 @@ void CommandUnload(Unit* unit, int x, int y, Unit* what, int flush)
 		if (!(order = GetNextOrder(unit, flush))) {
 			return;
 		}
+		memset(order, 0, sizeof(*order));
 
 		order->Action = UnitActionUnload;
 		order->X = x;
@@ -635,14 +611,10 @@ void CommandUnload(Unit* unit, int x, int y, Unit* what, int flush)
 		// Should be handled in action, but is not possible!
 		// Unit::Refs is used as timeout counter.
 		//
-		order->Goal = NoUnitP;
 		if (what && !what->Destroyed) {
 			order->Goal = what;
 			RefsIncrease(what);
 		}
-		order->Range = 0;
-		order->Type = NULL;
-		order->Arg1 = NULL;
 	}
 	ClearSavedAction(unit);
 }
@@ -672,9 +644,9 @@ void CommandBuildBuilding(Unit* unit, int x, int y,
 		} else if (!(order = GetNextOrder(unit, flush))) {
 			return;
 		}
+		memset(order, 0, sizeof(*order));
 
 		order->Action = UnitActionBuild;
-		order->Goal = NoUnitP;
 		order->X = x;
 		order->Y = y;
 		order->Width = what->TileWidth;
@@ -686,17 +658,12 @@ void CommandBuildBuilding(Unit* unit, int x, int y,
 			if (what->ShoreBuilding && unit->Type->UnitType == UnitTypeLand) {
 					// Peon won't dive :-)
 				order->Range = 1;
-			} else {
-				order->Range = 0;
 			}
 		}
 		order->Type = what;
 		if (what->BuilderOutside) {
 			order->MinRange = 1;
-		} else {
-			order->MinRange = 0;
 		}
-		order->Arg1 = NULL;
 	}
 	ClearSavedAction(unit);
 }
@@ -745,6 +712,7 @@ void CommandResourceLoc(Unit* unit, int x, int y, int flush)
 		} else if (!(order = GetNextOrder(unit, flush))) {
 			return;
 		}
+		memset(order, 0, sizeof(*order));
 
 		order->Action = UnitActionResource;
 
@@ -769,9 +737,6 @@ void CommandResourceLoc(Unit* unit, int x, int y, int flush)
 		order->Y = ny;
 
 		order->Range = 1;
-		order->Goal = NoUnitP;
-		order->Type = NULL;
-		order->Arg1 = NULL;
 	}
 	ClearSavedAction(unit);
 }
@@ -807,14 +772,13 @@ void CommandResource(Unit* unit, Unit* dest, int flush)
 		} else if (!(order = GetNextOrder(unit, flush))) {
 			return;
 		}
+		memset(order, 0, sizeof(*order));
 
 		order->Action = UnitActionResource;
 		order->X = order->Y = -1;
 		order->Goal = dest;
 		RefsIncrease(dest);
 		order->Range = 1;
-		order->Type = NULL;
-		order->Arg1 = NULL;
 	}
 	ClearSavedAction(unit);
 }
@@ -847,10 +811,10 @@ void CommandReturnGoods(Unit* unit, Unit* goal, int flush)
 		} else if (!(order = GetNextOrder(unit, flush))) {
 			return;
 		}
+		memset(order, 0, sizeof(*order));
 
 		order->Action = UnitActionReturnGoods;
 		order->X = order->Y = -1;
-		order->Goal = NoUnitP;
 		//
 		// Destination could be killed. NETWORK!
 		//
@@ -859,8 +823,6 @@ void CommandReturnGoods(Unit* unit, Unit* goal, int flush)
 			RefsIncrease(goal);
 		}
 		order->Range = 1;
-		order->Type = NULL;
-		order->Arg1 = NULL;
 	}
 	ClearSavedAction(unit);
 }
@@ -900,12 +862,11 @@ void CommandTrainUnit(Unit* unit, UnitType* type,
 		if (!(order = GetNextOrder(unit, 0))) {
 			return;
 		}
+		memset(order, 0, sizeof(*order));
 
 		order->Action = UnitActionTrain;
 		order->Type = type;
 		order->X = order->Y = -1;
-		order->Goal = NoUnitP;
-		order->Arg1 = NULL;
 		// FIXME: if you give quick an other order, the resources are lost!
 		PlayerSubUnitType(unit->Player, type);
 	}
@@ -938,7 +899,7 @@ void CommandCancelTraining(Unit* unit, int slot, const UnitType* type)
 				unit->Orders[0].Type->Stats[unit->Player->Player].Costs,
 				CancelTrainingCostsFactor);
 			RemoveOrder(unit, 0);
-		}	
+		}
 		unit->Data.Train.Ticks = 0;
 		unit->Wait = unit->Reset = 1; // immediately start next training
 		if (unit->Player == ThisPlayer && unit->Selected) {
@@ -1007,15 +968,14 @@ void CommandUpgradeTo(Unit* unit, UnitType* type, int flush)
 		if (!(order = GetNextOrder(unit, flush))) {
 			return;
 		}
+		memset(order, 0, sizeof(*order));
 
 		// FIXME: if you give quick an other order, the resources are lost!
 		PlayerSubUnitType(unit->Player, type);
 
 		order->Action = UnitActionUpgradeTo;
 		order->X = order->Y = -1;
-		order->Goal = NoUnitP;
 		order->Type = type;
-		order->Arg1 = NULL;
 	}
 	ClearSavedAction(unit);
 }
@@ -1038,11 +998,10 @@ void CommandCancelUpgradeTo(Unit* unit)
 			unit->Orders[0].Type->Stats[unit->Player->Player].Costs,
 			CancelUpgradeCostsFactor);
 
+		memset(unit->Orders, 0, sizeof(*unit->Orders));
+
 		unit->Orders[0].Action = UnitActionStill;
 		unit->Orders[0].X = unit->Orders[0].Y = -1;
-		unit->Orders[0].Goal = NoUnitP;
-		unit->Orders[0].Type = NULL;
-		unit->Orders[0].Arg1 = NULL;
 
 		unit->SubAction = 0;
 
@@ -1099,15 +1058,14 @@ void CommandResearch(Unit* unit, Upgrade* what, int flush)
 		if (!(order = GetNextOrder(unit, flush))) {
 			return;
 		}
+		memset(order, 0, sizeof(*order));
 
 		// FIXME: if you give quick an other order, the resources are lost!
 		PlayerSubCosts(unit->Player, what->Costs);
 
 		order->Action = UnitActionResearch;
 		order->X = order->Y = -1;
-		order->Goal = NoUnitP;
-		order->Type = NULL;
-		order->Arg1 = what;
+		order->Arg1.Upgrade = what;
 	}
 	ClearSavedAction(unit);
 }
@@ -1132,12 +1090,10 @@ void CommandCancelResearch(Unit* unit)
 
 		PlayerAddCostsFactor(unit->Player,upgrade->Costs,
 			CancelResearchCostsFactor);
+		memset(unit->Orders, 0, sizeof(*unit->Orders));
 
 		unit->Orders[0].Action = UnitActionStill;
 		unit->Orders[0].X = unit->Orders[0].Y = -1;
-		unit->Orders[0].Goal = NoUnitP;
-		unit->Orders[0].Type = NULL;
-		unit->Orders[0].Arg1 = NULL;
 
 		unit->SubAction = 0;
 
@@ -1183,6 +1139,7 @@ void CommandSpellCast(Unit* unit, int x, int y, Unit* dest,
 		if (!(order = GetNextOrder(unit, flush))) {
 			return;
 		}
+		memset(order, 0, sizeof(*order));
 
 		order->Action = UnitActionSpellCast;
 		order->Range = spell->Range;
@@ -1197,9 +1154,7 @@ void CommandSpellCast(Unit* unit, int x, int y, Unit* dest,
 				// FIXME: dest->Type is now set to 0. maybe we shouldn't bother.
 				order->X = dest->X /*+ dest->Type->TileWidth / 2*/  - order->Range;
 				order->Y = dest->Y /*+ dest->Type->TileHeight / 2*/ - order->Range;
-				order->Goal = NoUnitP;
-				order->Range <<= 1;
-				order->Range <<= 1;
+				order->Range <<= 2;
 			} else {
 				order->X = order->Y = -1;
 				order->Goal = dest;
@@ -1210,8 +1165,7 @@ void CommandSpellCast(Unit* unit, int x, int y, Unit* dest,
 			order->Y = y;
 			order->Range <<= 1;
 		}
-		order->Type = NULL;
-		order->Arg1 = spell;
+		order->Arg1.Spell = spell;
 	}
 	ClearSavedAction(unit);
 }
diff --git a/src/include/unit.h b/src/include/unit.h
index d7a75b791..1fe0ade3f 100644
--- a/src/include/unit.h
+++ b/src/include/unit.h
@@ -461,7 +461,16 @@ typedef struct _order_ {
 	int       Y;            ///< or Y tile coordinate of destination
 	struct _unit_type_* Type;///< Unit-type argument
 
-	void* Arg1;             ///< Extra command argument
+	union {
+		struct {
+			int X;                    ///< X position for patroling.
+			int Y;                    ///< Y position for patroling.
+		} Patrol;                     ///< position.
+		int ResourcePos;              ///< ResourcePos == (X<<16 | Y).
+		struct _spell_type_* Spell;   ///< spell when casting.
+		struct _upgrade_* Upgrade;    ///< upgrade.
+		struct _order_* Order;        ///< FIXME : seems to be a hack for free memory.
+	} Arg1;             ///< Extra command argument.
 } Order;
 
 /**
diff --git a/src/unit/script_unit.cpp b/src/unit/script_unit.cpp
index 73f776925..dc3730d35 100644
--- a/src/unit/script_unit.cpp
+++ b/src/unit/script_unit.cpp
@@ -282,57 +282,35 @@ void CclParseOrder(lua_State* l, Order* order)
 			lua_pop(l, 1);
 
 		} else if (!strcmp(value, "patrol")) {
-			int x1;
-			int x2;
-
 			++j;
 			lua_rawgeti(l, -1, j + 1);
 			if (!lua_istable(l, -1) || luaL_getn(l, -1) != 2) {
 				LuaError(l, "incorrect argument");
 			}
 			lua_rawgeti(l, -1, 1);
-			x1 = LuaToNumber(l, -1);
+			order->Arg1.Patrol.X = LuaToNumber(l, -1);
 			lua_pop(l, 1);
 			lua_rawgeti(l, -1, 2);
-			x2 = LuaToNumber(l, -1);
+			order->Arg1.Patrol.Y = LuaToNumber(l, -1);
 			lua_pop(l, 1);
-			order->Arg1 = (void*)((x1 << 16) | x2);
 			lua_pop(l, 1);
 
 		} else if (!strcmp(value, "spell")) {
 			++j;
 			lua_rawgeti(l, -1, j + 1);
-			order->Arg1 = SpellTypeByIdent(LuaToString(l, -1));
+			order->Arg1.Spell = SpellTypeByIdent(LuaToString(l, -1));
 			lua_pop(l, 1);
 
 		} else if (!strcmp(value, "upgrade")) {
 			++j;
 			lua_rawgeti(l, -1, j + 1);
-			order->Arg1 = UpgradeByIdent(LuaToString(l, -1));
+			order->Arg1.Upgrade = UpgradeByIdent(LuaToString(l, -1));
 			lua_pop(l, 1);
 
 		} else if (!strcmp(value, "mine")) {
-			int x1;
-			int x2;
-
 			++j;
 			lua_rawgeti(l, -1, j + 1);
-			if (!lua_istable(l, -1) || luaL_getn(l, -1) != 2) {
-				LuaError(l, "incorrect argument");
-			}
-			lua_rawgeti(l, -1, 1);
-			x1 = LuaToNumber(l, -1);
-			lua_pop(l, 1);
-			lua_rawgeti(l, -1, 2);
-			x2 = LuaToNumber(l, -1);
-			lua_pop(l, 1);
-			order->Arg1 = (void*)((x1 << 16) | x2);
-			lua_pop(l, 1);
-
-		} else if (!strcmp(value, "arg1")) {
-			++j;
-			lua_rawgeti(l, -1, j + 1);
-			order->Arg1 = (void*)(int)LuaToNumber(l, -1);
+			order->Arg1.ResourcePos = LuaToNumber(l, -1);
 			lua_pop(l, 1);
 
 		} else {
diff --git a/src/unit/unit.cpp b/src/unit/unit.cpp
index cb775e084..f5a68235f 100644
--- a/src/unit/unit.cpp
+++ b/src/unit/unit.cpp
@@ -240,12 +240,12 @@ void ReleaseUnit(Unit* unit)
 	free(unit->CacheLinks);
 
 	if (ReleasedOrderHead) {
-		ReleasedOrderTail->Arg1 = unit->Orders;
+		ReleasedOrderTail->Arg1.Order = unit->Orders;
 		ReleasedOrderTail = unit->Orders;
-		unit->Orders->Arg1 = NULL;
+		unit->Orders->Arg1.Order = NULL;
 	} else {
 		ReleasedOrderHead = ReleasedOrderTail = unit->Orders;
-		unit->Orders->Arg1 = NULL;
+		unit->Orders->Arg1.Order = NULL;
 	}
 	unit->Orders->X = GameCycle + (NetworkMaxLag << 1); // could be reuse after this time
 	unit->Orders->Y = unit->TotalOrders; // store order count for when reused
@@ -368,7 +368,7 @@ void InitUnit(Unit* unit, UnitType* type)
 	if (ReleasedOrderHead && (unsigned)ReleasedOrderHead->X < GameCycle) {
 		unit->Orders = ReleasedOrderHead;
 		unit->TotalOrders = ReleasedOrderHead->Y;
-		ReleasedOrderHead = (Order*)ReleasedOrderHead->Arg1;
+		ReleasedOrderHead = ReleasedOrderHead->Arg1.Order;
 	} else {
 		// No Available Orders in Memory, create new ones
 		unit->TotalOrders = DEFAULT_START_ORDERS;
@@ -3547,23 +3547,28 @@ void SaveOrder(const Order* order, CLFile* file)
 	if (order->Type) {
 		CLprintf(file, " \"type\", \"%s\",", order->Type->Ident);
 	}
-	if (order->Arg1) {
-		// patrol=pos, research=upgrade, spell=spell
-		switch (order->Action) {
-			case UnitActionPatrol:
-				CLprintf(file, " \"patrol\", {%d, %d},",
-					(int)order->Arg1 >> 16, (int)order->Arg1 & 0xFFFF);
-				break;
-			case UnitActionSpellCast:
-				CLprintf(file, " \"spell\", \"%s\",", ((SpellType*)order->Arg1)->Ident);
-				break;
-			case UnitActionResearch:
-				CLprintf(file, " \"upgrade\", \"%s\",", ((Upgrade*)order->Arg1)->Ident);
-				break;
-			default:
-				CLprintf(file, " \"arg1\", %d,", (int)order->Arg1);
-				break;
-		}
+	// Extra arg.
+	switch (order->Action) {
+		case UnitActionPatrol:
+			CLprintf(file, " \"patrol\", {%d, %d},",
+				order->Arg1.Patrol.X, order->Arg1.Patrol.Y);
+			break;
+		case UnitActionSpellCast:
+			if (order->Arg1.Spell) {
+				CLprintf(file, " \"spell\", \"%s\",", order->Arg1.Spell->Ident);
+			}
+			break;
+		case UnitActionResearch:
+			if (order->Arg1.Upgrade) {
+				CLprintf(file, " \"upgrade\", \"%s\",", order->Arg1.Upgrade->Ident);
+			}
+			break;
+		case UnitActionResource :
+		case UnitActionReturnGoods :
+			CLprintf(file, " \"mine\", %d,", order->Arg1.ResourcePos);
+			break;
+		default:
+			break;
 	}
 	CLprintf(file, "}");
 }
@@ -3960,7 +3965,7 @@ void CleanUnits(void)
 	//
 	//  Release memory of Orders in the release queue.
 	while ((order = ReleasedOrderHead)) {
-		ReleasedOrderHead = order->Arg1;
+		ReleasedOrderHead = order->Arg1.Order;
 		free(order);
 	}
 
diff --git a/src/unit/unit_draw.cpp b/src/unit/unit_draw.cpp
index 3f0ef3b67..0778f5952 100644
--- a/src/unit/unit_draw.cpp
+++ b/src/unit/unit_draw.cpp
@@ -1438,9 +1438,9 @@ static void ShowSingleOrder(const Unit* unit, int x1, int y1, const Order* order
 			VideoDrawLineClip(ColorGreen, x1, y1, x2, y2);
 			e_color = color = ColorBlue;
 			x1 = Map2ViewportX(CurrentViewport,
-				((int)order->Arg1) >> 16) + TileSizeX / 2;
+				order->Arg1.Patrol.X) + TileSizeX / 2;
 			y1 = Map2ViewportY(CurrentViewport,
-				((int)order->Arg1) & 0xFFFF) + TileSizeY / 2;
+				order->Arg1.Patrol.Y) + TileSizeY / 2;
 			dest = 1;
 			break;