function [MF_rule, V, c_param] = FCM(input, No_C, ex_w, distance_type, fcm_seed)

[n_samples, n_feat] = size(input);
c_param = 2 / (ex_w - 1);

use_gpu = false;
try
    if gpuDeviceCount > 0
        input = gpuArray(single(input));
        use_gpu = true;
    end
catch
    use_gpu = false;
end

if nargin < 4 || isempty(distance_type)
    distance_type = 1;
end
if nargin < 5
    fcm_seed = [];
end

if ~isempty(fcm_seed)
 
    fcm_stream = RandStream('twister', 'Seed', fcm_seed);
    if use_gpu
 
        U_cpu = rand(fcm_stream, No_C, n_samples, 'single');
        U = gpuArray(U_cpu);
    else
        U = rand(fcm_stream, No_C, n_samples);
    end
else

    if use_gpu
        U = gpuArray.rand(No_C, n_samples, 'single');
    else
        U = rand(No_C, n_samples);
    end
end
U = U ./ sum(U, 1);

max_iter  = 30;
tolerance = 1e-3;

for iter = 1:max_iter


    U_m = U .^ ex_w;
    V = (U_m * input) ./ sum(U_m, 2);

    x2 = sum(input.^2, 2);
    v2 = sum(V.^2, 2)';
    dist2 = x2 + v2 - 2*(input * V');
    dist2 = max(dist2, 0);


    zeroMask = dist2 < eps('single');
    hasZero = any(zeroMask, 2);

    alpha2 = c_param / 2;
    dist2(dist2 < eps('single')) = eps('single');

    W = dist2 .^ (-alpha2);
    Wsum = sum(W, 2);
    U_new = (W ./ Wsum)';

    if any(hasZero)
        idx = find(hasZero);
        for t = 1:numel(idx)
            ii = idx(t);
            jj = find(zeroMask(ii, :), 1, 'first');
            U_new(:, ii) = 0;
            U_new(jj, ii) = 1;
        end
    end

    if max(abs(U(:) - U_new(:))) < tolerance
        U = U_new;
        break;
    end
    U = U_new;
end


if use_gpu
    MF_rule = gather(U');
    V = gather(V);
    MF_rule = double(MF_rule);
    V = double(V);
else
    MF_rule = U';
end
end