#pragma GCC optimize ("Ofast")
#include<bits/stdc++.h>
using namespace std;
void * wmem;
char memarr[ 96000000 ] ;
template < class T> inline void walloc1d( T ** arr, int x, void ** mem = & wmem) {
static int skip[ 16 ] = { 0 , 15 , 14 , 13 , 12 , 11 , 10 , 9 , 8 , 7 , 6 , 5 , 4 , 3 , 2 , 1 } ;
( * mem) = ( void * ) ( ( ( char * ) ( * mem) ) + skip[ ( ( unsigned long long ) ( * mem) ) & 15 ] ) ;
( * arr) = ( T* ) ( * mem) ;
( * mem) = ( ( * arr) + x) ;
}
template < class T> inline void walloc1d( T ** arr, int x1, int x2, void ** mem = & wmem) {
walloc1d( arr, x2- x1, mem) ;
( * arr) - = x1;
}
template < class T1, class T2, class T3> void sortA_L( int N, T1 a[ ] , T2 b[ ] , T3 c[ ] , void * mem = wmem) {
int i;
pair< T1, pair< T2, T3> > * arr;
walloc1d( & arr, N, & mem) ;
for ( i= ( 0 ) ; i< ( N) ; i++ ) {
arr[ i] .first = a[ i] ;
arr[ i] .second .first = b[ i] ;
arr[ i] .second .second = c[ i] ;
}
sort( arr, arr+ N) ;
for ( i= ( 0 ) ; i< ( N) ; i++ ) {
a[ i] = arr[ i] .first ;
b[ i] = arr[ i] .second .first ;
c[ i] = arr[ i] .second .second ;
}
}
template < class T1, class T2, class T3, class T4> void sortA_L( int N, T1 a[ ] , T2 b[ ] , T3 c[ ] , T4 d[ ] , void * mem = wmem) {
int i;
pair< pair< T1, T2> , pair< T3, T4> > * arr;
walloc1d( & arr, N, & mem) ;
for ( i= ( 0 ) ; i< ( N) ; i++ ) {
arr[ i] .first .first = a[ i] ;
arr[ i] .first .second = b[ i] ;
arr[ i] .second .first = c[ i] ;
arr[ i] .second .second = d[ i] ;
}
sort( arr, arr+ N) ;
for ( i= ( 0 ) ; i< ( N) ; i++ ) {
a[ i] = arr[ i] .first .first ;
b[ i] = arr[ i] .first .second ;
c[ i] = arr[ i] .second .first ;
d[ i] = arr[ i] .second .second ;
}
}
inline int my_getchar_unlocked( ) {
static char buf[ 1048576 ] ;
static int s = 1048576 ;
static int e = 1048576 ;
if ( s == e && e == 1048576 ) {
e = fread_unlocked( buf, 1 , 1048576 , stdin ) ;
s = 0 ;
}
if ( s == e) {
return EOF ;
}
return buf[ s++ ] ;
}
inline void rd( int & x) {
int k;
int m= 0 ;
x= 0 ;
for ( ;; ) {
k = my_getchar_unlocked( ) ;
if ( k== '-' ) {
m= 1 ;
break ;
}
if ( '0' <= k&& k<= '9' ) {
x= k- '0' ;
break ;
}
}
for ( ;; ) {
k = my_getchar_unlocked( ) ;
if ( k< '0' || k> '9' ) {
break ;
}
x= x* 10 + k- '0' ;
}
if ( m) {
x= - x;
}
}
inline void rd( long long & x) {
int k;
int m= 0 ;
x= 0 ;
for ( ;; ) {
k = my_getchar_unlocked( ) ;
if ( k== '-' ) {
m= 1 ;
break ;
}
if ( '0' <= k&& k<= '9' ) {
x= k- '0' ;
break ;
}
}
for ( ;; ) {
k = my_getchar_unlocked( ) ;
if ( k< '0' || k> '9' ) {
break ;
}
x= x* 10 + k- '0' ;
}
if ( m) {
x= - x;
}
}
struct MY_WRITER{
char buf[ 1048576 ] ;
int s;
int e;
MY_WRITER( ) {
s = 0 ;
e = 1048576 ;
}
~MY_WRITER( ) {
if ( s) {
fwrite_unlocked( buf, 1 , s, stdout ) ;
}
}
}
;
MY_WRITER MY_WRITER_VAR;
void my_putchar_unlocked( int a) {
if ( MY_WRITER_VAR.s == MY_WRITER_VAR.e ) {
fwrite_unlocked( MY_WRITER_VAR.buf , 1 , MY_WRITER_VAR.s , stdout ) ;
MY_WRITER_VAR.s = 0 ;
}
MY_WRITER_VAR.buf [ MY_WRITER_VAR.s ++ ] = a;
}
inline void wt_L( char a) {
my_putchar_unlocked( a) ;
}
inline void wt_L( long long x) {
int s= 0 ;
int m= 0 ;
char f[ 20 ] ;
if ( x< 0 ) {
m= 1 ;
x= - x;
}
while ( x) {
f[ s++ ] = x% 10 ;
x/ = 10 ;
}
if ( ! s) {
f[ s++ ] = 0 ;
}
if ( m) {
my_putchar_unlocked( '-' ) ;
}
while ( s-- ) {
my_putchar_unlocked( f[ s] + '0' ) ;
}
}
template < class T, class U> inline T GCD_L( T a, U b) {
T r;
while ( b) {
r= a;
a= b;
b= r% a;
}
return a;
}
template < class S> inline void arrInsert( const int k, int & sz, S a[ ] , const S aval) {
int i;
sz++ ;
for ( i= sz- 1 ; i> k; i-- ) {
a[ i] = a[ i- 1 ] ;
}
a[ k] = aval;
}
template < class S, class T> inline void arrInsert( const int k, int & sz, S a[ ] , const S aval, T b[ ] , const T bval) {
int i;
sz++ ;
for ( i= sz- 1 ; i> k; i-- ) {
a[ i] = a[ i- 1 ] ;
}
for ( i= sz- 1 ; i> k; i-- ) {
b[ i] = b[ i- 1 ] ;
}
a[ k] = aval;
b[ k] = bval;
}
template < class S, class T, class U> inline void arrInsert( const int k, int & sz, S a[ ] , const S aval, T b[ ] , const T bval, U c[ ] , const U cval) {
int i;
sz++ ;
for ( i= sz- 1 ; i> k; i-- ) {
a[ i] = a[ i- 1 ] ;
}
for ( i= sz- 1 ; i> k; i-- ) {
b[ i] = b[ i- 1 ] ;
}
for ( i= sz- 1 ; i> k; i-- ) {
c[ i] = c[ i- 1 ] ;
}
a[ k] = aval;
b[ k] = bval;
c[ k] = cval;
}
template < class S, class T, class U, class V> inline void arrInsert( const int k, int & sz, S a[ ] , const S aval, T b[ ] , const T bval, U c[ ] , const U cval, V d[ ] , const V dval) {
int i;
sz++ ;
for ( i= sz- 1 ; i> k; i-- ) {
a[ i] = a[ i- 1 ] ;
}
for ( i= sz- 1 ; i> k; i-- ) {
b[ i] = b[ i- 1 ] ;
}
for ( i= sz- 1 ; i> k; i-- ) {
c[ i] = c[ i- 1 ] ;
}
for ( i= sz- 1 ; i> k; i-- ) {
d[ i] = d[ i- 1 ] ;
}
a[ k] = aval;
b[ k] = bval;
c[ k] = cval;
d[ k] = dval;
}
template < class T, class S> T inline cDiv( T a, S b) {
T m;
if ( b < 0 ) {
a = - a;
b = - b;
}
m = a % b;
if ( m == 0 ) {
return a / b;
}
if ( m < 0 ) {
m + = b;
}
return ( a + b - m) / b;
}
int N;
long long X[ 1000 ] ;
long long Y[ 1000 ] ;
int sz;
long long dx[ 1000000 ] ;
long long dy[ 1000000 ] ;
long long dp1[ 1000000 ] ;
long long dp2[ 1000000 ] ;
int ls;
long long lss[ 1000000 ] ;
int main( ) {
wmem = memarr;
long long a;
long long b;
long long c;
long long d;
long long i;
long long j;
long long k;
long long res = 0 ;
map< long long ,int > s;
map< pair< long long ,long long > ,int > ss;
rd( N) ;
{
int Lj4PdHRW;
for ( Lj4PdHRW= ( 0 ) ; Lj4PdHRW< ( N) ; Lj4PdHRW++ ) {
rd( X[ Lj4PdHRW] ) ;
rd( Y[ Lj4PdHRW] ) ;
}
}
for ( i= ( 0 ) ; i< ( N) ; i++ ) {
for ( j= ( i+ 1 ) ; j< ( N) ; j++ ) {
a = X[ i] - X[ j] ;
b = Y[ i] - Y[ j] ;
if ( a < 0 || ( a == 0 && b < 0 ) ) {
{
auto xr20shxY = ( - 1 ) ;
a * = xr20shxY;
b * = xr20shxY;
}
}
{
auto t_ynMSdg = ( GCD_L( abs ( a) , abs ( b) ) ) ;
a / = t_ynMSdg;
b / = t_ynMSdg;
}
{
auto KrdatlYV = ( X[ i] ) ;
auto ao_dF3pO = ( Y[ i] ) ;
c = KrdatlYV;
d = ao_dF3pO;
}
if ( a > 0 ) {
k = cDiv( - c, a) ;
{
auto tU__gIr_ = ( k* a) ;
auto a2conNHc = ( k* b) ;
c + = tU__gIr_;
d + = a2conNHc;
}
}
else {
d = 0 ;
}
arrInsert( sz, sz, dx, a, dy, b, dp1, c, dp2, d) ;
}
}
sortA_L( sz, dx, dy, dp1, dp2) ;
for ( i= ( 0 ) ; i< ( sz) ; i++ ) {
ls = 1 ;
lss[ 0 ] = 1 ;
while ( i+ 1 < sz && dx[ i] == dx[ i+ 1 ] && dy[ i] == dy[ i+ 1 ] ) {
if ( dp1[ i] ! = dp1[ i+ 1 ] || dp2[ i] ! = dp2[ i+ 1 ] ) {
lss[ ls++ ] = 0 ;
}
lss[ ls- 1 ] ++ ;
i++ ;
}
{
int jZyWAPpY;
long long jbtyPBGc;
if ( ls== 0 ) {
jbtyPBGc = 0 ;
}
else {
jbtyPBGc = lss[ 0 ] ;
for ( jZyWAPpY= ( 1 ) ; jZyWAPpY< ( ls) ; jZyWAPpY++ ) {
jbtyPBGc + = lss[ jZyWAPpY] ;
}
}
k = jbtyPBGc;
}
res + = k * ( k- 1 ) / 2 ;
for ( j= ( 0 ) ; j< ( ls) ; j++ ) {
res - = lss[ j] * ( lss[ j] - 1 ) / 2 ;
}
}
for ( i= ( 0 ) ; i< ( N) ; i++ ) {
for ( j= ( i+ 1 ) ; j< ( N) ; j++ ) {
d = ( X[ i] + X[ j] ) * ( 4000000000LL + 5 ) + ( Y[ i] + Y[ j] ) ;
a = X[ i] - X[ j] ;
b = Y[ i] - Y[ j] ;
if ( a < 0 || ( a == 0 && b < 0 ) ) {
{
auto gEg5UqEA = ( - 1 ) ;
a * = gEg5UqEA;
b * = gEg5UqEA;
}
}
{
auto Hjfu7Vx7 = ( GCD_L( abs ( a) , abs ( b) ) ) ;
a / = Hjfu7Vx7;
b / = Hjfu7Vx7;
}
c = a * ( 4000000000LL + 5 ) + b;
res - = ( s[ d] ++ ) ;
res + = ( ss[ make_pair( d,c) ] ++ ) ;
}
}
wt_L( res) ;
wt_L( '\n ' ) ;
return 0 ;
}
// cLay varsion 20201102-1
// --- original code ---
// int N;
// ll X[1000], Y[1000];
//
// int sz; ll dx[1d6], dy[1d6], dp1[1d6], dp2[1d6];
// int ls; ll lss[1d6];
// {
// ll a, b, c, d, i, j, k;
// ll res = 0;
// map<ll,int> s;
// map<pair<ll,ll>,int> ss;
// rd(N,(X,Y)(N));
//
// rep(i,N) rep(j,i+1,N){
// a = X[i] - X[j];
// b = Y[i] - Y[j];
// if(a < 0 || (a == 0 && b < 0)) (a, b) *= -1;
// (a, b) /= gcd(abs(a), abs(b));
// (c, d) = (X[i], Y[i]);
// if(a > 0){
// k = cDiv(-c, a);
// (c, d) += (k*a, k*b);
// } else {
// d = 0;
// }
// arrInsert(sz, sz, dx, a, dy, b, dp1, c, dp2, d);
// }
//
// sortA(sz, dx, dy, dp1, dp2);
// rep(i,sz){
// ls = 1;
// lss[0] = 1;
// while(i+1 < sz && dx[i]==dx[i+1] && dy[i]==dy[i+1]){
// if(dp1[i]!=dp1[i+1] || dp2[i]!=dp2[i+1]) lss[ls++] = 0;
// lss[ls-1]++;
// i++;
// }
// k = sum(lss(ls));
// res += k * (k-1) / 2;
// rep(j,ls) res -= lss[j] * (lss[j]-1) / 2;
// }
//
// rep(i,N) rep(j,i+1,N){
// d = (X[i] + X[j]) * (4d9 + 5) + (Y[i] + Y[j]);
//
// a = X[i] - X[j];
// b = Y[i] - Y[j];
// if(a < 0 || (a == 0 && b < 0)) (a, b) *= -1;
// (a, b) /= gcd(abs(a), abs(b));
// c = a * (4d9 + 5) + b;
//
// res -= (s[d]++);
// res += (ss[make_pair(d,c)]++);
// }
// wt(res);
// }
#pragma GCC optimize ("Ofast")
#include<bits/stdc++.h>
using namespace std;
void*wmem;
char memarr[96000000];
template<class T> inline void walloc1d(T **arr, int x, void **mem = &wmem){
  static int skip[16] = {0, 15, 14, 13, 12, 11, 10, 9, 8, 7, 6, 5, 4, 3, 2, 1};
  (*mem) = (void*)( ((char*)(*mem)) + skip[((unsigned long long)(*mem)) & 15] );
  (*arr)=(T*)(*mem);
  (*mem)=((*arr)+x);
}
template<class T> inline void walloc1d(T **arr, int x1, int x2, void **mem = &wmem){
  walloc1d(arr, x2-x1, mem);
  (*arr) -= x1;
}
template<class T1, class T2, class T3> void sortA_L(int N, T1 a[], T2 b[], T3 c[], void *mem = wmem){
  int i;
  pair<T1, pair<T2, T3> >*arr;
  walloc1d(&arr, N, &mem);
  for(i=(0);i<(N);i++){
    arr[i].first = a[i];
    arr[i].second.first = b[i];
    arr[i].second.second = c[i];
  }
  sort(arr, arr+N);
  for(i=(0);i<(N);i++){
    a[i] = arr[i].first;
    b[i] = arr[i].second.first;
    c[i] = arr[i].second.second;
  }
}
template<class T1, class T2, class T3, class T4> void sortA_L(int N, T1 a[], T2 b[], T3 c[], T4 d[], void *mem = wmem){
  int i;
  pair<pair<T1, T2>, pair<T3, T4> >*arr;
  walloc1d(&arr, N, &mem);
  for(i=(0);i<(N);i++){
    arr[i].first.first = a[i];
    arr[i].first.second = b[i];
    arr[i].second.first = c[i];
    arr[i].second.second = d[i];
  }
  sort(arr, arr+N);
  for(i=(0);i<(N);i++){
    a[i] = arr[i].first.first;
    b[i] = arr[i].first.second;
    c[i] = arr[i].second.first;
    d[i] = arr[i].second.second;
  }
}
inline int my_getchar_unlocked(){
  static char buf[1048576];
  static int s = 1048576;
  static int e = 1048576;
  if(s == e && e == 1048576){
    e = fread_unlocked(buf, 1, 1048576, stdin);
    s = 0;
  }
  if(s == e){
    return EOF;
  }
  return buf[s++];
}
inline void rd(int &x){
  int k;
  int m=0;
  x=0;
  for(;;){
    k = my_getchar_unlocked();
    if(k=='-'){
      m=1;
      break;
    }
    if('0'<=k&&k<='9'){
      x=k-'0';
      break;
    }
  }
  for(;;){
    k = my_getchar_unlocked();
    if(k<'0'||k>'9'){
      break;
    }
    x=x*10+k-'0';
  }
  if(m){
    x=-x;
  }
}
inline void rd(long long &x){
  int k;
  int m=0;
  x=0;
  for(;;){
    k = my_getchar_unlocked();
    if(k=='-'){
      m=1;
      break;
    }
    if('0'<=k&&k<='9'){
      x=k-'0';
      break;
    }
  }
  for(;;){
    k = my_getchar_unlocked();
    if(k<'0'||k>'9'){
      break;
    }
    x=x*10+k-'0';
  }
  if(m){
    x=-x;
  }
}
struct MY_WRITER{
  char buf[1048576];
  int s;
  int e;
  MY_WRITER(){
    s = 0;
    e = 1048576;
  }
  ~MY_WRITER(){
    if(s){
      fwrite_unlocked(buf, 1, s, stdout);
    }
  }
}
;
MY_WRITER MY_WRITER_VAR;
void my_putchar_unlocked(int a){
  if(MY_WRITER_VAR.s == MY_WRITER_VAR.e){
    fwrite_unlocked(MY_WRITER_VAR.buf, 1, MY_WRITER_VAR.s, stdout);
    MY_WRITER_VAR.s = 0;
  }
  MY_WRITER_VAR.buf[MY_WRITER_VAR.s++] = a;
}
inline void wt_L(char a){
  my_putchar_unlocked(a);
}
inline void wt_L(long long x){
  int s=0;
  int m=0;
  char f[20];
  if(x<0){
    m=1;
    x=-x;
  }
  while(x){
    f[s++]=x%10;
    x/=10;
  }
  if(!s){
    f[s++]=0;
  }
  if(m){
    my_putchar_unlocked('-');
  }
  while(s--){
    my_putchar_unlocked(f[s]+'0');
  }
}
template<class T, class U> inline T GCD_L(T a, U b){
  T r;
  while(b){
    r=a;
    a=b;
    b=r%a;
  }
  return a;
}
template<class S> inline void arrInsert(const int k, int &sz, S a[], const S aval){
  int i;
  sz++;
  for(i=sz-1;i>k;i--){
    a[i] = a[i-1];
  }
  a[k] = aval;
}
template<class S, class T> inline void arrInsert(const int k, int &sz, S a[], const S aval, T b[], const T bval){
  int i;
  sz++;
  for(i=sz-1;i>k;i--){
    a[i] = a[i-1];
  }
  for(i=sz-1;i>k;i--){
    b[i] = b[i-1];
  }
  a[k] = aval;
  b[k] = bval;
}
template<class S, class T, class U> inline void arrInsert(const int k, int &sz, S a[], const S aval, T b[], const T bval, U c[], const U cval){
  int i;
  sz++;
  for(i=sz-1;i>k;i--){
    a[i] = a[i-1];
  }
  for(i=sz-1;i>k;i--){
    b[i] = b[i-1];
  }
  for(i=sz-1;i>k;i--){
    c[i] = c[i-1];
  }
  a[k] = aval;
  b[k] = bval;
  c[k] = cval;
}
template<class S, class T, class U, class V> inline void arrInsert(const int k, int &sz, S a[], const S aval, T b[], const T bval, U c[], const U cval, V d[], const V dval){
  int i;
  sz++;
  for(i=sz-1;i>k;i--){
    a[i] = a[i-1];
  }
  for(i=sz-1;i>k;i--){
    b[i] = b[i-1];
  }
  for(i=sz-1;i>k;i--){
    c[i] = c[i-1];
  }
  for(i=sz-1;i>k;i--){
    d[i] = d[i-1];
  }
  a[k] = aval;
  b[k] = bval;
  c[k] = cval;
  d[k] = dval;
}
template<class T, class S> T inline cDiv(T a, S b){
  T m;
  if(b < 0){
    a = -a;
    b = -b;
  }
  m = a % b;
  if(m == 0){
    return a / b;
  }
  if(m < 0){
    m += b;
  }
  return (a + b - m) / b;
}
int N;
long long X[1000];
long long Y[1000];
int sz;
long long dx[1000000];
long long dy[1000000];
long long dp1[1000000];
long long dp2[1000000];
int ls;
long long lss[1000000];
int main(){
  wmem = memarr;
  long long a;
  long long b;
  long long c;
  long long d;
  long long i;
  long long j;
  long long k;
  long long res = 0;
  map<long long,int> s;
  map<pair<long long,long long>,int> ss;
  rd(N);
  {
    int Lj4PdHRW;
    for(Lj4PdHRW=(0);Lj4PdHRW<(N);Lj4PdHRW++){
      rd(X[Lj4PdHRW]);
      rd(Y[Lj4PdHRW]);
    }
  }
  for(i=(0);i<(N);i++){
    for(j=(i+1);j<(N);j++){
      a = X[i] - X[j];
      b = Y[i] - Y[j];
      if(a < 0 || (a == 0 && b < 0)){
        {
          auto xr20shxY = (-1);
          a *= xr20shxY;
          b *= xr20shxY;
        }
      }
      {
        auto t_ynMSdg = (GCD_L(abs(a), abs(b)));
        a /= t_ynMSdg;
        b /= t_ynMSdg;
      }
      {
        auto KrdatlYV = (X[i]);
        auto ao_dF3pO = ( Y[i]);
        c = KrdatlYV;
        d = ao_dF3pO;
      }
      if(a > 0){
        k = cDiv(-c, a);
        {
          auto tU__gIr_ = (k*a);
          auto a2conNHc = ( k*b);
          c += tU__gIr_;
          d += a2conNHc;
        }
      }
      else{
        d = 0;
      }
      arrInsert(sz, sz, dx, a, dy, b, dp1, c, dp2, d);
    }
  }
  sortA_L(sz, dx, dy, dp1, dp2);
  for(i=(0);i<(sz);i++){
    ls = 1;
    lss[0] = 1;
    while(i+1 < sz && dx[i]==dx[i+1] && dy[i]==dy[i+1]){
      if(dp1[i]!=dp1[i+1] || dp2[i]!=dp2[i+1]){
        lss[ls++] = 0;
      }
      lss[ls-1]++;
      i++;
    }
    {
      int jZyWAPpY;
      long long jbtyPBGc;
      if(ls==0){
        jbtyPBGc = 0;
      }
      else{
        jbtyPBGc = lss[0];
        for(jZyWAPpY=(1);jZyWAPpY<(ls);jZyWAPpY++){
          jbtyPBGc += lss[jZyWAPpY];
        }
      }
      k =jbtyPBGc;
    }
    res += k * (k-1) / 2;
    for(j=(0);j<(ls);j++){
      res -= lss[j] * (lss[j]-1) / 2;
    }
  }
  for(i=(0);i<(N);i++){
    for(j=(i+1);j<(N);j++){
      d = (X[i] + X[j]) * (4000000000LL + 5) + (Y[i] + Y[j]);
      a = X[i] - X[j];
      b = Y[i] - Y[j];
      if(a < 0 || (a == 0 && b < 0)){
        {
          auto gEg5UqEA = (-1);
          a *= gEg5UqEA;
          b *= gEg5UqEA;
        }
      }
      {
        auto Hjfu7Vx7 = (GCD_L(abs(a), abs(b)));
        a /= Hjfu7Vx7;
        b /= Hjfu7Vx7;
      }
      c = a * (4000000000LL + 5) + b;
      res -= (s[d]++);
      res += (ss[make_pair(d,c)]++);
    }
  }
  wt_L(res);
  wt_L('\n');
  return 0;
}
// cLay varsion 20201102-1

