program casino;
Uses sysutils;
{$H+}
const lung=1000000;
var N,M,C,w,v,coppie,index:Int64;
temp: AnsiString;
S,S_ruotate:array[0..lung] of AnsiString;
funz_errore : array[0..2000000] of Int64;
function LexicalMinRotation(var x: AnsiString):Int64;
var
len,K,i,j:Int64;
begin
temp:=x+x;
len:=length(x);
K:=1;
for j:=2 to 2*len do
begin
i:=funz_errore[j - k ];
while (i <> -1 ) and (x[j mod len] <> x[(k + i ) mod len]) do
if x[j mod len] < x[(k + i ) mod len] then k:= j - i - 1
else i:= funz_errore[i];
if (i = -1) and (s[j mod len] <> s[(k + i+1 ) mod len]) then
begin
if s[j mod len] < s[(k + i+1 ) mod len] then k:= j
else funz_errore[j - k+1]:= -1;
end
else
funz_errore[j - k+1]:= i + 1;
end;
LexicalMinRotation:=k;
end;
begin
(*assign(input, 'input.txt'); reset(input);
assign(output, 'output.txt'); rewrite(output);*)
readln (N,M);
for w:=0 to N-1 do readln(S[w]);
coppie:=0;
for w:=0 to N-1 do
begin
for v:=0 to 2*M do funz_errore[v]:=-1; (*inizializzo la funzione errore a -1*)
index:=LexicalMinRotation(S[w]);
writeln(temp);
S_ruotate[w]:=copy(temp,index+1,M);
writeln(S_ruotate[w]);
end;
for w:=0 to N-2 do
for v:=w+1 to N-1 do
begin
C:= CompareStr(S_ruotate[w], S_ruotate[v]);
if C=0 then coppie:=coppie+1;
end;
writeln (coppie);
end.
cHJvZ3JhbSBjYXNpbm87ClVzZXMgc3lzdXRpbHM7CnskSCt9CmNvbnN0IGx1bmc9MTAwMDAwMDsKdmFyICBOLE0sQyx3LHYsY29wcGllLGluZGV4OkludDY0OwogICAgIHRlbXA6IEFuc2lTdHJpbmc7CiAgICAgUyxTX3J1b3RhdGU6YXJyYXlbMC4ubHVuZ10gb2YgQW5zaVN0cmluZzsKICAgICBmdW56X2Vycm9yZSA6IGFycmF5WzAuLjIwMDAwMDBdIG9mIEludDY0OwpmdW5jdGlvbiBMZXhpY2FsTWluUm90YXRpb24odmFyIHg6IEFuc2lTdHJpbmcpOkludDY0Owp2YXIgCmxlbixLLGksajpJbnQ2NDsKCmJlZ2luCiAgIHRlbXA6PXgreDsKICAgbGVuOj1sZW5ndGgoeCk7CiAgIEs6PTE7CiAgIGZvciBqOj0yIHRvIDIqbGVuIGRvCiAgICAgICAgICAgYmVnaW4KICAgICAgICAgICAgIGk6PWZ1bnpfZXJyb3JlW2ogLSBrIF07CiAgICAgICAgICAgICB3aGlsZSAoaSA8PiAtMSApIGFuZCAoeFtqIG1vZCBsZW5dIDw+IHhbKGsgKyBpICkgbW9kIGxlbl0pIGRvCiAgICAgICAgICAgICAgICAgICAgIGlmIHhbaiBtb2QgbGVuXSA8IHhbKGsgKyBpICkgbW9kIGxlbl0gdGhlbiBrOj0gaiAtIGkgLSAxCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZWxzZSBpOj0gZnVuel9lcnJvcmVbaV07CiAgICAgICAgICAgIGlmIChpID0gLTEpIGFuZCAoc1tqIG1vZCBsZW5dIDw+IHNbKGsgKyBpKzEgKSBtb2QgbGVuXSkgdGhlbgogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJlZ2luCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgaWYgc1tqIG1vZCBsZW5dIDwgc1soayArIGkrMSApIG1vZCBsZW5dIHRoZW4gazo9IGoKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZWxzZSBmdW56X2Vycm9yZVtqIC0gaysxXTo9IC0xOwogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGVuZAogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGVsc2UKICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgZnVuel9lcnJvcmVbaiAtIGsrMV06PSBpICsgMTsKICAgICAgICAgICAgICAgCiAgICAgICAgIGVuZDsgICAKIExleGljYWxNaW5Sb3RhdGlvbjo9azsKICAgICAKZW5kOwoKYmVnaW4KICAgKCphc3NpZ24oaW5wdXQsICdpbnB1dC50eHQnKTsgcmVzZXQoaW5wdXQpOwogICBhc3NpZ24ob3V0cHV0LCAnb3V0cHV0LnR4dCcpOyByZXdyaXRlKG91dHB1dCk7KikKICAgcmVhZGxuIChOLE0pOwogICBmb3Igdzo9MCB0byBOLTEgZG8gcmVhZGxuKFNbd10pOyAKICAgCiAgIAogICBjb3BwaWU6PTA7IAogICBmb3Igdzo9MCB0byBOLTEgZG8KICAgICAgICAgIGJlZ2luCiAgICAgICAgICAgIGZvciB2Oj0wIHRvIDIqTSBkbyBmdW56X2Vycm9yZVt2XTo9LTE7ICgqaW5pemlhbGl6em8gbGEgZnVuemlvbmUgZXJyb3JlIGEgLTEqKQogICAgICAgICAgICBpbmRleDo9TGV4aWNhbE1pblJvdGF0aW9uKFNbd10pOwogICAgICAgICAgICB3cml0ZWxuKHRlbXApOwogICAgICAgICAgICBTX3J1b3RhdGVbd106PWNvcHkodGVtcCxpbmRleCsxLE0pOwogICAgICAgICAgICB3cml0ZWxuKFNfcnVvdGF0ZVt3XSk7CiAgICAgICAgICBlbmQ7IAogICAgZm9yIHc6PTAgdG8gTi0yIGRvICAKICAgICAgICAgICAgIGZvciB2Oj13KzEgdG8gTi0xIGRvCiAgICAgICAgICAgICAgICAgICAgICAgIGJlZ2luCiAgICAgICAgICAgICAgICAgICAgICAgICAgQzo9IENvbXBhcmVTdHIoU19ydW90YXRlW3ddLCBTX3J1b3RhdGVbdl0pOwogICAgICAgICAgICAgICAgICAgICAgICAgIGlmIEM9MCB0aGVuIGNvcHBpZTo9Y29wcGllKzE7CiAgICAgICAgICAgICAgICAgICAgICAgIGVuZDsgCiAgICB3cml0ZWxuIChjb3BwaWUpOwplbmQu