function  [y] = UGM_Decode_AlphaExpansion(nodePot, edgePot, edgeStruct, decodeFunc,y)
% INPUT
% nodePot(node,class)
% edgePot(class,class,edge) where e is referenced by V,E (must be the same
% between feature engine and inference engine)
%
% OUTPUT
% nodeLabel(node)

[nNodes,maxStates] = size(nodePot);
nEdges = size(edgePot,3);
edgeEnds = edgeStruct.edgeEnds;
V = edgeStruct.V;
E = edgeStruct.E;
nStates = edgeStruct.nStates;
maxState = max(nStates);

% Initialize
if nargin < 4
	decodeFunc = @UGM_Decode_GraphCut;
end
if nargin < 5
	[junk y] = max(nodePot,[],2);
end
y = int32(y);
if edgeStruct.useMex
	pot = UGM_LogConfigurationPotentialC(int32(y),nodePot,edgePot,int32(edgeEnds));
else
	pot = UGM_LogConfigurationPotential(y,nodePot,edgePot,edgeStruct.edgeEnds);
end

% Do Alpha-Expansions until convergence
while 1
	y_old = y;
	
	for s = 1:maxState
		swapPositions = find(y~=s);
		if ~isempty(swapPositions)
			fprintf('Expanding %d\n',s);
			clamped = y;
			clamped(swapPositions) = 0;
			[clampedNP,clampedEP,clampedES] = UGM_makeClampedPotentials(nodePot,edgePot,edgeStruct,clamped);
			
			if edgeStruct.useMex
				[modifiedNP,modifiedEP] = UGM_Decode_AlphaExpansionC(clampedNP,clampedEP,clampedES.edgeEnds,int32(s-1),y,int32(swapPositions));
			else
				nClampedNodes = size(clampedNP,1);
				modifiedNP = zeros(nClampedNodes,2);
				for n = 1:nClampedNodes
					modifiedNP(n,:) = [clampedNP(n,s) clampedNP(n,y(swapPositions(n)))];
				end
				nClampedEdges = size(clampedES.edgeEnds,1);
				modifiedEP = zeros(2,2,nClampedEdges);
				for e = 1:nClampedEdges
					n1 = clampedES.edgeEnds(e,1);
					n2 = clampedES.edgeEnds(e,2);
					modifiedEP(:,:,e) = [clampedEP(s,s,e) clampedEP(s,y(swapPositions(n2)),e)
						clampedEP(y(swapPositions(n1)),s,e) clampedEP(y(swapPositions(n1)),y(swapPositions(n2)),e)];
				end
			end
			clampedES.nStates(:) = 2;
			
			% Decreasing energy of state (2,2) so that edges are
			% sub-modular (does nothing if edges are metric)
			edgeEnergy = -log(modifiedEP);
			edgeEnergy(2,2,:) = min(edgeEnergy(2,2,:),edgeEnergy(1,2,:)+edgeEnergy(2,1,:)-edgeEnergy(1,1,:));
			modifiedEP = exp(-edgeEnergy);
			
			ytmp = decodeFunc(modifiedNP,modifiedEP,clampedES);
			swapInd = find(ytmp == 1);
			
			y2 = y;
			y(swapPositions(swapInd)) = s;
			
		end
	end
	
	if all(y==y_old)
		break;
	end
	pot_old = pot;
	if edgeStruct.useMex
		pot = UGM_LogConfigurationPotentialC(int32(y),nodePot,edgePot,int32(edgeEnds));
	else
		pot = UGM_LogConfigurationPotential(y,nodePot,edgePot,edgeStruct.edgeEnds);
	end
	fprintf('logPot = %f, changes = %d\n',pot,sum(y~=y_old));
	if pot-pot_old == 0
		break;
	end
end