// --- original code ---
// int N;
// ll X[1000], Y[1000];
// 
// int sz; ll dx[1d6], dy[1d6], dp1[1d6], dp2[1d6];
// int ls; ll lss[1d6];
// {
//   ll a, b, c, d, i, j, k;
//   ll res = 0;
//   map<ll,int> s;
//   map<pair<ll,ll>,int> ss;
//   rd(N,(X,Y)(N));
// 
//   rep(i,N) rep(j,i+1,N){
//     a = X[i] - X[j];
//     b = Y[i] - Y[j];
//     if(a < 0 || (a == 0 && b < 0)) (a, b) *= -1;
//     (a, b) /= gcd(abs(a), abs(b));
//     (c, d) = (X[i], Y[i]);
//     if(a > 0){
//       k = cDiv(-c, a);
//       (c, d) += (k*a, k*b);
//     } else {
//       d = 0;
//     }
//     arrInsert(sz, sz, dx, a, dy, b, dp1, c, dp2, d);
//   }
// 
//   sortA(sz, dx, dy, dp1, dp2);
//   rep(i,sz){
//     ls = 1;
//     lss[0] = 1;
//     while(i+1 < sz && dx[i]==dx[i+1] && dy[i]==dy[i+1]){
//       if(dp1[i]!=dp1[i+1] || dp2[i]!=dp2[i+1]) lss[ls++] = 0;
//       lss[ls-1]++;
//       i++;
//     }
//     k = sum(lss(ls));
//     res += k * (k-1) / 2;
//     rep(j,ls) res -= lss[j] * (lss[j]-1) / 2;
//   }
// 
//   rep(i,N) rep(j,i+1,N){
//     d = (X[i] + X[j]) * (4d9 + 5) + (Y[i] + Y[j]);
// 
//     a = X[i] - X[j];
//     b = Y[i] - Y[j];
//     if(a < 0 || (a == 0 && b < 0)) (a, b) *= -1;
//     (a, b) /= gcd(abs(a), abs(b));
//     c = a * (4d9 + 5) + b;
// 
//     res -= (s[d]++);
//     res += (ss[make_pair(d,c)]++);
//   }
//   wt(res);
// }
