Commit 16f85390 authored by Tom Lane's avatar Tom Lane

Support for emulating RTREE indexing in GiST. Contributed by

Oleg Bartunov and Teodor Sigaev.
parent 3043810d
# $Header: /cvsroot/pgsql/contrib/Makefile,v 1.20 2001/05/10 15:51:05 momjian Exp $ # $Header: /cvsroot/pgsql/contrib/Makefile,v 1.21 2001/05/31 18:27:18 tgl Exp $
subdir = contrib subdir = contrib
top_builddir = .. top_builddir = ..
...@@ -27,6 +27,7 @@ WANTED_DIRS = \ ...@@ -27,6 +27,7 @@ WANTED_DIRS = \
pgbench \ pgbench \
pgcrypto \ pgcrypto \
rserv \ rserv \
rtree_gist \
seg \ seg \
soundex \ soundex \
spi \ spi \
......
...@@ -133,6 +133,10 @@ rserv - ...@@ -133,6 +133,10 @@ rserv -
replication server replication server
by Vadim B. Mikheev <vadim4o@email.com> by Vadim B. Mikheev <vadim4o@email.com>
rtree_gist -
Support for emulating RTREE indexing in GiST
by Oleg Bartunov <oleg@sai.msu.su> and Teodor Sigaev <teodor@stack.net>
seg - seg -
Confidence-interval datatype (GiST indexing example) Confidence-interval datatype (GiST indexing example)
by Gene Selkov, Jr. <selkovjr@mcs.anl.gov> by Gene Selkov, Jr. <selkovjr@mcs.anl.gov>
......
#
# $Header: /cvsroot/pgsql/contrib/rtree_gist/Attic/Makefile,v 1.1 2001/05/31 18:27:18 tgl Exp $
#
subdir = contrib/rtree_gist
top_builddir = ../..
include $(top_builddir)/src/Makefile.global
# override libdir to install shlib in contrib not main directory
libdir := $(libdir)/contrib
# shared library parameters
NAME= rtree_gist
SO_MAJOR_VERSION= 1
SO_MINOR_VERSION= 0
override CPPFLAGS := -I$(srcdir) $(CPPFLAGS)
OBJS= rtree_gist.o
all: all-lib $(NAME).sql
# Shared library stuff
include $(top_srcdir)/src/Makefile.shlib
$(NAME).sql: $(NAME).sql.in
sed -e 's:MODULE_PATHNAME:$(libdir)/$(shlib):g' < $< > $@
.PHONY: submake
submake:
$(MAKE) -C $(top_builddir)/src/test/regress pg_regress
# against installed postmaster
installcheck: submake
$(top_builddir)/src/test/regress/pg_regress rtree_gist
# in-tree test doesn't work yet (no way to install my shared library)
#check: all submake
# $(top_builddir)/src/test/regress/pg_regress --temp-install \
# --top-builddir=$(top_builddir) rtree_gist
check:
@echo "'make check' is not supported."
@echo "Do 'make install', then 'make installcheck' instead."
install: all installdirs install-lib
$(INSTALL_DATA) $(srcdir)/README.$(NAME) $(docdir)/contrib
$(INSTALL_DATA) $(NAME).sql $(datadir)/contrib
installdirs:
$(mkinstalldirs) $(docdir)/contrib $(datadir)/contrib $(libdir)
uninstall: uninstall-lib
rm -f $(docdir)/contrib/README.$(NAME) $(datadir)/contrib/$(NAME).sql
clean distclean maintainer-clean: clean-lib
rm -f $(OBJS) $(NAME).sql
# things created by various check targets
rm -rf results tmp_check log
rm -f regression.diffs regression.out regress.out run_check.out
ifeq ($(PORTNAME), win)
rm -f regress.def
endif
depend dep:
$(CC) -MM $(CFLAGS) *.c >depend
ifeq (depend,$(wildcard depend))
include depend
endif
This is R-Tree implementation using GiST.
Code (for PG95) are taken from http://s2k-ftp.cs.berkeley.edu:8000/gist/pggist/
and changed according to new version of GiST (7.1 and above)
All work was done by Teodor Sigaev (teodor@stack.net) and Oleg Bartunov
(oleg@sai.msu.su). See http://www.sai.msu.su/~megera/postgres/gist
for additional information.
CHANGES:
Tue May 29 17:04:16 MSD 2001
1. Small fixes in polygon code
Thanks to Dave Blasby <dblasby@refractions.net>
Mon May 28 19:42:14 MSD 2001
1. Full implementation of R-tree using GiST - gist_box_ops,gist_poly_ops
2. gist_poly_ops is lossy
3. NULLs support
4. works with multi-key GiST
NOTICE:
This version will works only with postgresql version 7.1 and above
because of changes in interface of function calling.
INSTALLATION:
gmake
gmake install
-- load functions
psql <database> < rtree_gist.sql
REGRESSION TEST:
gmake installcheck
EXAMPLE USAGE:
create table boxtmp (b box);
-- create index
create index bix on boxtmp using gist (b gist_box_ops);
-- query
select * from boxtmp where b && '(1000,1000,0,0)'::box;
BENCHMARKS:
subdirectory bench contains benchmark suite.
Prerequisities: perl, DBI, DBD:Pg, Time::HiRes
cd ./bench
1. createdb TEST
2. psql TEST < ../box.sql
3. ./create_test.pl | psql TEST
-- change $NUM - number of rows in test dataset
4. ./bench.pl - perl script to benchmark queries.
Run script without arguments to see available options.
a)test without GiST index, using built-in R-Tree
./bench.pl -d TEST
b)test R-Tree using GiST index
./bench.pl -d TEST -g
RESULTS:
1. One interesting thing is that insertion time for built-in R-Tree is
about 8 times more than ones for GiST implementation of R-Tree !!!
2. Postmaster requires much more memory for built-in R-Tree
3. Search time depends on dataset. In our case we got:
+------------+-----------+--------------+
|Number boxes|R-tree, sec|R-tree using |
| | | GiST, sec |
+------------+-----------+--------------+
| 10| 0.002| 0.002|
+------------+-----------+--------------+
| 100| 0.002| 0.002|
+------------+-----------+--------------+
| 1000| 0.002| 0.002|
+------------+-----------+--------------+
| 10000| 0.015| 0.025|
+------------+-----------+--------------+
| 20000| 0.029| 0.048|
+------------+-----------+--------------+
| 40000| 0.055| 0.092|
+------------+-----------+--------------+
| 80000| 0.113| 0.178|
+------------+-----------+--------------+
| 160000| 0.338| 0.337|
+------------+-----------+--------------+
| 320000| 0.674| 0.673|
+------------+-----------+--------------+
#!/usr/bin/perl -w
use strict;
# make sure we are in a sane environment.
use DBI();
use DBD::Pg();
use Time::HiRes qw( usleep ualarm gettimeofday tv_interval );
use Getopt::Std;
my %opt;
getopts('d:b:gv', \%opt);
if ( !( scalar %opt ) ) {
print <<EOT;
Usage:
$0 -d DATABASE -b N [-v] [-g]
-d DATABASE - DATABASE name
-b N -number of cycles
-v - print sql
-g -use GiST index( default built-in R-tree )
EOT
exit;
}
$opt{d} ||= 'TEST';
my $dbi=DBI->connect('DBI:Pg:dbname='.$opt{d}) || die "Couldn't connect DB: $opt{d} !\n";
my $sql;
my $notice;
my $sss = '(3000,3000,2990,2990)';
if ( $opt{g} ) {
$notice = "Testing GiST implementation of R-Tree";
$sql = "select count(*) from boxtmp where b && '$sss'::box;";
} else {
$notice = "Testing built-in implementation of R-Tree";
$sql = "select count(*) from boxtmp2 where b && '$sss'::box;";
}
my $t0 = [gettimeofday];
my $count=0;
my $b=$opt{b};
$b ||=1;
foreach ( 1..$b ) {
my @a=exec_sql($dbi,$sql);
$count=$#a;
}
my $elapsed = tv_interval ( $t0, [gettimeofday]);
print "$notice:\n";
print "$sql\n" if ( $opt{v} );
print "Done\n";
print sprintf("total: %.02f sec; number: %d; for one: %.03f sec; found %d docs\n", $elapsed, $b, $elapsed/$b, $count+1 );
$dbi -> disconnect;
sub exec_sql {
my ($dbi, $sql, @keys) = @_;
my $sth=$dbi->prepare($sql) || die;
$sth->execute( @keys ) || die;
my $r;
my @row;
while ( defined ( $r=$sth->fetchrow_hashref ) ) {
push @row, $r;
}
$sth->finish;
return @row;
}
#!/usr/bin/perl
use strict;
my $NUM = 20000;
print "drop table boxtmp;\n";
print "drop table boxtmp2;\n";
print "create table boxtmp (b box);\n";
print "create table boxtmp2 (b box);\n";
srand(1);
open(DAT,">bbb.dat") || die;
foreach ( 1..$NUM ) {
#print DAT '(',int( 500+500*rand() ),',',int( 500+500*rand() ),',',int( 500*rand() ),',',int( 500*rand() ),")\n";
my ( $x1,$y1, $x2,$y2 ) = (
10000*rand(),
10000*rand(),
10000*rand(),
10000*rand()
);
print DAT '(',
max($x1,$x2),',',
max($y1,$y2),',',
min($x1,$x2),',',
min($y1,$y2),")\n";
}
close DAT;
print "copy boxtmp from stdin;\n";
open(DAT,"bbb.dat") || die;
while(<DAT>) { print; }
close DAT;
print "\\.\n";
print "copy boxtmp2 from stdin;\n";
open(DAT,"bbb.dat") || die;
while(<DAT>) { print; }
close DAT;
print "\\.\n";
print "create index bix on boxtmp using gist (b gist_box_ops);\n";
print "create index bix2 on boxtmp2 using rtree (b box_ops);\n";
sub min {
return ( $_[0] < $_[1] ) ? $_[0] : $_[1];
}
sub max {
return ( $_[0] > $_[1] ) ? $_[0] : $_[1];
}
(12699,9028,12654,8987)
(22689,4680,22614,4626)
(43263,47296,43217,47217)
(6184,8397,6182,8379)
(863,28537,788,28456)
(33783,4733,33746,4693)
(40456,47134,40426,47087)
(45950,8153,45887,8060)
(33433,36474,33399,36460)
(41106,22017,41086,21962)
(19214,36781,19179,36767)
(11582,40823,11498,40737)
(35565,5404,35546,5360)
(26489,17387,26405,17356)
(30874,13849,30796,13814)
(38255,1619,38227,1593)
(4445,32006,4405,31914)
(3923,32921,3876,32913)
(36054,39464,36032,39434)
(46540,6780,46524,6758)
(12184,45811,12118,45787)
(13198,17090,13143,17051)
(30939,44578,30865,44486)
(12502,4939,12431,4902)
(3250,1108,3169,1063)
(34029,41240,33976,41180)
(47057,44018,46967,43927)
(699,10114,686,10058)
(5925,26020,5845,25979)
(9462,39388,9382,39388)
(270,32616,226,32607)
(3959,49145,3861,49115)
(207,40886,179,40879)
(48480,43312,48412,43233)
(37183,37209,37161,37110)
(13576,13505,13521,13487)
(5877,1037,5818,1036)
(6777,16694,6776,16692)
(49362,13905,49299,13845)
(29356,14606,29313,14562)
(5492,6976,5441,6971)
(288,49588,204,49571)
(36698,37213,36682,37158)
(718,41336,645,41272)
(8725,23369,8660,23333)
(40115,9894,40025,9818)
(40051,41181,40015,41153)
(5739,1740,5715,1731)
(25120,27935,25054,27876)
(27475,46084,27447,46003)
\N
(33197,3252,33161,3245)
(10892,15691,10869,15662)
(39012,44712,38995,44640)
(4506,6484,4458,6459)
(13970,26316,13964,26236)
(28009,28104,27968,28030)
(5991,27613,5906,27607)
(23649,6338,23610,6314)
(25942,10008,25911,9928)
(25651,29943,25590,29906)
\N
(24555,40334,24546,40330)
(46870,43762,46789,43709)
\N
(20030,2752,19945,2687)
(30758,26754,30718,26678)
\N
(4320,44673,4286,44625)
\N
(1011,15576,939,15574)
(41936,40699,41854,40655)
(20594,19002,20561,18995)
(9388,41056,9325,41042)
(34771,46693,34751,46645)
(49398,46359,49332,46357)
\N
(23115,35380,23036,35306)
(46305,34840,46283,34765)
(16768,21692,16691,21647)
(28695,3128,28654,3112)
(22182,7107,22107,7074)
(14567,1210,14468,1139)
(14156,37139,14136,37119)
(33500,38351,33477,38286)
(39983,41981,39944,41954)
(26773,20824,26719,20813)
(42516,22947,42460,22932)
(26127,10701,26044,10650)
(17808,13803,17724,13710)
(14913,49873,14849,49836)
(37013,820,36955,736)
(39071,1399,39022,1381)
\N
(9785,42546,9687,42540)
(13423,14066,13354,14052)
(3417,14558,3336,14478)
(25212,46368,25128,46316)
(10124,39848,10027,39820)
(39722,39226,39656,39162)
(6298,28101,6250,28076)
(45852,5846,45809,5750)
(48292,4885,48290,4841)
(18905,4454,18894,4424)
(18965,43474,18902,43444)
(39843,28239,39761,28199)
(18087,44660,18019,44632)
(33886,10382,33794,10286)
(38383,13163,38362,13092)
(18861,25050,18842,24965)
(29887,14326,29806,14274)
(18733,11644,18698,11644)
(5119,37952,5089,37950)
(16191,34884,16149,34864)
(29544,1104,29496,1062)
(27740,41555,27701,41540)
(4672,4087,4633,4060)
(45441,38994,45377,38958)
(3272,1176,3232,1146)
(12820,26606,12790,26575)
(30910,7590,30877,7512)
(42476,39152,42377,39127)
(6562,38490,6542,38447)
(30046,20332,29988,20259)
(40723,15950,40671,15949)
(4945,46857,4908,46817)
(47986,16882,47963,16877)
(9842,22339,9805,22305)
(29831,23169,29818,23122)
(12322,34404,12250,34312)
(22846,11091,22759,10992)
(47627,2424,47603,2397)
(18375,43632,18347,43577)
(40441,974,40394,965)
(34260,10573,34194,10522)
(32914,9549,32828,9503)
(49023,37827,48978,37799)
(22183,10691,22111,10669)
\N
(38036,15828,38014,15759)
(34604,16801,34508,16746)
(26737,29997,26675,29976)
(47375,40298,47293,40210)
(771,2661,732,2649)
(28514,25659,28504,25577)
(13438,46494,13376,46455)
(7187,17877,7125,17786)
(49957,43390,49897,43384)
(26543,20067,26482,20057)
(16416,29803,16385,29724)
(36353,7484,36286,7414)
(26498,3377,26415,3358)
(28990,32205,28936,32193)
(45005,3842,45001,3816)
(21672,23566,21603,23566)
(33360,43465,33302,43429)
\N
(29884,9544,29838,9520)
\N
(5599,15012,5596,14930)
(22396,21481,22344,21422)
(24810,14955,24780,14887)
(47114,18866,47081,18784)
(39013,39245,38953,39237)
(12863,40534,12803,40529)
(351,37068,310,37019)
\N
(12916,34327,12891,34240)
\N
(49191,2694,49170,2628)
(24127,38407,24050,38325)
(3264,23053,3213,23007)
(8172,30385,8144,30336)
(19630,35716,19573,35640)
(42554,5148,42521,5117)
(42168,33453,42136,33426)
(17732,32093,17666,32057)
(1039,16626,1037,16587)
(21287,7757,21265,7679)
(47063,8260,47039,8225)
(38645,16238,38561,16204)
(18258,25358,18196,25341)
(30458,1742,30458,1695)
(35147,9273,35121,9233)
(7670,16625,7642,16545)
(49503,23432,49484,23383)
(31089,23146,31062,23093)
(47758,2734,47670,2703)
\N
(35276,1027,35259,972)
(26337,17603,26313,17579)
(35649,16777,35626,16777)
(42454,5105,42362,5101)
(21682,24951,21646,24920)
\N
(48383,25174,48303,25156)
(14672,3532,14601,3460)
(22570,22587,22515,22512)
(23566,25623,23484,25573)
(9530,24542,9504,24459)
(41271,451,41236,401)
(5556,37528,5502,37527)
(12479,25042,12447,24991)
(16568,22916,16499,22864)
(42700,13084,42676,12992)
\N
(35523,40973,35504,40932)
(32948,16962,32857,16901)
(7808,13469,7712,13469)
(13920,35203,13870,35131)
(22731,31563,22658,31557)
\N
(22909,43956,22900,43857)
(33077,35080,33074,35030)
(48064,29307,48022,29280)
(20232,46682,20212,46613)
(29949,16790,29867,16711)
(30260,32029,30180,31979)
(17184,34503,17110,34482)
(16066,42687,16039,42648)
(2947,19819,2857,19788)
(4900,47934,4818,47894)
(27193,19014,27174,18976)
\N
(15597,27948,15590,27939)
(11090,28623,11002,28589)
(26956,18651,26920,18620)
(3107,47753,3103,47711)
(6745,24151,6711,24083)
(43923,19213,43871,19124)
(33451,23578,33370,23534)
(8944,20605,8862,20601)
(14905,7536,14892,7441)
(2412,18357,2383,18354)
(37060,1443,36974,1366)
(15501,6230,15429,6190)
\N
(30333,50,30273,6)
(35567,9965,35482,9912)
(49847,7128,49798,7067)
\N
(27685,36396,27668,36384)
(43832,18491,43825,18431)
(36849,34600,36785,34589)
(2348,47938,2307,47902)
\N
(20473,22131,20445,22113)
(38486,4293,38471,4288)
(30611,30451,30553,30400)
(3883,21299,3819,21260)
(7696,37555,7644,37534)
(22399,7913,22317,7911)
(42565,38605,42500,38598)
(36595,12151,36500,12106)
(587,35217,571,35123)
(5764,15300,5764,15231)
(12003,21265,11983,21210)
(42564,4803,42470,4737)
(42359,36834,42271,36746)
(44700,14680,44658,14670)
(19690,5627,19620,5607)
(17780,43602,17714,43565)
(45073,3491,45041,3434)
(35043,2136,35017,2084)
\N
(39653,19215,39646,19198)
(23970,25560,23935,25502)
(28698,49233,28600,49223)
(30266,3605,30245,3540)
(25538,7857,25500,7791)
(17711,1757,17708,1756)
(5248,594,5190,587)
(2730,32454,2671,32436)
(1722,49089,1635,49067)
(40954,5743,40921,5722)
\N
(21382,4426,21298,4331)
(7885,18629,7872,18605)
(42838,6459,42748,6451)
(8217,19894,8207,19845)
(20489,18524,20433,18520)
(17383,23559,17309,23515)
(38952,38968,38934,38913)
(44665,18137,44636,18051)
(22416,41220,22383,41213)
(9901,664,9818,646)
(23475,21981,23449,21973)
(41875,17991,41818,17988)
(36517,47731,36509,47713)
(37595,49849,37581,49834)
(38771,32720,38748,32684)
(810,38523,736,38452)
(29695,14942,29665,14907)
(31911,15168,31906,15113)
(3454,36839,3438,36831)
(4832,47554,4820,47473)
\N
(11590,8292,11539,8272)
(8193,33323,8106,33317)
(16043,14799,16001,14710)
(19574,11395,19514,11316)
(26290,41424,26224,41342)
(22844,12516,22807,12471)
\N
(15709,49580,15655,49553)
(13387,28084,13379,28066)
(2780,38807,2690,38711)
(22031,32458,22028,32377)
(13511,3351,13440,3297)
(14648,26473,14614,26383)
(17798,19885,17726,19852)
(32355,27940,32324,27861)
(43773,21031,43767,20985)
(15419,45759,15403,45666)
(770,38863,729,38806)
(21221,35619,21183,35596)
(38924,31021,38894,30961)
(7395,32439,7345,32416)
(2324,25118,2268,25074)
(2958,15089,2935,15087)
(2424,160,2424,81)
(12123,18644,12099,18616)
(7459,30276,7422,30218)
(15847,45488,15814,45428)
(26409,29897,26389,29863)
(12336,34322,12279,34322)
(9440,23550,9396,23466)
(4991,30850,4905,30768)
(47262,11940,47201,11939)
(30584,42868,30555,42838)
(23144,24089,23056,24067)
\N
(35930,11609,35847,11573)
(7812,17271,7789,17203)
(17946,37554,17878,37480)
(27356,32869,27298,32813)
(29971,47783,29933,47697)
(26075,46494,25988,46451)
(39314,41366,39289,41269)
(31708,42900,31688,42865)
(4510,10231,4439,10203)
(43806,8482,43758,8446)
(45990,49694,45927,49617)
(48815,27640,48782,27573)
(41675,26733,41622,26723)
(23229,7709,23175,7693)
(48976,17733,48962,17731)
(10686,41470,10597,41434)
(18053,27059,17989,27012)
\N
(35495,25950,35459,25912)
(41896,45014,41881,44999)
(22654,41896,22572,41801)
(18581,7087,18524,6988)
\N
(14697,22406,14681,22311)
(40092,28122,40043,28030)
(35844,24243,35816,24238)
(1254,25653,1250,25644)
(1603,21730,1556,21640)
(33048,21779,32991,21763)
(29979,1632,29916,1592)
(8620,633,8580,620)
(22992,27035,22932,27008)
(21409,29315,21390,29309)
(3610,44748,3547,44699)
(20402,9318,20343,9267)
(31001,8709,30908,8658)
(46840,47640,46773,47551)
(49173,4705,49143,4630)
(5339,31657,5251,31622)
(8644,49668,8630,49648)
(45387,2893,45309,2885)
(47641,31020,47584,30941)
(40238,10636,40208,10568)
(19247,36924,19227,36924)
(917,19957,827,19887)
(40967,17841,40870,17820)
(15850,4109,15794,4085)
(20181,30916,20085,30870)
(161,24465,107,24374)
(21737,49690,21667,49663)
(10328,20911,10232,20852)
(24187,49823,24128,49768)
(36084,4578,36007,4501)
(38771,31741,38673,31674)
(2202,30102,2111,30006)
(27322,16074,27228,16039)
(6843,17280,6765,17248)
(16972,39744,16912,39700)
(10608,38741,10553,38708)
\N
(4917,34801,4828,34766)
(39281,33659,39268,33618)
(31706,7119,31645,7063)
(3427,44006,3422,44004)
\N
(10134,42608,10044,42599)
(26294,32080,26200,32068)
(21777,34680,21769,34606)
(23373,25957,23314,25915)
(10710,8401,10681,8400)
(42062,19458,42019,19394)
(26530,43036,26458,43004)
(3394,46081,3360,46077)
(38743,33953,38677,33924)
(32438,8226,32345,8160)
(9210,27333,9118,27301)
(19594,1600,19568,1551)
(10003,12278,9952,12255)
(31737,7206,31650,7146)
(16594,15821,16502,15759)
(28208,30296,28189,30278)
(30602,46237,30555,46185)
(20715,5155,20697,5140)
(48892,35271,48793,35210)
(3175,5590,3113,5525)
(34220,27947,34132,27865)
(35105,39792,35011,39727)
(21919,27314,21839,27286)
\N
(23963,3723,23917,3699)
(16312,14078,16236,14045)
(19233,49824,19185,49794)
(1447,11768,1356,11699)
(17311,17709,17224,17653)
(11962,31709,11871,31627)
(21355,40131,21355,40085)
(33750,35273,33724,35180)
(38896,25539,38879,25524)
(39569,44899,39569,44893)
(11075,41547,11039,41500)
(3215,12202,3199,12127)
(46215,33458,46132,33455)
(15121,38012,15083,37974)
(44448,18726,44412,18690)
(3899,38263,3870,38262)
(13854,13353,13786,13298)
(8252,5402,8191,5320)
(46849,37968,46820,37897)
(16422,13957,16376,13897)
(47369,7665,47353,7629)
(11982,40874,11956,40806)
\N
(9552,27580,9496,27562)
(32247,19399,32176,19337)
(32704,2169,32635,2091)
(7471,44213,7411,44130)
(48433,7096,48379,7089)
(37357,6543,37338,6452)
(30460,29624,30433,29535)
(20350,28794,20341,28705)
(6326,32360,6267,32317)
(1711,47519,1654,47430)
(49540,16510,49521,16426)
\N
(26975,618,26908,579)
(24118,30880,24020,30821)
(3675,15477,3625,15418)
(44953,9577,44953,9530)
(38323,7965,38235,7910)
(6629,36482,6579,36448)
(33953,16460,33878,16408)
(49222,16790,49186,16695)
(17308,16951,17274,16904)
(14135,6888,14077,6833)
(38617,47768,38603,47760)
(7345,10992,7290,10914)
(35261,42152,35176,42096)
(28586,4809,28544,4735)
(37521,25299,37495,25217)
(41941,17954,41912,17915)
(1209,46863,1171,46863)
(20103,34947,20048,34896)
(32716,33816,32656,33769)
(11113,6531,11036,6467)
(48635,7321,48563,7262)
(28435,37059,28349,37014)
(12311,17208,12232,17112)
(1466,48010,1379,48008)
(11226,11997,11223,11925)
(46896,32540,46821,32510)
(32661,31255,32632,31187)
(37739,20376,37655,20306)
(44002,43326,43920,43257)
(30337,1023,30271,968)
(34436,23357,34432,23345)
\N
(21367,8168,21353,8091)
(36370,21611,36369,21569)
(4152,36488,4080,36476)
(17696,13924,17664,13853)
(34252,19395,34159,19316)
(12574,3072,12573,2975)
(3995,21243,3943,21167)
(44553,30126,44513,30108)
\N
(4599,45275,4552,45254)
(33191,11404,33176,11348)
\N
(14245,18633,14177,18540)
(32457,20705,32393,20700)
(40052,10499,40016,10457)
(29824,44065,29785,44037)
(31613,12565,31557,12543)
(42692,29000,42652,28996)
\N
(40680,22219,40603,22140)
\N
(33575,27661,33488,27644)
(46194,1385,46184,1355)
(38442,48501,38407,48426)
(25305,21544,25236,21523)
(15562,8226,15561,8208)
\N
(20844,43614,20752,43558)
(22566,30541,22554,30532)
(2760,47802,2672,47789)
(25515,30745,25433,30675)
(48382,45134,48382,45093)
(9940,27094,9871,27087)
\N
(48690,44361,48610,44338)
(18992,11585,18899,11582)
(21551,49983,21492,49885)
(46778,29113,46770,29071)
(43219,9593,43212,9548)
(40291,1248,40224,1190)
(12687,22225,12635,22219)
(49372,38790,49306,38721)
(49503,46808,49411,46798)
(24745,5162,24732,5138)
(5046,26517,5023,26424)
(5583,46538,5495,46531)
(6084,35950,6079,35895)
(3503,23096,3437,23024)
\N
(45275,8420,45244,8418)
(13514,45251,13491,45249)
(42112,2748,42047,2668)
\N
(7810,21907,7806,21878)
(48378,36029,48303,35979)
(32568,48605,32510,48563)
(859,18915,810,18915)
(41963,17950,41939,17915)
\N
(42723,8031,42685,7955)
\N
(19587,5965,19556,5961)
(8713,33083,8629,32996)
(21243,7769,21226,7740)
(43752,43026,43720,42944)
(7883,41311,7859,41242)
(10178,47874,10157,47826)
(32177,48725,32093,48646)
(22960,2784,22953,2774)
(25101,49159,25087,49090)
(32142,48915,32086,48850)
(6636,44887,6590,44825)
(37814,11606,37769,11578)
(2870,23198,2820,23121)
(21025,16364,20947,16271)
(31341,36137,31269,36114)
(38921,7906,38888,7831)
(6966,17259,6922,17199)
(32426,13344,32401,13253)
(8084,30572,8078,30572)
(42230,47674,42150,47603)
(20724,44854,20724,44830)
(27471,38453,27454,38430)
(24590,37973,24544,37941)
(45832,26077,45772,26031)
(9589,24239,9582,24156)
(37484,49472,37409,49432)
(30044,19340,30004,19333)
(16966,14632,16936,14572)
(9439,40491,9403,40482)
(28945,5814,28913,5805)
(43788,41302,43746,41231)
(33631,43451,33614,43354)
(17590,49396,17510,49324)
(15173,32572,15109,32507)
(1912,23580,1840,23504)
(38165,16185,38076,16154)
(6729,1179,6637,1177)
\N
(6994,45406,6983,45325)
(2912,21327,2908,21305)
(14678,14244,14659,14222)
(29944,14959,29898,14900)
(47432,35658,47407,35610)
(25542,39243,25466,39149)
(5330,7206,5304,7165)
(24790,27196,24695,27118)
(38806,1961,38795,1906)
(23290,4487,23212,4416)
\N
(35035,24337,34990,24297)
(5549,38948,5549,38891)
(24558,15492,24501,15425)
(4636,3011,4574,2933)
(26522,39986,26451,39940)
(33486,18424,33410,18366)
(36638,14324,36625,14287)
(35115,41236,35055,41191)
(31927,16896,31841,16806)
(5796,43937,5697,43886)
(25681,41645,25663,41608)
(10962,42777,10894,42732)
(32715,11026,32672,10991)
(45803,20406,45710,20371)
(34730,17672,34658,17606)
(8809,6323,8798,6232)
\N
(39471,23837,39390,23749)
\N
(34078,17435,33987,17433)
(9133,4544,9041,4509)
(47274,29126,47242,29060)
(6404,28488,6403,28475)
(48894,49751,48846,49694)
(17324,43023,17301,42972)
(15599,8433,15557,8386)
(48575,10202,48488,10175)
(27638,24428,27608,24378)
(45277,47456,45240,47422)
(26482,46607,26482,46570)
(41400,33898,41397,33802)
\N
(49853,18504,49848,18503)
(11528,25165,11476,25080)
(49902,41752,49818,41746)
(1956,47506,1922,47424)
(21834,22058,21802,21964)
\N
(19414,21842,19386,21822)
(34801,13722,34744,13681)
(13924,29243,13835,29160)
(47749,21986,47664,21894)
(47051,39582,46974,39489)
(31287,49923,31236,49913)
(47429,8625,47337,8585)
(46987,44364,46901,44277)
(16158,27510,16099,27467)
(41184,6400,41148,6317)
(1847,42471,1829,42426)
\N
(14409,48602,14320,48555)
\N
(38137,42951,38045,42918)
(42875,2312,42832,2243)
(27242,30617,27181,30535)
(24882,44559,24812,44548)
(22021,1596,22015,1581)
(24300,1523,24250,1443)
(43946,35909,43869,35868)
(816,15988,776,15967)
(25243,9401,25237,9332)
(27967,25958,27928,25949)
(6575,33949,6484,33900)
(44812,35980,44800,35913)
(37577,13064,37495,13019)
\N
(30891,29967,30814,29884)
(15829,28836,15753,28807)
(11128,34180,11126,34117)
(9834,12537,9801,12508)
(4899,29069,4809,29024)
(29370,38459,29276,38382)
(40743,46653,40647,46559)
(9618,2723,9578,2631)
(32542,26837,32515,26769)
(5625,13409,5576,13355)
(47490,19229,47472,19203)
(48118,40275,48063,40203)
(19245,20549,19227,20546)
(25312,22243,25280,22164)
(18797,28934,18723,28881)
(31609,49393,31512,49366)
(26183,32888,26135,32824)
(46198,26153,46180,26149)
\N
(45383,16904,45353,16888)
(7132,11408,7091,11338)
(48262,43227,48236,43159)
(31722,12861,31675,12810)
\N
(41695,48924,41691,48921)
(48318,12877,48287,12802)
(12069,32241,11978,32231)
(8395,2694,8380,2661)
(19552,34590,19550,34497)
(12203,26166,12187,26143)
(35745,9571,35654,9542)
(22384,22535,22352,22439)
(21459,28189,21360,28189)
(7418,7203,7343,7182)
(39497,48412,39413,48318)
(1058,11132,979,11051)
(45623,31417,45548,31381)
\N
(23887,31921,23876,31891)
(7797,1244,7785,1155)
(23679,43650,23594,43644)
(21891,30561,21833,30485)
(4069,6870,4019,6785)
(5134,25117,5103,25034)
(36101,41895,36085,41810)
(39617,39211,39544,39191)
(37437,6604,37434,6585)
\N
(7749,32601,7740,32515)
(26203,34991,26159,34946)
(31856,39006,31783,39003)
(45828,24767,45788,24723)
\N
(49836,35965,49757,35871)
(44113,49024,44033,48995)
(38237,22326,38187,22253)
(45235,19087,45190,19005)
(1588,45285,1520,45254)
(46628,8701,46552,8665)
(47707,18258,47668,18250)
(9377,26162,9325,26079)
(28331,16766,28302,16731)
(15792,27875,15727,27809)
(16454,1972,16415,1967)
(21012,15828,20972,15784)
(27465,30603,27390,30560)
(39256,7697,39225,7604)
(25908,32801,25854,32770)
(25215,40109,25201,40106)
\N
(23280,4613,23190,4596)
(32440,30879,32405,30807)
(49156,4224,49126,4126)
(20005,40423,19911,40370)
(20978,8226,20930,8170)
(32127,22611,32126,22579)
(21764,26509,21701,26455)
\N
(32923,2834,32914,2830)
(7499,25331,7426,25300)
(6163,36942,6107,36908)
(41118,14583,41034,14486)
(21211,33369,21208,33331)
(7899,27682,7853,27603)
(16546,48436,16535,48400)
(24898,40195,24855,40174)
(43029,982,43004,952)
(26266,7962,26252,7950)
\N
(11308,44367,11210,44322)
(8902,28402,8808,28334)
(11671,19619,11665,19549)
(47202,23593,47153,23505)
(21981,40220,21905,40160)
(46721,2514,46687,2471)
(3450,33839,3424,33811)
(41854,45864,41762,45792)
(40183,47816,40114,47742)
(26119,33910,26077,33816)
(3430,16518,3365,16500)
(40063,32176,40005,32166)
(38702,15253,38679,15187)
(17719,12291,17658,12257)
(46131,30669,46068,30587)
(42738,10952,42731,10907)
(8721,45155,8650,45076)
(45317,26123,45244,26113)
(42694,11561,42614,11490)
(10043,12479,10009,12391)
(27584,2345,27578,2257)
(30889,8253,30866,8167)
\N
(5176,48928,5107,48838)
(9781,21023,9745,20976)
(32430,27908,32404,27859)
(3984,7391,3973,7352)
(18904,8094,18842,8091)
(20573,5508,20482,5496)
(7806,44368,7753,44297)
(18875,41452,18817,41376)
(6632,12142,6566,12079)
(33066,17865,33055,17854)
(45726,19628,45714,19589)
(26971,18459,26941,18423)
(26554,23641,26515,23592)
(45503,1325,45441,1231)
(11898,20164,11880,20115)
(27868,22837,27843,22776)
(34931,8206,34855,8144)
(42375,33603,42350,33539)
(3184,8308,3129,8238)
(26667,15813,26661,15785)
\N
(5760,49617,5730,49546)
(794,27001,777,26992)
(13518,45289,13459,45235)
\N
(34430,29754,34363,29736)
(37912,24574,37880,24543)
(8130,2270,8083,2258)
\N
(26930,21516,26848,21455)
(3634,33511,3592,33489)
(33080,5036,33035,4972)
(48389,13942,48316,13915)
(9231,5298,9150,5232)
(1357,10601,1321,10548)
\N
(35175,15295,35091,15269)
(33917,36863,33879,36784)
(8279,12052,8239,12021)
(11868,19083,11862,19034)
(24019,30777,24006,30703)
(44619,6959,44618,6938)
(28610,2626,28523,2582)
(29579,41801,29482,41775)
(23448,37609,23396,37534)
(40676,11252,40670,11191)
(39656,14077,39564,13999)
(33060,31042,33033,30950)
(11720,6816,11654,6792)
(13775,28873,13730,28868)
(47851,39121,47802,39084)
(30923,40255,30860,40199)
(44169,15070,44085,15015)
(42574,28664,42558,28590)
(8993,43487,8941,43460)
(40782,11648,40763,11631)
(18516,10143,18423,10137)
(39068,551,39005,491)
\N
(39672,12000,39575,11913)
(18508,37761,18464,37712)
(19083,35318,19079,35280)
(30286,13736,30222,13672)
(7223,9164,7132,9069)
(20764,29286,20700,29210)
(5733,8063,5699,8058)
(8566,43873,8549,43797)
(22126,27444,22062,27366)
(15105,8717,15078,8660)
(43987,33145,43940,33083)
\N
(46833,38652,46755,38612)
(47768,27202,47681,27169)
(22792,1183,22731,1152)
(25650,43310,25562,43247)
(37084,20116,37045,20057)
(47461,32556,47423,32555)
\N
(41225,18124,41215,18117)
(17623,25218,17553,25158)
(13770,21703,13770,21700)
(48958,35441,48870,35388)
(2976,1808,2892,1802)
(45118,22318,45049,22224)
(42287,26616,42281,26560)
(25525,6327,25468,6244)
\N
(40756,31634,40713,31568)
(23105,26565,23078,26565)
(48268,39862,48265,39827)
(41656,26254,41567,26243)
(28062,17920,28045,17825)
(6443,17321,6402,17238)
(10191,45466,10151,45447)
(18097,39706,18043,39649)
(37592,3244,37569,3197)
(29809,5978,29762,5950)
(12145,11251,12130,11202)
(37507,42999,37446,42956)
(10820,2866,10782,2830)
(36440,42904,36421,42832)
(38370,3386,38279,3311)
(9345,17279,9313,17197)
(20477,14864,20395,14807)
(37147,37769,37110,37729)
(15325,36135,15284,36053)
(29034,32897,29009,32854)
(2116,22274,2037,22216)
(15078,38330,15048,38251)
(7968,33600,7914,33573)
(832,23851,770,23786)
(38669,4348,38594,4344)
(8521,48573,8425,48564)
(1060,43320,969,43289)
(26170,10150,26144,10069)
(32324,8539,32285,8506)
(13121,18044,13109,18021)
(1597,9383,1594,9367)
(49539,35164,49505,35065)
(39464,10295,39409,10261)
(8921,37898,8825,37803)
(31171,47076,31093,47039)
(7178,41397,7108,41304)
(16240,34832,16162,34761)
(2829,20119,2782,20091)
(45854,21265,45810,21250)
(6382,12106,6315,12030)
(22301,46291,22291,46274)
(34142,14181,34078,14158)
(11258,29748,11198,29742)
\N
(37450,6943,37398,6882)
(41675,27207,41643,27130)
(13578,49562,13573,49479)
(37132,37397,37081,37301)
(49404,37193,49332,37170)
(33536,31809,33444,31735)
(45990,42751,45893,42708)
(38852,20510,38802,20509)
(27453,15836,27391,15802)
(9347,29004,9284,28946)
(44871,27727,44778,27668)
(14978,19646,14970,19644)
(23243,47091,23166,47080)
(45204,21431,45167,21370)
(14082,22316,14078,22235)
(42778,22694,42744,22606)
(4834,25241,4760,25196)
(20497,18110,20494,18038)
(45738,35524,45706,35496)
(21575,5151,21493,5092)
(2194,10052,2172,9960)
\N
(47735,24472,47682,24460)
(46740,35700,46695,35609)
(24647,42807,24568,42779)
(18000,30576,17975,30506)
(48638,46630,48544,46628)
(48508,33600,48477,33578)
(38703,45408,38670,45313)
(21712,15015,21625,14956)
(5840,42007,5768,41992)
(44011,11138,43953,11117)
(3899,33262,3897,33238)
(30142,23967,30096,23927)
(36950,13226,36908,13141)
(13130,26915,13071,26873)
(38576,35408,38539,35392)
(16776,46244,16700,46176)
(38251,25969,38168,25948)
\N
(3512,32256,3417,32242)
(31923,31225,31832,31197)
(5144,4969,5124,4937)
(34499,46164,34430,46162)
\N
(39432,31907,39388,31828)
(17316,24606,17221,24533)
(20751,49352,20709,49323)
(41673,30418,41623,30377)
(29026,24400,28971,24345)
(21929,30617,21894,30598)
(35539,12421,35536,12355)
(24938,45583,24870,45525)
\N
(27442,33090,27353,33064)
(23949,12046,23949,12036)
(11399,377,11360,294)
(47099,9989,47023,9942)
(641,33118,639,33084)
(13687,41308,13682,41290)
\N
(3682,17727,3645,17660)
(13262,19396,13185,19357)
(18791,389,18774,366)
(12489,45384,12403,45369)
\N
(12065,6364,12015,6325)
\N
(32705,23886,32619,23827)
\N
(7004,37333,6911,37240)
(28594,38078,28530,38050)
(5805,21797,5710,21701)
(41145,18905,41058,18873)
(35599,10002,35591,9956)
(5387,39087,5326,38994)
(11703,14003,11671,13912)
(4093,10472,4091,10470)
\N
(14110,49740,14063,49695)
(4170,470,4097,463)
(22219,17296,22164,17221)
(2505,20879,2446,20842)
\N
(47235,24744,47151,24667)
(30035,23234,30013,23197)
(3489,11659,3461,11607)
(38435,46322,38429,46230)
(12315,32880,12277,32854)
(33350,35297,33317,35263)
(18845,37671,18836,37589)
(24855,23554,24783,23520)
(48251,44461,48188,44408)
(17695,43353,17605,43286)
(4964,21292,4893,21270)
(33919,29907,33852,29878)
(29139,40010,29084,39957)
(41611,37750,41572,37741)
(41773,34717,41682,34700)
(8225,7424,8221,7363)
(1785,28248,1771,28219)
(21553,36307,21505,36257)
(7552,18199,7527,18119)
\N
(14410,30977,14349,30944)
\N
(20940,49142,20901,49069)
(36892,5522,36810,5478)
(40192,20926,40179,20926)
(44702,15182,44641,15117)
(43431,4921,43337,4827)
(41129,21654,41084,21642)
(6205,42785,6113,42722)
(23714,10224,23666,10205)
(9318,35175,9274,35139)
(40698,12676,40618,12627)
(49954,1340,49905,1294)
(32774,33062,32763,33062)
(4336,22183,4241,22157)
(10241,47657,10151,47592)
(6746,16718,6666,16634)
(26842,49694,26839,49680)
(34870,47437,34820,47347)
(26365,22266,26326,22183)
(39859,932,39829,840)
(33995,10888,33902,10793)
(32972,22342,32951,22340)
\N
(19951,10161,19932,10111)
(26779,45188,26745,45151)
(11235,13593,11184,13589)
(27334,20968,27288,20953)
(9586,43102,9488,43085)
(43935,49759,43925,49680)
(10548,37032,10474,36955)
(9326,14927,9295,14848)
(41340,11312,41311,11303)
(6500,44553,6454,44515)
\N
(8198,26841,8104,26749)
(47761,34183,47702,34140)
(43637,17912,43577,17910)
(17623,11138,17590,11122)
(48122,13132,48077,13060)
(27911,39796,27908,39777)
(1108,7918,1080,7832)
(18776,24329,18699,24326)
(1171,37901,1075,37871)
(38437,33948,38364,33907)
(1913,11593,1817,11533)
(22684,266,22656,181)
(13299,17075,13241,17074)
(6924,30196,6851,30113)
\N
(4367,13150,4298,13053)
(37381,6101,37380,6046)
(10307,28383,10270,28349)
(12283,8636,12256,8610)
(20230,32775,20144,32723)
(32942,12812,32905,12714)
(46140,7138,46140,7047)
(37235,29436,37161,29425)
(42486,25454,42478,25444)
(47860,46973,47842,46961)
(41760,21026,41662,20955)
(29663,20088,29566,20026)
(19167,33241,19101,33235)
(12306,37845,12301,37803)
(11288,873,11203,857)
(30309,5120,30282,5060)
(46927,19737,46856,19687)
(16664,20052,16649,19989)
(7330,8675,7296,8613)
(45067,45724,44991,45631)
(45317,10862,45218,10842)
(15012,47009,14998,46956)
(47882,10146,47813,10099)
(31571,46215,31511,46148)
(32257,2619,32187,2531)
(38924,41305,38872,41285)
(49981,34876,49898,34786)
(30501,35099,30418,35011)
\N
(45862,41438,45854,41434)
(38448,31878,38391,31822)
(8278,43463,8274,43378)
(5883,30629,5878,30564)
(49501,40346,49447,40275)
(31651,43116,31560,43106)
(44244,32940,44244,32926)
\N
(17941,18079,17938,18035)
(9518,32524,9470,32511)
(30707,43469,30686,43457)
(3284,46542,3187,46477)
(43423,29642,43393,29602)
(19940,16825,19877,16736)
(26194,47446,26194,47407)
(30386,24675,30333,24652)
(42707,44466,42688,44456)
\N
(43395,18525,43320,18467)
(28346,32259,28276,32196)
(45106,40786,45026,40767)
(36734,20414,36722,20363)
(37140,11569,37099,11475)
(8967,6409,8882,6341)
(31036,27923,30993,27890)
(22442,47682,22347,47663)
(32511,24029,32482,23970)
(22593,34444,22519,34399)
(41534,15495,41518,15455)
\N
(35862,19997,35818,19928)
(31419,8323,31404,8285)
(31036,19023,30978,19000)
(46900,15192,46891,15102)
(12774,9651,12765,9604)
(49985,6436,49927,6338)
(7184,47344,7089,47285)
(12792,45021,12740,45011)
(15019,27192,14940,27096)
(35415,23106,35381,23095)
(42129,14283,42095,14245)
(29375,45807,29347,45743)
(21763,24916,21700,24889)
(47656,8794,47579,8774)
(6139,49571,6059,49472)
(44492,45607,44483,45532)
(22699,4301,22628,4240)
(27407,24241,27335,24158)
\N
(38424,34460,38403,34458)
(46572,48456,46554,48402)
(39676,29056,39643,28981)
(4202,33076,4107,33010)
(32499,10592,32482,10575)
(22504,45417,22459,45378)
(49619,40322,49619,40268)
(14463,9305,14426,9224)
(10070,20300,10035,20211)
(35060,28561,34965,28553)
(23970,47522,23887,47428)
(46803,19155,46790,19131)
\N
(46151,49848,46058,49830)
(45266,40766,45209,40738)
(31041,32195,31007,32110)
(41401,17245,41334,17224)
(37445,654,37435,602)
(45568,31904,45508,31857)
(29326,7923,29285,7896)
(27078,34643,27027,34606)
(34492,43443,34437,43345)
(34109,4307,34083,4265)
(2755,45325,2727,45312)
(12571,24218,12536,24195)
(41224,2454,41149,2445)
(711,34828,655,34788)
(9104,18865,9036,18850)
(3508,26816,3456,26771)
(20159,16212,20116,16160)
(36871,7425,36777,7421)
(2751,45244,2734,45222)
(35867,28071,35769,28052)
(46878,35730,46850,35725)
(20610,35086,20513,35037)
(3903,32612,3887,32517)
(9330,40226,9289,40169)
(6338,28242,6329,28184)
(35668,18344,35606,18304)
(29892,48927,29878,48879)
(26999,646,26932,612)
(36377,38898,36338,38847)
(40289,31459,40236,31436)
(30377,1164,30306,1069)
(7642,12183,7590,12112)
(40325,1716,40296,1662)
(36412,38787,36318,38691)
(3967,33268,3923,33261)
(33914,40774,33873,40763)
(45978,41431,45963,41332)
(39195,12546,39120,12520)
(29962,30878,29941,30846)
(9365,10732,9310,10726)
(28801,23943,28740,23885)
(28934,38858,28928,38807)
(22126,45897,22068,45803)
(2923,33832,2918,33751)
(25116,2276,25083,2272)
(31174,14546,31144,14460)
(11728,9072,11658,9004)
(19804,49195,19730,49125)
(23090,28826,23010,28787)
(33989,27553,33947,27486)
(39702,47613,39641,47553)
(31397,3607,31304,3519)
(5835,9262,5791,9226)
(40112,37022,40038,36926)
(12346,29356,12282,29344)
(28503,9623,28469,9591)
(38449,43143,38378,43066)
(36950,37311,36905,37265)
(34824,5729,34818,5706)
(9288,26969,9225,26900)
(2535,42176,2478,42159)
(29098,49051,29085,49031)
(44759,33326,44727,33230)
(42849,2970,42821,2919)
(46014,27193,45985,27151)
(14506,13713,14417,13626)
(19342,44905,19332,44895)
(38178,37003,38147,36925)
(29179,27310,29084,27288)
(42713,10158,42671,10060)
(43336,38389,43290,38326)
(41260,34410,41245,34327)
(27907,2695,27830,2596)
(16309,44972,16222,44966)
(6230,22262,6214,22249)
\N
(9266,39458,9175,39447)
(33120,33548,33087,33538)
(43659,11416,43599,11375)
(49707,39258,49702,39159)
(23520,22140,23486,22072)
(24736,46502,24668,46412)
(7826,16851,7730,16807)
(39114,6048,39056,5965)
(11859,8753,11764,8701)
(42254,48367,42240,48328)
(26136,49185,26056,49175)
(38395,11209,38334,11137)
(33249,9425,33209,9348)
(22131,38502,22112,38460)
(5306,24344,5267,24268)
(30292,1198,30233,1149)
(9903,10896,9850,10806)
(25568,22911,25487,22868)
(22048,43391,22043,43362)
(20852,25827,20851,25766)
(35204,17119,35114,17093)
(5575,43431,5554,43410)
(17727,13623,17678,13560)
(14721,29520,14709,29461)
(40317,42220,40267,42166)
(31435,31012,31386,30931)
(40655,10103,40645,10006)
(35783,17802,35773,17763)
(34874,10210,34856,10200)
(3694,14279,3610,14239)
(27854,5493,27799,5433)
(34913,7234,34894,7220)
(15758,26445,15738,26421)
(23710,7272,23705,7270)
\N
(33679,13468,33628,13415)
\N
(31271,40495,31178,40461)
(759,187,662,163)
(14419,40434,14402,40381)
(45879,42933,45814,42872)
(167,17214,92,17184)
(9964,12210,9958,12195)
(35834,46257,35817,46211)
(26077,5629,25978,5621)
(46177,44640,46082,44544)
(44780,28753,44707,28692)
(35491,24729,35425,24690)
(33914,34190,33914,34131)
(17709,33253,17668,33227)
(45516,11888,45423,11848)
(24497,24752,24411,24710)
(30333,5952,30331,5886)
(444,12587,430,12497)
(7592,22353,7541,22287)
\N
(13387,37414,13329,37318)
\N
(21504,35227,21449,35210)
(18533,12909,18438,12848)
(41049,27148,41048,27088)
(18205,12222,18151,12140)
(18026,5164,18026,5156)
(34104,29862,34006,29815)
(18520,49686,18454,49602)
(37000,41493,36920,41424)
(43025,25711,42986,25687)
(38620,47018,38535,46934)
(24119,36813,24023,36739)
(48887,26359,48879,26302)
(47827,14625,47810,14609)
(10792,30746,10776,30716)
(30384,40672,30318,40582)
(48417,22790,48358,22746)
(14854,5819,14785,5798)
(19142,44414,19085,44406)
(31179,27081,31145,27005)
\N
(19692,8711,19659,8642)
(39689,14082,39603,14051)
(11181,39091,11119,39002)
(46015,23374,45936,23328)
(12517,49702,12427,49690)
(21926,21137,21841,21111)
(31956,12509,31870,12494)
(5895,2030,5851,2020)
(27094,5447,27014,5377)
(35781,8717,35780,8618)
(14012,12023,13972,12015)
(1702,12442,1696,12419)
(28549,5251,28462,5248)
(26441,21007,26360,20925)
(49820,7990,49771,7967)
(26424,29698,26339,29693)
(35146,6820,35071,6817)
\N
(15438,18788,15435,18729)
(47115,5235,47096,5143)
(33982,9002,33915,8925)
(14206,37041,14174,36955)
(24300,36616,24232,36613)
(44658,1788,44580,1769)
\N
(31539,43550,31463,43464)
\N
(16722,9673,16633,9652)
(44813,20573,44733,20544)
\N
(42114,32559,42040,32552)
(41561,36244,41477,36241)
(39589,33796,39548,33716)
(20365,26770,20329,26709)
(28511,208,28479,114)
(10010,25524,9930,25508)
\N
(1549,45666,1512,45621)
(16193,1927,16166,1869)
(34486,11500,34421,11401)
(14048,37944,13994,37901)
(21692,9594,21617,9496)
(2568,37899,2557,37811)
(4360,24503,4278,24443)
(50027,49230,49951,49214)
(44849,14867,44836,14813)
(16695,34896,16683,34840)
(12600,35217,12593,35129)
(23113,24009,23030,23962)
(49907,30225,49810,30158)
(18026,25208,17970,25208)
(49711,39844,49651,39790)
(5427,42682,5357,42637)
(23901,14221,23802,14184)
(15470,12185,15376,12163)
(47302,34023,47292,34001)
(24336,17418,24315,17393)
(13948,17043,13903,16970)
(8555,8986,8530,8953)
(48830,6038,48743,5986)
(48720,40687,48623,40610)
(21161,30970,21146,30896)
(9507,36316,9411,36261)
\N
(36643,18136,36614,18106)
(1858,7457,1851,7402)
(24452,44306,24372,44252)
\N
(3292,807,3205,806)
(6845,30694,6792,30627)
(21333,25786,21237,25751)
(23008,22574,22999,22511)
(8790,8893,8772,8806)
(43333,47968,43264,47900)
(5377,24103,5302,24076)
(18410,23993,18329,23907)
(24752,19126,24713,19069)
(49772,11378,49696,11293)
(3468,12920,3396,12873)
(1746,40342,1736,40333)
(49187,29737,49139,29681)
(27657,44952,27581,44917)
\N
(35407,30177,35345,30151)
(4071,40568,4058,40544)
(25998,30513,25965,30452)
(8195,45403,8097,45310)
(8276,41689,8183,41670)
\N
(48435,28550,48355,28455)
\N
(8139,25449,8136,25380)
(20302,25574,20297,25531)
\N
(22055,46659,22034,46567)
(3531,49962,3463,49934)
(46828,46938,46739,46902)
(42294,786,42212,739)
(8779,3292,8761,3275)
(48146,46170,48082,46151)
(21571,10000,21531,9919)
(35526,26029,35450,25945)
(38893,22225,38865,22197)
(22189,37520,22132,37497)
(810,43261,751,43198)
(10352,39144,10290,39093)
(8740,35435,8720,35432)
(31657,13551,31583,13484)
(39803,4019,39755,4014)
(46353,7853,46312,7824)
(30078,48975,30021,48970)
(2847,32036,2819,31966)
(25250,10147,25165,10140)
\N
(15643,38953,15585,38947)
(40792,29798,40731,29731)
(43249,26858,43215,26835)
(47229,2199,47201,2134)
(10052,23601,9958,23570)
(38981,21615,38892,21604)
(3651,45004,3570,44917)
(21503,8261,21409,8166)
(13518,34201,13465,34105)
(13899,25117,13836,25114)
(18327,17403,18301,17349)
(19503,13648,19483,13607)
(3554,19487,3529,19466)
(41102,43355,41070,43314)
(4663,45858,4583,45765)
(3971,3023,3931,2975)
(37124,7061,37080,6993)
(48530,47172,48459,47160)
(14575,29843,14509,29750)
(43443,23124,43357,23038)
(8864,48290,8857,48263)
(41597,39852,41577,39791)
(35610,33392,35556,33353)
(36415,17906,36328,17846)
(24919,43933,24839,43883)
(7457,14056,7395,14051)
(43851,4090,43801,4080)
(43567,18468,43471,18388)
(16711,6084,16652,6055)
(45888,45934,45846,45880)
(45630,9313,45585,9248)
(27119,25969,27094,25884)
(36155,11420,36120,11405)
(41880,47111,41808,47049)
\N
(17554,20379,17482,20374)
(38848,5936,38763,5869)
(28324,31019,28276,30944)
(43257,17152,43176,17091)
(42717,24613,42691,24527)
(16786,41486,16763,41403)
(19259,28780,19160,28711)
(25843,28265,25760,28171)
(48645,34816,48546,34755)
(7004,49289,6976,49236)
(30261,21833,30181,21776)
(5290,46672,5219,46661)
(21237,31901,21188,31849)
(23340,38537,23253,38472)
(17269,3682,17183,3586)
\N
(48200,15377,48110,15369)
(16546,22195,16477,22142)
(21436,8460,21378,8449)
\N
(46598,17235,46577,17138)
\N
(30212,36184,30152,36092)
(18037,155,17941,109)
(4945,29201,4933,29184)
(32835,18782,32770,18750)
(34160,33104,34120,33007)
(5151,26989,5149,26909)
(1801,15549,1710,15461)
(48988,34819,48951,34764)
(20904,32547,20856,32497)
\N
(32654,35183,32606,35144)
(14336,11763,14328,11712)
(30546,23808,30463,23773)
(6813,21006,6781,20924)
\N
(14199,22030,14185,21934)
(3783,14709,3747,14658)
(49428,47052,49422,46973)
(29551,27682,29470,27654)
(29170,37260,29151,37181)
(48924,24689,48894,24680)
(48497,34052,48453,33966)
\N
(21263,8203,21242,8176)
(46537,3797,46462,3735)
(18406,14579,18393,14563)
\N
(11583,16529,11536,16471)
(10564,46257,10478,46228)
(49769,34513,49761,34458)
\N
(9202,6482,9138,6391)
(40387,37411,40357,37360)
(11966,11802,11888,11751)
(15551,47438,15486,47406)
(12017,43288,11969,43230)
(9717,22574,9701,22495)
\N
(35083,49443,35075,49355)
(33857,9320,33813,9269)
(32106,10581,32012,10560)
(14345,12485,14273,12424)
(24187,46416,24175,46402)
(43854,42159,43808,42129)
(35399,40707,35359,40646)
(29585,25576,29493,25556)
(24919,7829,24911,7753)
\N
(17049,48390,17022,48304)
(25224,35012,25217,34922)
(47397,20853,47346,20779)
(17221,16558,17181,16516)
(8669,16491,8645,16486)
(23502,44241,23484,44164)
(36169,37046,36072,37010)
(44775,32394,44763,32357)
(30685,36871,30662,36792)
(21783,47642,21714,47630)
(34847,27467,34761,27372)
(43925,49912,43888,49878)
(16455,27861,16364,27813)
(38406,18310,38329,18309)
(5408,9461,5319,9426)
(41856,36900,41784,36854)
(23723,4460,23646,4448)
(18454,40138,18430,40046)
(17505,36822,17418,36763)
(36686,33534,36641,33476)
(11347,9454,11289,9436)
\N
(27816,34752,27745,34736)
(44213,8559,44162,8461)
(45359,26789,45315,26776)
(31249,19475,31224,19421)
(25917,44239,25819,44149)
(47313,40691,47264,40685)
(40577,33848,40513,33794)
(9606,45253,9582,45174)
(30005,24521,29910,24496)
(49332,35375,49309,35299)
\N
(12164,33871,12075,33820)
(19598,43327,19593,43314)
\N
(3818,28584,3815,28504)
\N
(35579,8611,35541,8604)
(8811,20986,8750,20954)
(16139,44777,16128,44686)
(35550,41501,35534,41458)
(43180,11927,43109,11891)
(45798,8465,45711,8460)
(18196,6886,18126,6845)
(1774,32167,1701,32073)
(7030,40790,7029,40711)
(11676,23009,11665,22915)
(33990,22561,33953,22474)
\N
(30366,9447,30284,9353)
(37626,32913,37596,32853)
(7730,42561,7665,42470)
(49347,8403,49315,8387)
(6874,3499,6812,3458)
(44189,16999,44169,16964)
(6312,30167,6231,30083)
(18932,6611,18909,6518)
(32262,13076,32223,13057)
(45989,249,45910,222)
(42710,855,42692,796)
(25562,9849,25535,9802)
(13348,46719,13260,46689)
(30022,42196,30005,42160)
\N
(22263,45954,22243,45950)
(18918,18890,18820,18795)
(31918,12003,31852,11989)
(12252,39453,12211,39398)
(40208,9789,40194,9759)
(35943,21767,35914,21693)
(18439,10706,18383,10618)
(2803,18999,2778,18925)
(14953,27444,14875,27397)
(12587,22025,12545,21928)
(33930,21090,33918,21009)
(10444,2606,10407,2553)
(28700,29782,28665,29703)
\N
(1402,13497,1397,13465)
\N
(24155,3075,24083,3062)
(38378,1864,38339,1849)
(29261,49910,29247,49818)
(38139,37073,38098,37057)
\N
(24468,41130,24418,41053)
(9989,1015,9959,939)
(47001,33561,46994,33518)
(47058,16030,46983,16012)
(35509,1814,35426,1748)
(3630,48019,3597,47923)
(47781,12986,47741,12947)
(16364,9908,16356,9882)
(17290,41508,17287,41410)
(42423,26477,42349,26434)
(10039,920,9952,833)
(16851,21338,16846,21314)
\N
(23104,7700,23062,7688)
(5619,2079,5611,2075)
(31471,49632,31375,49549)
(25793,12526,25783,12456)
(3935,29528,3866,29513)
\N
(5957,1646,5947,1595)
(2467,22376,2429,22349)
(43715,32673,43664,32595)
(6726,13093,6636,12994)
(31477,18347,31421,18299)
(34232,36635,34200,36552)
(49061,14516,49008,14442)
(43996,6129,43955,6074)
(7728,33802,7670,33703)
\N
(6131,36766,6053,36749)
(35791,16361,35696,16329)
(45759,8935,45675,8886)
(43634,2029,43537,1940)
(4916,32233,4844,32181)
(46701,23508,46623,23477)
(29590,4893,29552,4871)
(38647,4423,38574,4396)
(7593,25845,7497,25751)
(8510,43552,8432,43492)
(18791,39181,18730,39162)
(7462,2956,7454,2858)
(1394,26795,1392,26780)
(16707,21993,16609,21932)
(26838,10866,26803,10836)
(31642,29842,31585,29760)
(21891,3502,21863,3406)
(13258,587,13250,507)
(6072,47397,6021,47369)
(16605,49730,16579,49659)
(42830,40981,42791,40981)
(12975,3706,12913,3637)
(30925,21660,30826,21649)
(1455,14229,1410,14156)
\N
(17583,16486,17562,16474)
(33377,3387,33333,3381)
(784,6177,750,6095)
(22111,44110,22106,44013)
(1444,403,1346,344)
(4010,46220,3982,46212)
(17932,8150,17861,8127)
(38685,31466,38636,31416)
(14257,11549,14242,11522)
(14990,15217,14904,15211)
(21395,21533,21307,21520)
\N
(31948,33725,31885,33694)
(433,49033,390,48961)
(45205,609,45173,523)
(25065,35494,25003,35455)
(33265,6677,33224,6611)
(18179,22345,18133,22256)
(3916,13759,3820,13732)
(1696,13478,1604,13436)
(47203,25980,47130,25907)
(24913,13361,24868,13268)
(13824,40177,13792,40130)
(25671,13555,25585,13494)
\N
(20133,37769,20105,37679)
\N
(26368,16734,26288,16726)
(30545,35438,30458,35376)
(48816,22926,48812,22831)
(48807,31389,48739,31330)
(11003,10859,10950,10765)
(17288,8570,17247,8485)
(38377,31415,38331,31379)
\N
(19085,23425,19059,23326)
(40059,17068,40052,17006)
(18811,13493,18734,13394)
(36319,17197,36225,17181)
(14939,38780,14863,38714)
(49539,17656,49479,17629)
(42530,45951,42466,45854)
(27318,26654,27233,26610)
(49980,35004,49937,34963)
(18326,32558,18322,32502)
(45951,28555,45896,28481)
(12104,33531,12014,33501)
(22311,41113,22215,41066)
(25073,18721,25047,18656)
\N
(14524,13486,14510,13390)
(40040,36688,40000,36599)
(21594,11473,21563,11436)
(44031,22274,43938,22187)
(729,30683,668,30601)
(14114,20873,14102,20803)
(28239,41377,28222,41308)
(26404,11922,26317,11843)
(41660,34586,41585,34501)
\N
(21128,2384,21101,2368)
(30209,16952,30156,16858)
(39078,24963,39045,24898)
(5598,1348,5499,1294)
\N
(38474,7436,38450,7364)
(15117,45734,15024,45693)
\N
(23909,39853,23888,39780)
(24292,30183,24282,30148)
(48871,17661,48868,17637)
(918,18752,847,18708)
\N
(43615,16162,43606,16104)
(33763,47410,33751,47409)
(4798,6485,4773,6388)
\N
(18524,41539,18433,41518)
(47745,42449,47651,42364)
(38936,21237,38864,21204)
\N
(5251,3516,5194,3475)
(22269,36269,22183,36228)
(18736,40983,18685,40947)
(38393,15444,38356,15363)
(38134,29898,38103,29862)
(37789,39557,37732,39474)
(31906,23005,31838,23003)
(10647,40094,10560,40040)
(9914,41547,9867,41545)
(44221,443,44125,433)
(41479,10936,41381,10847)
(42586,6301,42563,6235)
(2504,17588,2449,17554)
(7045,18782,7028,18764)
(41840,32018,41768,31938)
(38416,17158,38330,17060)
\N
(8605,39015,8605,38933)
(5764,43548,5719,43496)
\N
(20789,29902,20696,29843)
\N
(36104,47896,36079,47816)
(31736,13834,31722,13832)
(32617,19701,32597,19684)
(1671,18997,1622,18945)
(36007,26545,36005,26535)
(31864,17494,31820,17455)
(27346,28388,27303,28289)
(8191,9653,8133,9589)
(7501,21616,7405,21536)
(35450,9580,35368,9563)
(29281,37276,29247,37255)
(6225,17192,6200,17135)
\N
(43689,8119,43670,8028)
(41917,49601,41835,49563)
(44295,13116,44205,13078)
(22721,44772,22667,44748)
(32640,11107,32636,11050)
(20639,28851,20613,28839)
\N
(32479,10159,32446,10061)
(27251,16978,27196,16959)
(41401,33148,41339,33074)
\N
(49001,8538,48989,8444)
(37958,35843,37874,35802)
(46969,41229,46903,41138)
(18541,8876,18541,8870)
(4080,31634,4061,31627)
(8097,35240,8040,35152)
(18470,21414,18463,21412)
(20914,17897,20838,17869)
(42688,11681,42666,11641)
(47525,25005,47443,24907)
(32439,14438,32397,14400)
\N
(39667,19626,39622,19542)
(1212,44525,1169,44516)
(29766,4433,29668,4401)
(25847,49657,25813,49605)
(33859,17356,33827,17263)
(28989,45953,28904,45854)
(37211,30830,37113,30819)
\N
(45220,26382,45219,26340)
(12312,43250,12234,43246)
(37775,41504,37762,41421)
(45889,33499,45822,33411)
(49461,22601,49369,22553)
(39857,33844,39816,33824)
(46102,15822,46030,15778)
(46605,31239,46598,31170)
(23925,5856,23862,5808)
(15459,4262,15407,4241)
(12019,4907,12015,4818)
(38258,17973,38229,17923)
(40575,29566,40477,29521)
\N
(29715,45919,29697,45891)
(11694,9510,11670,9490)
(7053,44257,7012,44231)
(16465,8603,16391,8505)
(29170,15592,29098,15527)
(20400,37354,20345,37328)
(5281,10265,5252,10184)
(6084,48782,6058,48727)
(11006,6889,10971,6796)
(16299,19461,16286,19411)
(13718,29192,13642,29106)
(3999,2965,3963,2903)
(18509,12235,18430,12208)
(49542,38575,49537,38534)
(15093,41715,15071,41634)
(6802,8385,6714,8300)
(15127,17507,15097,17424)
(36921,3025,36835,2995)
(32117,24327,32101,24262)
(27244,24151,27165,24104)
(36339,42360,36313,42358)
(47288,46252,47245,46184)
(37867,6649,37818,6565)
(14886,22103,14865,22089)
(39611,17952,39513,17951)
(37329,31436,37298,31436)
(5715,39115,5698,39099)
(13266,7364,13203,7296)
(16076,10945,16006,10942)
(7197,41509,7126,41413)
(14411,40868,14330,40772)
(12872,33481,12862,33454)
(17786,19616,17758,19560)
(1052,37358,996,37311)
(42825,12643,42762,12625)
(20007,49858,19921,49778)
(27155,6355,27072,6257)
(14117,40208,14022,40155)
(47280,34069,47279,34028)
(17551,15803,17482,15763)
(1725,6673,1676,6649)
(43984,31128,43961,31105)
(43772,47042,43731,47038)
(46901,47317,46817,47228)
(19877,14179,19837,14168)
(20691,19989,20675,19935)
(4011,18914,3963,18817)
(1023,23378,933,23317)
(30051,46118,29966,46039)
(43499,46488,43496,46409)
\N
(43531,2412,43447,2396)
\N
(16034,32285,15976,32220)
(12817,21365,12740,21298)
(7607,47293,7585,47293)
(32512,12218,32463,12170)
(1848,21496,1839,21439)
(17567,23073,17478,23046)
(35813,31847,35807,31792)
\N
(563,30859,540,30842)
(13145,15488,13063,15433)
(36754,37479,36731,37411)
(1125,26069,1057,25997)
(4539,20676,4519,20618)
(8476,34721,8409,34681)
(7794,25691,7727,25656)
(23842,514,23800,473)
(47678,41396,47668,41365)
(6837,25974,6799,25892)
(13355,11174,13304,11161)
\N
(37243,25548,37158,25471)
(12528,30208,12441,30205)
(14929,1672,14886,1607)
(27263,49026,27263,49010)
(15892,21645,15835,21642)
(29446,48978,29360,48967)
(41304,9892,41211,9825)
(37418,49393,37338,49296)
(41146,32178,41120,32165)
(28738,13326,28722,13266)
(14899,36595,14873,36559)
(1973,31435,1921,31426)
(19485,17742,19421,17661)
(33072,20995,32980,20903)
(47091,30055,47080,30037)
(45753,12998,45686,12992)
\N
(11528,7826,11509,7794)
(21104,13921,21060,13836)
(16768,15491,16747,15470)
(13279,20396,13249,20326)
(4342,49518,4339,49446)
(20413,15476,20349,15447)
(45532,5649,45484,5627)
(18647,27196,18619,27115)
(1326,17473,1261,17400)
(47646,19644,47588,19609)
(35088,1813,35080,1732)
(38461,34839,38410,34838)
(34358,11540,34285,11506)
\N
(26969,7078,26953,6989)
(12629,40352,12617,40264)
(33800,7037,33731,6992)
(24462,13518,24392,13486)
(33164,47357,33096,47329)
(15422,18451,15413,18376)
(19643,12916,19567,12912)
(40860,42125,40770,42050)
(49103,29614,49039,29606)
(36319,35582,36222,35528)
(8924,36083,8873,36018)
(49603,44022,49505,44021)
(7783,40633,7702,40618)
(25388,49107,25346,49042)
(28375,38947,28306,38919)
(47324,22672,47321,22660)
(2287,8808,2266,8719)
(44343,16339,44248,16318)
(2374,28839,2336,28798)
(22913,40710,22819,40688)
\N
(47747,684,47658,627)
(16043,46011,16021,45984)
(34958,32168,34903,32092)
(4840,49328,4752,49258)
(24341,2087,24330,2009)
(18378,19374,18327,19358)
(48165,7217,48156,7141)
(14232,6044,14182,6004)
(23080,4196,22983,4191)
(259,1850,175,1820)
(270,29508,264,29440)
(45088,11375,45050,11295)
(29666,39386,29656,39302)
(8712,8782,8660,8713)
(15900,6650,15855,6561)
(28946,28348,28917,28347)
(32544,25845,32538,25779)
(44047,6957,43951,6942)
(36465,588,36382,503)
\N
(28167,26679,28150,26673)
(16065,4268,15975,4180)
(12950,23494,12893,23494)
(30145,24679,30056,24654)
(3027,16162,3001,16071)
(8259,34537,8202,34484)
(41447,1515,41427,1454)
(18407,28362,18309,28303)
(21393,41872,21328,41816)
(46040,26497,45996,26408)
\N
(49944,25163,49902,25153)
\N
(16195,11843,16159,11831)
(44257,15270,44254,15214)
(49760,4791,49699,4713)
(22558,33709,22519,33681)
(28375,10003,28336,9938)
(18179,24310,18106,24256)
(707,30688,664,30669)
(5851,26118,5822,26037)
(4266,1292,4221,1217)
(16516,11331,16432,11248)
(32374,38277,32313,38245)
(21939,8015,21927,7952)
(34322,32051,34242,32003)
(6262,35977,6260,35953)
(16717,38594,16622,38498)
(14564,3433,14535,3425)
(21078,1000,20994,974)
(28584,956,28575,868)
(5538,9962,5465,9870)
(34183,44102,34175,44085)
\N
(42507,10289,42441,10288)
(12671,19936,12594,19920)
(24835,12179,24770,12173)
(15664,11538,15598,11494)
(28892,24446,28821,24350)
(41654,26720,41570,26632)
(36583,387,36503,357)
(10842,34824,10795,34788)
(11518,42588,11429,42565)
(12577,40322,12486,40266)
(2453,4045,2439,3956)
(31837,33705,31803,33681)
(24403,27711,24383,27705)
(4431,2748,4337,2656)
\N
(3036,2887,3014,2826)
(37664,16118,37615,16022)
(8606,18063,8587,18038)
(24738,25458,24656,25362)
(45756,34022,45671,33948)
(34079,15236,33981,15171)
(9251,22488,9228,22470)
(25136,2809,25126,2717)
(5548,47695,5543,47685)
(13765,40800,13707,40754)
(25216,30678,25144,30677)
(22441,17169,22392,17106)
(1091,4770,1054,4734)
(36311,50073,36258,49987)
(22461,33163,22457,33128)
(35873,28907,35845,28867)
(42907,15848,42904,15785)
(6549,24897,6540,24861)
(21928,37764,21891,37681)
(21237,41132,21139,41086)
(12207,24266,12173,24235)
(40643,49770,40574,49687)
(32833,35686,32815,35674)
\N
(14545,18143,14541,18098)
(33892,42783,33884,42707)
(33933,8381,33921,8369)
(12450,19044,12403,19002)
(10176,45158,10088,45145)
(35828,12080,35732,12022)
(28102,13694,28061,13666)
(49432,31744,49340,31711)
(16192,37743,16162,37697)
(46830,867,46756,790)
(9200,28048,9159,27986)
(13397,19369,13340,19288)
(30879,43562,30785,43545)
(21995,48224,21920,48143)
\N
(11871,47569,11809,47568)
(29366,22196,29280,22154)
(26243,28176,26203,28116)
(28995,35031,28906,35014)
(29384,39276,29352,39183)
(8497,13798,8471,13789)
(7412,27226,7334,27220)
(25403,47678,25363,47654)
(11599,5556,11574,5502)
(44056,5123,44008,5111)
(49603,30877,49579,30840)
(32261,45876,32206,45865)
(35104,41659,35048,41587)
\N
(5457,35844,5376,35782)
(29423,3977,29354,3959)
(18059,3001,17965,2961)
(8509,5691,8463,5620)
\N
(27118,5762,27083,5747)
(2991,48605,2939,48559)
(44482,3484,44425,3459)
(45143,16439,45046,16365)
(2236,37531,2147,37530)
(41561,3217,41490,3210)
\N
(6270,27200,6171,27166)
(49195,24871,49138,24798)
\N
(46985,38881,46897,38845)
(37486,23522,37404,23441)
(26907,14490,26900,14391)
(30829,16111,30756,16056)
(3644,17291,3587,17262)
(20508,49775,20472,49680)
\N
(43279,8972,43198,8936)
(33744,7470,33734,7439)
(46303,20538,46284,20498)
(10365,48246,10291,48154)
(12636,24987,12545,24933)
(40998,46992,40989,46916)
(30536,6073,30531,6018)
(22102,9643,22051,9594)
(18616,34348,18530,34332)
(8222,8907,8123,8848)
(45698,28860,45698,28770)
(26958,1748,26924,1726)
\N
(26735,35073,26659,35025)
(48370,40813,48293,40737)
(13140,993,13108,934)
(10588,22893,10528,22883)
(23645,40789,23567,40698)
(49548,12374,49546,12329)
(41135,39626,41100,39602)
(41374,10856,41328,10769)
(12234,5765,12146,5674)
(12832,46941,12764,46917)
(47886,34532,47851,34500)
(23777,10549,23735,10495)
(1291,16913,1194,16873)
\N
(29239,30554,29202,30500)
\N
(36485,30007,36454,29924)
(7067,11320,7045,11229)
(16939,30482,16904,30462)
(27423,34386,27379,34303)
(35170,32021,35155,31979)
(42570,36477,42474,36457)
(19695,679,19682,594)
(47537,39450,47446,39450)
(19410,22942,19375,22922)
(34216,40166,34152,40158)
(37000,24351,36972,24299)
(24989,1681,24954,1672)
(54,38679,3,38602)
(41461,40693,41411,40599)
(7576,46054,7545,45963)
(35505,28262,35413,28222)
(1158,16976,1145,16927)
(23494,42291,23437,42229)
(32894,32519,32880,32485)
(604,13413,509,13401)
(18396,19712,18355,19646)
\N
(26657,28234,26597,28191)
(24240,47211,24154,47191)
(41778,10741,41766,10730)
(44022,43776,44010,43677)
(35967,30055,35906,29969)
(28878,18042,28806,18027)
(31507,27302,31428,27267)
(13267,21935,13265,21872)
(122,46832,64,46762)
(10348,45916,10306,45844)
(22962,12644,22927,12607)
(6320,22290,6284,22247)
(2297,11372,2216,11298)
(29366,36660,29325,36654)
(13962,39307,13921,39220)
(11094,19151,11092,19143)
(32289,23776,32258,23760)
\N
(36044,17356,35956,17273)
(46304,38692,46232,38675)
(10934,42999,10922,42909)
(4271,21177,4207,21093)
(7837,19926,7747,19905)
(25537,36605,25477,36584)
(22161,14999,22079,14962)
(5127,31243,5074,31213)
\N
(14904,40664,14838,40593)
(29308,8480,29268,8438)
(17731,7410,17699,7352)
(44840,29293,44797,29248)
(15523,31519,15505,31485)
(34429,38479,34421,38478)
(3530,23456,3440,23390)
(4699,6889,4603,6796)
(47405,48524,47389,48514)
\N
(23357,43160,23305,43156)
(16923,1995,16860,1937)
(47592,33853,47537,33758)
(31624,37490,31595,37473)
(42321,13380,42303,13337)
(3088,16094,3079,16060)
(22884,2955,22856,2857)
(17784,23073,17724,23044)
(32638,45577,32553,45512)
(13876,44091,13801,44000)
(27844,24384,27758,24330)
(28178,10225,28155,10167)
(39910,14277,39857,14241)
(30372,19524,30301,19514)
(38732,43151,38724,43151)
(32628,2068,32547,2068)
(13950,28652,13932,28566)
(38996,41070,38919,40993)
(31759,45246,31676,45215)
\N
(5424,34145,5382,34106)
(14727,45600,14699,45547)
\N
(31429,21537,31414,21499)
(14740,3420,14650,3323)
(21793,39498,21743,39471)
(18102,25924,18037,25868)
(33299,683,33213,594)
(45882,48765,45809,48721)
(49215,4098,49180,4067)
(49698,33743,49614,33663)
(21532,5215,21514,5151)
(24840,26877,24826,26808)
(32680,28433,32631,28364)
(20661,27511,20584,27414)
(28048,30385,28009,30315)
(45403,42533,45389,42464)
(46531,36947,46531,36850)
(36943,32817,36865,32737)
\N
(37984,43763,37888,43748)
(20593,10650,20557,10610)
(5387,40595,5326,40585)
(34412,10600,34352,10539)
(7237,47546,7206,47451)
(39931,26644,39915,26598)
(29843,4734,29800,4669)
(37503,8867,37406,8821)
(2583,2373,2570,2294)
(29275,46433,29256,46350)
(3332,45620,3287,45581)
(22472,39287,22472,39257)
(36786,18907,36708,18884)
(45503,28576,45482,28494)
(33262,28386,33163,28365)
(3606,49757,3538,49697)
(2082,49380,1991,49281)
(12065,3734,11983,3663)
(15606,9048,15596,9028)
(14687,19309,14637,19263)
(4568,15461,4499,15428)
\N
(43938,7429,43923,7391)
\N
(2168,50012,2108,49914)
(16022,8934,15963,8928)
(24567,39147,24561,39102)
\N
(42781,14149,42765,14088)
(39501,21084,39468,21078)
(6697,29628,6693,29584)
(11441,16164,11364,16125)
(39946,1920,39868,1844)
\N
(18138,45512,18111,45438)
\N
(20799,41217,20718,41138)
(30264,16697,30240,16639)
\N
(30746,50040,30727,49992)
(37429,43273,37423,43205)
(22854,28863,22789,28810)
(11380,48298,11287,48242)
(16471,37273,16439,37223)
(32737,39842,32661,39811)
(30959,3447,30949,3357)
(36396,13263,36348,13187)
(29607,14625,29531,14619)
(7851,43399,7824,43334)
(38515,14575,38496,14492)
(29125,3289,29086,3264)
(6866,10476,6839,10424)
(318,31489,235,31404)
(1140,7007,1113,6945)
(36574,9291,36484,9275)
\N
(40320,40937,40246,40866)
(588,25849,552,25801)
(6728,42539,6645,42507)
(12180,6185,12123,6123)
(32913,44123,32899,44037)
(25464,16803,25441,16749)
(23711,5829,23695,5750)
(31424,34930,31377,34906)
(42171,8298,42124,8222)
(451,31104,375,31083)
(39996,3278,39943,3260)
(25816,40396,25735,40362)
(34471,28587,34399,28547)
(45344,21540,45297,21496)
(27269,16787,27246,16763)
(18070,4469,18022,4423)
\N
(12668,16367,12645,16295)
(13823,17276,13730,17251)
(20555,45544,20511,45498)
(35893,42189,35861,42177)
(37081,45730,37076,45705)
(17270,15651,17201,15552)
(48690,46034,48667,45945)
(456,16088,368,16023)
(48707,12416,48670,12363)
(29692,11509,29614,11483)
(7005,3668,6981,3574)
(12162,389,12103,309)
(12371,24983,12366,24964)
(6886,48414,6868,48327)
(10653,26234,10624,26142)
(8526,48205,8517,48117)
(10521,31892,10480,31798)
(43353,1086,43281,1071)
(21007,35650,20998,35649)
(2343,4396,2310,4320)
(29379,12895,29284,12891)
(27662,17407,27570,17313)
(9845,29346,9807,29321)
(43855,38669,43790,38599)
\N
(20461,44189,20397,44158)
(11627,17368,11581,17289)
(2971,38855,2938,38807)
(43204,47082,43128,47018)
(9930,46902,9909,46871)
(30561,48461,30536,48365)
(44059,7591,44038,7563)
(46260,16898,46162,16886)
(27491,2891,27396,2814)
(36512,26034,36455,25941)
(31193,20022,31100,19942)
(17057,13643,16960,13621)
(26897,3399,26844,3318)
(1760,5504,1683,5431)
(29347,5511,29346,5450)
(38761,42083,38688,41999)
(11226,4089,11165,4068)
(46427,42983,46361,42970)
(12958,30737,12912,30712)
(44432,46521,44333,46443)
(16124,2948,16113,2852)
\N
(24704,25422,24635,25340)
(30833,46152,30790,46122)
(4487,37006,4473,36968)
(41047,23376,41036,23327)
(16312,49392,16298,49330)
(30081,14687,30042,14660)
(11160,13954,11103,13938)
(33207,23246,33143,23168)
(14872,7635,14860,7585)
(20139,23987,20059,23955)
(10946,49757,10923,49746)
(39438,36158,39426,36134)
(35502,2385,35464,2327)
(17073,42173,16987,42130)
(6079,17258,6068,17195)
(40458,15752,40364,15728)
(23340,7879,23313,7806)
\N
(31819,15096,31762,15059)
(31159,40864,31158,40780)
(26975,32144,26915,32113)
(34530,10378,34440,10298)
(18855,49577,18780,49528)
(16787,16625,16723,16586)
(32330,26538,32314,26458)
(34270,28674,34265,28595)
(10022,16026,10006,15962)
(23143,1479,23095,1469)
(33676,4483,33583,4408)
(31066,22074,31059,22035)
(21603,47121,21563,47082)
(30051,4244,30021,4157)
(30634,39478,30615,39446)
(34404,48724,34393,48724)
(31103,21414,31039,21380)
(22945,47397,22849,47313)
(18133,32025,18073,31941)
(4053,25759,3977,25667)
(39185,39091,39102,39068)
(43287,7407,43225,7314)
(13137,31188,13112,31182)
(46264,1438,46258,1389)
(22804,43892,22769,43822)
(7542,1044,7487,983)
(33022,8321,32925,8267)
(384,39161,286,39073)
(28205,24401,28142,24382)
(31708,39086,31696,39026)
(36626,15708,36560,15690)
(17099,16924,17079,16924)
(10817,6989,10747,6955)
(24338,19293,24291,19277)
(27566,17576,27544,17545)
(23041,38384,22970,38320)
\N
(12786,8485,12702,8435)
(13876,49473,13813,49448)
(31585,46998,31490,46929)
\N
(30227,8768,30206,8715)
(32062,39306,32023,39292)
(25003,35753,24921,35687)
(3281,6758,3232,6704)
\N
(11395,30299,11376,30220)
(5088,15275,5007,15203)
(31100,39538,31003,39444)
(2741,17877,2726,17793)
(42897,48620,42860,48537)
(4230,15778,4181,15776)
(17835,27530,17815,27431)
(34189,10933,34135,10921)
(7537,39974,7494,39973)
(21554,3507,21528,3476)
(9350,32326,9273,32275)
(16455,8874,16420,8793)
\N
(7346,34235,7330,34224)
(16417,48134,16352,48066)
\N
(41916,4971,41849,4886)
(15856,1522,15807,1521)
(41549,40218,41494,40144)
\N
(9978,16226,9972,16181)
(14856,13312,14808,13283)
(38490,41641,38428,41583)
(25828,7438,25807,7378)
(21876,30633,21796,30587)
(1908,14279,1825,14247)
\N
(32207,10251,32121,10184)
(370,9493,328,9441)
(42072,17634,41974,17600)
\N
(47298,9910,47235,9846)
(17856,11266,17782,11225)
(35009,21400,34956,21396)
(18337,11145,18335,11133)
\N
(25425,9139,25381,9085)
(35642,27783,35621,27782)
(3629,33164,3575,33163)
(17151,41255,17115,41204)
(17417,5835,17402,5751)
(33407,14226,33329,14141)
(1930,29955,1889,29931)
(41101,10942,41065,10844)
(36333,27288,36281,27233)
(21423,36868,21367,36825)
(36385,19566,36341,19510)
(27073,38301,27066,38232)
(43989,34187,43984,34174)
(48366,7488,48316,7483)
(37497,36075,37415,36043)
(46917,9891,46887,9870)
(37179,657,37103,634)
(3877,44736,3811,44684)
(30556,2975,30547,2962)
(7629,11447,7547,11416)
(45687,48147,45591,48088)
(5635,7184,5571,7146)
(9611,47327,9541,47246)
(7119,48224,7117,48152)
(15233,26480,15138,26430)
(37468,1526,37466,1513)
\N
(20855,2786,20828,2711)
(30538,44084,30480,44061)
(42231,41527,42149,41454)
(14963,13239,14952,13146)
(26819,43996,26745,43934)
(42172,35953,42086,35928)
(28785,12611,28710,12534)
(14089,1704,14047,1629)
(4343,26242,4341,26169)
(20327,42244,20231,42212)
(33671,12700,33666,12630)
(42144,32642,42128,32569)
(26590,19483,26503,19442)
(21741,46259,21723,46226)
(8822,34700,8760,34693)
\N
(2710,33521,2675,33505)
(26067,19998,26026,19989)
(12244,34509,12202,34489)
\N
(47162,598,47119,499)
(33093,49382,33068,49359)
(35170,26340,35153,26264)
(22552,35785,22490,35735)
(36791,23032,36781,22976)
(22857,10857,22833,10797)
\N
(47207,37405,47138,37365)
(21867,2836,21854,2811)
(3387,31487,3311,31456)
(47174,48121,47167,48101)
(24415,22232,24366,22224)
(7970,29251,7959,29211)
(18635,31294,18539,31221)
(8403,13380,8370,13372)
(738,18097,737,18054)
(37238,19195,37218,19114)
(582,47934,570,47897)
(12359,4635,12350,4619)
(43272,2013,43195,1958)
(47568,27149,47521,27088)
(24695,12827,24661,12796)
(26259,14077,26168,14019)
\N
(48478,36135,48425,36092)
(5230,39250,5206,39174)
(3488,18562,3423,18489)
(39502,16331,39460,16275)
(18296,1478,18233,1471)
\N
(28627,12430,28559,12410)
(25257,21981,25206,21954)
\N
(2410,41192,2325,41142)
(43681,9631,43587,9538)
\N
(15086,45309,15064,45270)
(13824,40807,13759,40787)
(7090,2207,7062,2159)
(3685,2480,3630,2391)
(14810,38335,14801,38275)
(26668,38018,26581,38012)
(45562,1517,45506,1424)
(11001,32481,10962,32402)
(27743,25245,27673,25161)
(15952,10598,15948,10535)
(12705,13308,12694,13232)
(31992,21195,31975,21118)
(25834,16652,25745,16626)
(21022,43625,20990,43576)
(45094,27254,45000,27240)
(9688,42601,9643,42533)
(17746,24659,17694,24616)
(1509,38859,1503,38809)
(2067,20438,2041,20369)
(7885,44528,7839,44444)
(27432,33052,27422,32987)
(26577,17157,26563,17142)
(10815,35985,10734,35908)
(44891,24067,44794,23979)
(48626,1900,48595,1850)
\N
(40659,35541,40659,35489)
(22231,26628,22210,26579)
(37408,23016,37375,22919)
(5920,15916,5906,15895)
\N
(33125,9952,33037,9880)
(12142,29705,12141,29670)
(3672,20995,3649,20899)
(39147,31967,39101,31907)
\N
(33812,48458,33748,48399)
(25038,14639,24978,14586)
(3859,16010,3857,15994)
(31926,39496,31889,39417)
(49300,28064,49297,28026)
(24121,38305,24048,38256)
(9252,4205,9155,4149)
(36124,30451,36056,30395)
(28809,49557,28794,49533)
(30500,44504,30471,44476)
(26866,42395,26822,42332)
(48195,1784,48101,1734)
(46201,14109,46112,14097)
\N
(2415,9975,2354,9914)
(30485,9581,30415,9558)
(6385,36838,6305,36838)
(2799,11189,2723,11095)
(21998,20503,21923,20406)
(29151,10714,29090,10671)
(28850,29276,28757,29207)
(43386,48845,43305,48834)
(25173,8310,25101,8294)
(34244,32352,34204,32342)
(35595,23728,35533,23672)
(1122,13581,1119,13538)
\N
(388,21716,296,21678)
(48782,11064,48701,11005)
(40293,12997,40213,12927)
\N
(28194,46428,28113,46414)
(4791,18118,4708,18105)
(471,29808,448,29775)
(3536,37803,3447,37737)
(1336,28416,1275,28392)
(16484,48478,16422,48454)
(25846,19320,25811,19296)
(48669,27703,48575,27615)
(24032,44217,24029,44127)
(12236,5019,12233,4986)
(1179,29838,1113,29778)
(33893,22049,33867,21955)
(16718,19462,16700,19440)
(17992,49438,17894,49433)
(35163,39941,35081,39885)
(33897,8362,33853,8328)
(2480,6640,2456,6599)
(28011,19729,27937,19679)
(15819,41516,15809,41440)
(29818,9136,29747,9089)
(28551,37016,28529,36941)
(36406,26879,36374,26872)
(16821,48925,16758,48914)
(23692,48163,23595,48160)
\N
(4803,10619,4759,10522)
(46600,33581,46553,33518)
(41349,11767,41310,11710)
(20856,29642,20799,29562)
(16559,46161,16504,46131)
(23041,1300,23003,1287)
(16630,44902,16554,44853)
(43065,14299,43013,14274)
(24818,22397,24796,22348)
(22282,24949,22218,24921)
(36668,28538,36631,28456)
(8080,1220,8018,1146)
(47282,34302,47277,34269)
(35603,33558,35557,33495)
(44764,32189,44700,32175)
\N
(46488,23965,46449,23868)
(46314,15047,46216,15013)
(6348,25381,6286,25363)
(3871,49288,3819,49251)
(462,38894,398,38867)
(23196,29214,23136,29169)
(29024,9775,29016,9759)
(42016,18555,41934,18472)
(8772,45981,8692,45973)
(11028,1351,10986,1278)
(26684,21668,26641,21656)
\N
(37262,26005,37260,25947)
(14899,44069,14814,44066)
\N
(39635,18701,39587,18698)
(28528,22948,28457,22857)
(7755,36528,7681,36454)
(32461,1172,32427,1106)
\N
(18775,27359,18736,27329)
(15379,20031,15337,19934)
(45888,33592,45881,33544)
(44013,24694,43962,24645)
\N
(43347,10699,43343,10699)
(49999,27218,49908,27176)
(13698,17326,13630,17317)
(34850,44313,34775,44302)
(38076,49235,37983,49214)
(35570,40218,35500,40136)
(40062,28973,40032,28878)
(3567,39847,3523,39781)
(498,2442,480,2401)
(29660,43620,29577,43561)
(10946,47356,10878,47351)
(8073,44233,8005,44144)
(9720,13473,9710,13462)
(3643,38014,3598,37932)
(16887,1408,16810,1375)
(7559,27914,7508,27874)
(30356,18573,30275,18569)
(12193,48176,12130,48116)
(11884,7756,11819,7731)
(18293,33272,18227,33234)
(46697,47874,46696,47828)
(35788,32517,35760,32446)
(33877,36987,33821,36958)
(31253,22819,31184,22808)
(7744,23115,7729,23103)
(21291,39817,21219,39778)
(13877,43379,13861,43290)
(42955,1406,42876,1382)
(49232,15950,49210,15880)
(48419,32001,48326,31902)
(18940,43246,18860,43150)
(32317,38240,32310,38201)
(11307,48298,11304,48222)
(38015,18190,38000,18176)
(27821,1177,27818,1131)
(18935,26757,18865,26682)
(42659,48284,42562,48244)
(30185,23350,30146,23291)
\N
(16496,11970,16441,11919)
(162,26040,120,25963)
(24238,47784,24185,47746)
(32326,8612,32274,8568)
(26141,13423,26051,13407)
(40132,22815,40089,22812)
(21151,48794,21056,48740)
\N
(22044,28358,22031,28334)
(6680,14746,6605,14669)
(40686,25139,40632,25070)
(22823,27549,22816,27507)
(2513,22841,2427,22811)
(36316,27787,36218,27728)
(554,35489,540,35441)
(536,30674,534,30609)
\N
(25385,38468,25295,38416)
(19467,47386,19437,47317)
(22425,38591,22387,38536)
(32493,17321,32396,17298)
(40115,47315,40109,47235)
(25002,2107,24963,2104)
(3901,9790,3898,9706)
\N
(40316,1721,40315,1658)
(40089,3454,40074,3443)
(793,17897,761,17897)
(6490,43552,6434,43522)
(10825,487,10820,405)
(47703,36067,47641,36011)
\N
(4480,11671,4468,11653)
(37713,10642,37711,10615)
(12315,5302,12273,5203)
\N
(8709,6617,8647,6557)
(24467,30535,24455,30494)
(40440,32757,40369,32668)
(49449,42447,49426,42428)
(44867,11197,44792,11137)
(39173,33241,39143,33187)
(43836,2212,43803,2184)
(23819,47613,23739,47575)
(20583,2134,20485,2042)
(48922,6169,48889,6111)
(5230,44613,5131,44604)
(37060,8051,37032,7975)
(19148,36711,19112,36704)
(36305,4216,36243,4118)
(6329,39089,6302,39047)
(36703,26367,36623,26307)
(44753,19721,44701,19631)
(42094,43310,42094,43285)
(4276,22377,4241,22352)
(30329,18906,30327,18815)
(21970,19605,21871,19590)
(23722,41924,23709,41861)
(30965,39775,30908,39692)
(32394,37895,32351,37890)
(23968,42162,23873,42095)
(1776,2621,1732,2548)
(24951,47758,24900,47679)
(32917,35771,32847,35753)
(5428,27773,5343,27769)
\N
(19650,142,19630,51)
(39769,17276,39743,17229)
(5171,24562,5119,24470)
(32976,35249,32917,35199)
\N
(4174,24603,4099,24504)
(38565,36960,38535,36926)
(39084,4328,39031,4301)
(32153,38043,32070,37990)
(38085,30640,38041,30603)
(14269,18426,14185,18422)
(42941,30850,42892,30788)
(32403,25999,32339,25960)
(16906,191,16816,139)
(3456,48722,3418,48721)
(3050,18287,3022,18243)
(6331,8439,6234,8364)
(5331,20797,5319,20793)
(39225,37408,39216,37348)
(34510,19838,34488,19810)
(45789,33873,45770,33786)
(369,1457,278,1409)
(16531,43785,16482,43729)
(11974,14789,11973,14730)
(23128,6811,23094,6798)
(43962,33659,43944,33599)
(20967,3115,20947,3079)
(39257,38606,39241,38595)
(22431,8246,22381,8235)
(26007,14672,25996,14593)
(24762,4261,24675,4261)
(35402,32077,35343,31988)
(5141,16476,5139,16393)
(16439,17564,16344,17472)
(36983,46663,36903,46567)
(35170,14144,35162,14048)
(22290,7841,22283,7810)
(22414,38398,22404,38319)
(9011,18177,8932,18150)
\N
(154,4019,138,3990)
(20447,4998,20383,4970)
(38867,35757,38795,35659)
(32322,15845,32227,15804)
\N
(29889,12142,29852,12055)
(36235,36918,36217,36897)
(41620,6581,41568,6581)
(24758,38504,24731,38483)
(42524,12904,42473,12895)
(17954,49975,17865,49915)
(1938,39019,1927,39013)
(4864,33279,4817,33258)
(45373,41967,45313,41885)
(28786,19028,28782,18978)
(41913,44950,41911,44908)
(33408,14698,33392,14681)
(27602,3460,27576,3419)
(3336,3728,3334,3715)
(9099,910,9080,813)
(34141,6403,34071,6367)
(48270,17216,48252,17130)
(2549,16546,2461,16474)
(27802,33669,27735,33642)
(48419,1682,48323,1583)
(5094,41211,5002,41123)
(11192,6217,11190,6146)
(6979,18503,6959,18421)
(41210,48187,41140,48143)
(15303,29527,15273,29441)
(12326,45572,12267,45570)
(29293,5861,29212,5826)
(23847,37241,23761,37178)
(44656,23926,44653,23831)
(30043,16194,29977,16105)
(902,9358,879,9339)
(23850,46501,23834,46494)
(42333,13300,42287,13246)
(25226,18086,25169,18005)
(40252,12082,40183,12038)
(49275,18076,49216,18055)
(8255,28878,8238,28862)
(11325,41286,11320,41235)
(16948,18588,16926,18528)
(31394,1099,31374,1038)
(30705,35772,30637,35766)
(3858,39131,3771,39125)
(17565,24892,17515,24808)
(9221,49715,9216,49661)
(44945,25769,44875,25722)
(33408,13563,33310,13527)
(48505,4407,48408,4373)
(21859,37217,21763,37217)
(39393,14422,39335,14364)
\N
(19905,1154,19841,1098)
(25946,10388,25906,10366)
(10104,13748,10027,13746)
(5822,24629,5820,24599)
(38194,11287,38127,11252)
(15694,46757,15625,46716)
(326,18837,285,18817)
(49611,47078,49533,47052)
(48233,18850,48150,18842)
\N
(29239,9962,29208,9875)
(40062,44554,39973,44460)
(19135,20729,19059,20643)
(31969,40664,31896,40643)
\N
(3725,9191,3711,9095)
(44280,40158,44264,40108)
(37236,42756,37160,42694)
(27958,19055,27888,18959)
(45270,17661,45187,17601)
(12115,39546,12061,39525)
(10227,32295,10168,32231)
(39264,31123,39226,31085)
(6566,40000,6532,39904)
(30058,6975,30012,6903)
(49631,6909,49597,6823)
(42168,10926,42134,10905)
(44892,30042,44858,29970)
(19540,19803,19495,19788)
(18403,25454,18371,25404)
(22929,26795,22841,26722)
(16648,30213,16626,30174)
(3440,7495,3429,7468)
(30708,49028,30643,48998)
(26258,14164,26255,14151)
(44206,31653,44121,31637)
(1510,15179,1426,15130)
(6986,30496,6887,30416)
(7192,43403,7138,43339)
(39921,22071,39866,21976)
(45870,17011,45796,16919)
(15939,9563,15917,9539)
(23728,24737,23691,24725)
(6444,40416,6363,40375)
(21899,23861,21857,23765)
(20610,36765,20533,36742)
(46520,33082,46433,32983)
(21406,20902,21311,20895)
\N
(37913,42300,37814,42269)
(18216,8177,18161,8173)
(32967,8258,32899,8244)
(14978,40230,14971,40149)
(30343,39152,30266,39101)
(25917,5835,25843,5806)
\N
(5169,45366,5141,45314)
\N
(16221,20898,16209,20875)
(13151,19869,13145,19811)
(44399,2801,44337,2713)
\N
(10959,48311,10957,48230)
(4794,11711,4732,11661)
(764,10149,762,10091)
(15985,46067,15898,46028)
(41434,22870,41342,22867)
(43769,23796,43743,23756)
(10017,18440,9919,18384)
(21141,43119,21097,43112)
(7782,13424,7694,13398)
(25088,36224,25059,36150)
(46325,48722,46241,48631)
\N
(11042,33125,11011,33071)
(22347,13460,22290,13375)
(3508,20538,3483,20536)
(5331,42945,5272,42875)
\N
(2368,15537,2339,15503)
(45314,31830,45254,31817)
(34358,2649,34319,2589)
\N
(17576,30407,17572,30323)
(29836,41324,29746,41287)
(21036,39996,21014,39899)
(26886,6460,26787,6400)
(15709,5625,15627,5558)
(37415,15979,37414,15911)
(47761,16860,47728,16813)
(35814,48252,35755,48173)
\N
(28559,20810,28496,20715)
(12034,11921,12002,11905)
(1818,27450,1805,27406)
(33810,45499,33806,45413)
(17376,18175,17323,18138)
(34106,28135,34049,28106)
(44947,23165,44919,23091)
(37670,41904,37616,41840)
(12614,15027,12555,14969)
(43301,75,43227,43)
\N
(27526,15096,27450,15088)
(26947,33409,26853,33333)
(1537,43572,1471,43499)
\N
(21607,35452,21605,35375)
(24869,46565,24818,46531)
(4774,30335,4723,30257)
(11615,18316,11579,18310)
(18444,15819,18354,15763)
(47267,22574,47203,22518)
(22287,49538,22203,49511)
(43010,16270,43010,16202)
\N
(1623,8350,1578,8254)
(21220,43808,21137,43748)
(40397,16471,40358,16434)
\N
(34839,1377,34744,1327)
(17096,5730,17090,5637)
\N
(28156,37782,28155,37723)
(3672,5686,3586,5638)
(21856,48656,21840,48638)
(6907,7791,6892,7761)
(17952,21370,17862,21350)
(37793,13461,37784,13381)
(14740,49655,14709,49604)
(21690,6337,21593,6289)
\N
(10423,33548,10364,33498)
\N
(39187,23274,39136,23197)
\N
(21882,37247,21835,37167)
\N
(11343,16957,11281,16914)
(38279,43400,38264,43352)
(23167,30271,23086,30224)
(46278,6037,46180,5964)
(28626,31165,28605,31095)
\N
(31018,367,30946,333)
(23541,12541,23530,12523)
(49741,14535,49691,14511)
(31444,12702,31425,12612)
\N
(22406,26536,22316,26534)
(6807,9761,6758,9723)
(15698,1941,15687,1848)
(49310,4625,49295,4584)
(21345,18939,21269,18887)
(31433,30493,31411,30439)
(44980,12400,44950,12372)
(25054,13949,24984,13949)
(40538,7253,40483,7212)
(16967,8627,16936,8604)
(26872,3646,26804,3594)
(24575,42883,24530,42883)
(11823,5755,11771,5721)
\N
(2553,46189,2513,46174)
(24993,14552,24898,14470)
(28453,1719,28419,1665)
(8925,22603,8878,22589)
(47635,15380,47546,15378)
(35378,18112,35324,18058)
(27347,22264,27293,22200)
\N
(44323,29044,44273,28958)
(41538,38324,41484,38290)
(19128,49932,19112,49849)
(17904,12548,17867,12503)
(35103,14426,35092,14336)
(29807,10142,29714,10052)
(44507,22903,44462,22847)
(11419,13324,11399,13251)
(8573,42221,8562,42123)
(46798,45843,46765,45765)
(12028,31783,11967,31749)
(10635,45300,10604,45251)
(9626,8248,9587,8194)
(18290,741,18246,732)
(39949,44672,39932,44641)
(7897,11692,7893,11637)
(20165,42246,20112,42168)
(4341,48390,4285,48338)
(30126,28913,30088,28869)
(40565,1733,40472,1721)
(9981,30147,9915,30133)
(47292,25511,47217,25462)
(20137,24489,20104,24392)
(2385,28283,2381,28189)
(20429,10052,20357,10009)
(8395,38568,8348,38480)
(17381,36112,17349,36038)
(37845,30953,37759,30926)
(27452,12732,27411,12652)
(38196,32186,38114,32116)
\N
(6527,49356,6508,49315)
(43891,29789,43856,29723)
(6146,37192,6085,37107)
\N
(42012,28897,41939,28808)
\N
(14909,13815,14846,13757)
(11120,24095,11035,24049)
(3132,41545,3053,41526)
(40084,40315,39994,40261)
(39671,17445,39576,17361)
(47135,35853,47085,35831)
(39297,1941,39290,1911)
(47143,35898,47072,35880)
(16017,6711,15989,6686)
(47110,30305,47087,30213)
(38102,27639,38091,27602)
(17954,22544,17863,22453)
(39891,11791,39815,11739)
(13996,20290,13922,20278)
(22284,23143,22190,23081)
(25345,24019,25313,24017)
(47134,44803,47055,44761)
(41360,16573,41326,16503)
(10464,1071,10457,998)
\N
(23515,47517,23451,47499)
(9308,8452,9238,8392)
(28695,5657,28671,5644)
(45104,9913,45077,9871)
(337,455,240,359)
(11562,45479,11472,45428)
(11952,18466,11931,18425)
\N
(35789,5154,35775,5128)
(19024,18299,18979,18230)
(43056,38113,42975,38067)
(10075,26847,10064,26806)
(3065,8107,3029,8038)
(24766,19059,24749,18985)
(14438,24805,14413,24708)
(9523,3058,9485,2998)
(24516,31262,24478,31204)
(49513,26044,49434,26035)
(14110,38528,14103,38461)
(31679,35618,31619,35618)
(10029,20258,10008,20248)
(39269,37586,39233,37539)
(12343,8197,12247,8113)
(11155,44223,11111,44134)
(25437,20606,25338,20534)
(46604,16156,46570,16131)
(4636,14004,4592,13941)
(15975,29628,15912,29556)
(49887,24274,49805,24184)
(11812,13440,11723,13418)
(21589,38179,21531,38085)
(32255,44463,32219,44454)
(15023,12698,14989,12687)
(28906,48630,28818,48568)
(28886,38905,28861,38832)
(34786,22285,34740,22240)
\N
(46513,46780,46425,46780)
\N
(26626,31759,26551,31677)
(19792,25967,19763,25933)
(20432,14394,20388,14365)
(27092,7301,27052,7278)
(22283,987,22198,928)
(6197,24363,6112,24311)
(46601,49259,46551,49231)
(12392,48052,12363,48038)
(46116,31386,46067,31356)
(7354,16855,7289,16778)
(47501,42808,47495,42761)
(16461,25487,16391,25398)
(42678,18798,42678,18756)
(9466,18207,9419,18185)
(17467,14177,17416,14097)
(28533,31886,28487,31832)
(13225,38472,13188,38395)
(5180,40970,5173,40902)
(83,10271,15,10265)
(2111,6784,2016,6690)
(41835,11064,41798,10995)
(29273,48585,29181,48536)
(29066,21615,28985,21543)
(19805,44143,19727,44128)
(48919,21468,48875,21467)
(28790,34287,28721,34251)
(10911,33074,10869,32989)
(6111,16519,6032,16489)
(43889,33838,43837,33768)
(32323,21685,32304,21644)
(9552,27819,9539,27753)
(38266,49852,38233,49844)
(37672,48362,37663,48277)
(32550,47029,32529,46931)
(46307,6620,46272,6616)
(23192,46608,23105,46566)
(30399,48330,30335,48239)
(36268,25058,36235,24984)
(19181,8120,19089,8098)
(24376,19983,24294,19925)
(18297,18375,18202,18292)
\N
(31608,6215,31575,6168)
(12788,49510,12784,49468)
(46071,13013,46035,12991)
(27647,8218,27582,8201)
(49580,11076,49537,11050)
\N
(35501,33782,35501,33687)
(19969,3148,19964,3082)
(37728,49153,37726,49152)
(5322,48440,5321,48435)
(48003,10096,47904,10005)
(39361,22318,39348,22236)
(30488,7456,30437,7430)
(18533,39476,18481,39394)
(39462,23701,39433,23604)
(26701,18300,26686,18235)
(17405,35577,17387,35517)
(33971,29928,33953,29919)
(6328,10241,6276,10217)
(32459,44259,32453,44217)
(1715,42385,1647,42357)
(48113,6960,48103,6872)
(30561,4255,30476,4240)
(38907,43619,38827,43553)
(29149,20773,29070,20698)
(17006,1543,16970,1497)
\N
(11737,18808,11714,18788)
(13019,30534,13005,30481)
(39224,31729,39191,31683)
(4942,41680,4907,41596)
(12287,37187,12188,37172)
(30758,29579,30725,29531)
\N
(16604,17963,16581,17912)
(19459,15888,19409,15812)
(34696,24783,34600,24725)
(21621,14159,21558,14110)
(12193,46149,12145,46096)
(37781,4715,37692,4635)
(41854,44125,41807,44040)
(23604,23585,23571,23533)
(7853,36967,7797,36908)
(2755,13279,2720,13206)
(4314,15424,4283,15383)
(29584,12685,29493,12594)
(25138,33726,25042,33691)
(38393,10270,38326,10185)
(4247,12615,4225,12567)
(36100,33156,36100,33107)
(20024,40796,20016,40708)
(3927,44892,3914,44843)
(10317,43168,10226,43096)
(22057,3419,22042,3334)
(37097,21814,37025,21811)
(32084,21564,31996,21491)
(34079,39921,34058,39911)
(23078,47459,23018,47373)
(38109,616,38082,568)
(11862,40382,11764,40292)
(33403,33320,33389,33289)
(36639,24829,36623,24829)
(12995,45080,12992,45040)
(16545,19981,16532,19891)
(26155,10659,26154,10634)
(24423,255,24360,213)
(823,22487,781,22442)
(12823,20064,12735,20040)
(19688,11710,19681,11654)
(2892,20452,2836,20424)
(15533,10807,15464,10711)
(46994,41143,46955,41082)
(18155,2421,18069,2392)
(2628,12688,2605,12602)
(35128,8396,35044,8365)
(44765,49615,44758,49524)
(11226,44529,11178,44515)
(31334,32463,31291,32456)
(43224,23387,43168,23364)
(30882,10414,30798,10395)
(29139,967,29139,923)
(29959,45244,29877,45223)
(19946,217,19941,118)
(49732,22033,49642,22012)
(32914,15360,32879,15290)
(47825,21097,47747,21030)
(10788,5131,10746,5086)
\N
(15497,9698,15481,9678)
(10617,47195,10601,47117)
(42392,10583,42340,10550)
(10753,33520,10669,33509)
(5553,21580,5521,21527)
(36840,12336,36817,12320)
(49785,12554,49702,12553)
(17737,38349,17639,38277)
(48000,7823,47956,7814)
(5019,3184,4931,3160)
(30120,3524,30063,3492)
(37044,2016,37001,1942)
(23496,38566,23469,38528)
(17255,48957,17200,48903)
(27815,2138,27808,2090)
(40440,11129,40368,11105)
(35305,21772,35272,21717)
(41308,45065,41229,44973)
(14893,28807,14817,28789)
(30776,45824,30731,45772)
(742,40724,652,40672)
(5985,41133,5927,41097)
(9576,10226,9540,10218)
(21407,23207,21323,23160)
(44880,34228,44877,34169)
(29146,49694,29143,49682)
(28502,34886,28471,34832)
\N
(30662,5584,30604,5528)
(12612,26081,12552,26001)
(17166,49308,17098,49270)
(9586,14116,9488,14104)
(37323,47576,37264,47482)
(48009,49713,48004,49614)
(49308,23780,49297,23760)
(8667,32342,8592,32294)
(37826,48560,37822,48485)
\N
(24493,18653,24486,18616)
(17914,3850,17887,3775)
(34270,43873,34231,43826)
(7753,44715,7660,44651)
(44328,36364,44265,36350)
(10146,3030,10111,2975)
(35273,40106,35269,40062)
\N
(38566,43846,38547,43760)
(12400,41394,12377,41378)
(45196,38286,45153,38250)
(48511,14972,48428,14883)
(25939,36328,25886,36277)
(38997,11007,38979,10917)
(30342,518,30244,453)
(6876,7468,6867,7454)
(17566,27575,17566,27480)
(18869,28538,18858,28475)
(16825,33309,16726,33255)
(14585,26111,14490,26035)
(28743,49392,28664,49349)
(26652,23359,26618,23297)
\N
(40129,33653,40102,33584)
(41074,26393,41038,26389)
(3869,33564,3869,33536)
(28455,14205,28364,14163)
(13866,45603,13770,45543)
(21666,30586,21578,30544)
(29978,11931,29893,11868)
(1594,1043,1517,971)
(948,1201,907,1156)
(27547,13692,27545,13677)
(13661,38184,13566,38154)
(2389,40026,2317,39938)
(35481,46379,35481,46320)
\N
(26917,45698,26864,45689)
(23933,41617,23909,41539)
(8912,8471,8862,8401)
(9625,4747,9558,4692)
(34743,35056,34721,34969)
(39544,21762,39475,21717)
\N
(11741,26330,11656,26293)
(39015,1315,38966,1285)
(13418,44237,13326,44202)
(2107,17672,2093,17616)
(42448,28844,42370,28764)
(49843,5175,49808,5145)
(6536,23000,6467,22958)
(11114,5822,11027,5739)
(48457,11074,48384,11024)
(12343,23110,12310,23074)
(17300,24847,17276,24825)
(8823,8253,8793,8238)
(3449,171,3354,108)
\N
(21650,23955,21605,23883)
(13260,3234,13193,3214)
(25361,10896,25305,10806)
(25051,25042,25011,25001)
(25044,25088,25015,25005)
\N
(25007,25061,25002,25013)
(25066,25105,25003,25007)
(25028,25012,25015,25011)
(25031,25057,25006,25018)
(25015,25042,25004,25012)
(25091,25049,25019,25019)
(25023,25011,25000,25004)
\N
(25053,25104,25010,25012)
(25058,25001,25018,25000)
(25059,25051,25008,25016)
(25043,25069,25007,25004)
(25006,25101,25002,25002)
(25095,25012,25014,25007)
(25054,25052,25019,25013)
(25108,25077,25009,25018)
(25007,25023,25003,25002)
\N
(25076,25098,25002,25016)
(25030,25077,25012,25006)
--
-- first, define the datatype. Turn off echoing so that expected file
-- does not depend on contents of seg.sql.
--
\set ECHO none
create table boxtmp (b box);
\copy boxtmp from 'data/test_box.data'
select count(*) from boxtmp where b && '(1000,1000,0,0)'::box;
count
-------
2
(1 row)
create index bix on boxtmp using rtree (b);
select count(*) from boxtmp where b && '(1000,1000,0,0)'::box;
count
-------
2
(1 row)
drop index bix;
create index bix on boxtmp using gist (b gist_box_ops);
select count(*) from boxtmp where b && '(1000,1000,0,0)'::box;
count
-------
2
(1 row)
create table polytmp (p polygon);
\copy polytmp from 'data/test_box.data'
create index pix on polytmp using rtree (p);
select count(*) from polytmp where p && '(1000,1000),(0,0)'::polygon;
count
-------
2
(1 row)
drop index pix;
create index pix on polytmp using gist (p gist_poly_ops) with(islossy);
select count(*) from polytmp where p && '(1000,1000),(0,0)'::polygon;
count
-------
2
(1 row)
/*-------------------------------------------------------------------------
*
* rtree_gist.c
* pg_amproc entries for GiSTs over 2-D boxes.
* This gives R-tree behavior, with Guttman's poly-time split algorithm.
*
*
*
* IDENTIFICATION
* $Header: /cvsroot/pgsql/contrib/rtree_gist/Attic/rtree_gist.c,v 1.1 2001/05/31 18:27:18 tgl Exp $
*
*-------------------------------------------------------------------------
*/
#include "postgres.h"
#include "access/gist.h"
#include "access/itup.h"
#include "access/rtree.h"
#include "utils/palloc.h"
#include "utils/geo_decls.h"
#include "utils/elog.h"
typedef Datum (*RDF)(PG_FUNCTION_ARGS);
typedef Datum (*BINARY_UNION)(Datum, Datum, int*);
typedef float (*SIZE_BOX)(Datum);
/*
** Workaround for index_formtuple
*/
typedef struct polykey {
int32 size; /* size in varlena terms */
BOX key;
} POLYKEY;
/*
** box ops
*/
PG_FUNCTION_INFO_V1(gbox_compress);
PG_FUNCTION_INFO_V1(gbox_union);
PG_FUNCTION_INFO_V1(gbox_picksplit);
PG_FUNCTION_INFO_V1(gbox_consistent);
PG_FUNCTION_INFO_V1(gbox_penalty);
PG_FUNCTION_INFO_V1(gbox_same);
GISTENTRY * gbox_compress(PG_FUNCTION_ARGS);
BOX *gbox_union(PG_FUNCTION_ARGS);
GIST_SPLITVEC * gbox_picksplit(PG_FUNCTION_ARGS);
bool gbox_consistent(PG_FUNCTION_ARGS);
float * gbox_penalty(PG_FUNCTION_ARGS);
bool * gbox_same(PG_FUNCTION_ARGS);
static Datum gbox_binary_union(Datum r1, Datum r2, int *sizep);
static bool gbox_leaf_consistent(BOX *key, BOX *query, StrategyNumber strategy);
static float size_box( Datum box );
/*
** Polygon ops
*/
PG_FUNCTION_INFO_V1(gpoly_compress);
PG_FUNCTION_INFO_V1(gpoly_union);
PG_FUNCTION_INFO_V1(gpoly_picksplit);
PG_FUNCTION_INFO_V1(gpoly_consistent);
PG_FUNCTION_INFO_V1(gpoly_penalty);
PG_FUNCTION_INFO_V1(gpoly_same);
GISTENTRY * gpoly_compress(PG_FUNCTION_ARGS);
POLYKEY *gpoly_union(PG_FUNCTION_ARGS);
GIST_SPLITVEC * gpoly_picksplit(PG_FUNCTION_ARGS);
bool gpoly_consistent(PG_FUNCTION_ARGS);
float * gpoly_penalty(PG_FUNCTION_ARGS);
bool * gpoly_same(PG_FUNCTION_ARGS);
static Datum gpoly_binary_union(Datum r1, Datum r2, int *sizep);
static float size_polykey( Datum pk );
PG_FUNCTION_INFO_V1(gpoly_inter);
Datum gpoly_inter(PG_FUNCTION_ARGS);
/*
** Common rtree-function (for all ops)
*/
static Datum rtree_union(bytea *entryvec, int *sizep, BINARY_UNION bu);
static float * rtree_penalty(GISTENTRY *origentry, GISTENTRY *newentry,
float *result, BINARY_UNION bu, SIZE_BOX sb);
static GIST_SPLITVEC * rtree_picksplit(bytea *entryvec, GIST_SPLITVEC *v,
int keylen, BINARY_UNION bu, RDF interop, SIZE_BOX sb);
static bool rtree_internal_consistent(BOX *key, BOX *query, StrategyNumber strategy);
PG_FUNCTION_INFO_V1(rtree_decompress);
GISTENTRY * rtree_decompress(PG_FUNCTION_ARGS);
/**************************************************
* Box ops
**************************************************/
/*
** The GiST Consistent method for boxes
** Should return false if for all data items x below entry,
** the predicate x op query == FALSE, where op is the oper
** corresponding to strategy in the pg_amop table.
*/
bool
gbox_consistent(PG_FUNCTION_ARGS)
{
GISTENTRY *entry = (GISTENTRY*) PG_GETARG_POINTER(0);
BOX *query = (BOX*) PG_GETARG_POINTER(1);
StrategyNumber strategy = (StrategyNumber) PG_GETARG_UINT16(2);
/*
** if entry is not leaf, use gbox_internal_consistent,
** else use gbox_leaf_consistent
*/
if ( ! (DatumGetPointer(entry->key) != NULL && query) )
return FALSE;
if (GIST_LEAF(entry))
PG_RETURN_BOOL(gbox_leaf_consistent((BOX *) DatumGetPointer(entry->key), query, strategy));
else
PG_RETURN_BOOL(rtree_internal_consistent((BOX *) DatumGetPointer(entry->key), query, strategy));
}
/*
** The GiST Union method for boxes
** returns the minimal bounding box that encloses all the entries in entryvec
*/
BOX *
gbox_union(PG_FUNCTION_ARGS)
{
return (BOX*)
DatumGetPointer(rtree_union(
(bytea*) PG_GETARG_POINTER(0),
(int*) PG_GETARG_POINTER(1),
gbox_binary_union
));
}
/*
** GiST Compress methods for boxes
** do not do anything.
*/
GISTENTRY *
gbox_compress(PG_FUNCTION_ARGS)
{
return((GISTENTRY*)PG_GETARG_POINTER(0));
}
/*
** The GiST Penalty method for boxes
** As in the R-tree paper, we use change in area as our penalty metric
*/
float *
gbox_penalty(PG_FUNCTION_ARGS)
{
return rtree_penalty(
(GISTENTRY*) PG_GETARG_POINTER(0),
(GISTENTRY*) PG_GETARG_POINTER(1),
(float*) PG_GETARG_POINTER(2),
gbox_binary_union,
size_box
);
}
/*
** The GiST PickSplit method for boxes
** We use Guttman's poly time split algorithm
*/
GIST_SPLITVEC *
gbox_picksplit(PG_FUNCTION_ARGS)
{
return rtree_picksplit(
(bytea*)PG_GETARG_POINTER(0),
(GIST_SPLITVEC*)PG_GETARG_POINTER(1),
sizeof(BOX),
gbox_binary_union,
rt_box_inter,
size_box
);
}
/*
** Equality method
*/
bool *
gbox_same(PG_FUNCTION_ARGS)
{
BOX *b1 = (BOX*) PG_GETARG_POINTER(0);
BOX *b2 = (BOX*) PG_GETARG_POINTER(1);
bool *result = (bool*) PG_GETARG_POINTER(2);
if ( b1 && b2 )
*result = DatumGetBool( DirectFunctionCall2( box_same, PointerGetDatum(b1), PointerGetDatum(b2)) );
else
*result = ( b1==NULL && b2==NULL ) ? TRUE : FALSE;
return(result);
}
/*
** SUPPORT ROUTINES for boxes
*/
static bool
gbox_leaf_consistent(BOX *key,
BOX *query,
StrategyNumber strategy)
{
bool retval;
switch(strategy) {
case RTLeftStrategyNumber:
retval = DatumGetBool( DirectFunctionCall2( box_left, PointerGetDatum(key), PointerGetDatum(query) ) );
break;
case RTOverLeftStrategyNumber:
retval = DatumGetBool( DirectFunctionCall2( box_overleft, PointerGetDatum(key), PointerGetDatum(query) ) );
break;
case RTOverlapStrategyNumber:
retval = DatumGetBool( DirectFunctionCall2( box_overlap, PointerGetDatum(key), PointerGetDatum(query) ) );
break;
case RTOverRightStrategyNumber:
retval = DatumGetBool( DirectFunctionCall2( box_overright, PointerGetDatum(key), PointerGetDatum(query) ) );
break;
case RTRightStrategyNumber:
retval = DatumGetBool( DirectFunctionCall2( box_right, PointerGetDatum(key), PointerGetDatum(query) ) );
break;
case RTSameStrategyNumber:
retval = DatumGetBool( DirectFunctionCall2( box_same, PointerGetDatum(key), PointerGetDatum(query) ) );
break;
case RTContainsStrategyNumber:
retval = DatumGetBool( DirectFunctionCall2( box_contain, PointerGetDatum(key), PointerGetDatum(query) ) );
break;
case RTContainedByStrategyNumber:
retval = DatumGetBool( DirectFunctionCall2( box_contained, PointerGetDatum(key), PointerGetDatum(query) ) );
break;
default:
retval = FALSE;
}
return(retval);
}
static Datum
gbox_binary_union(Datum r1, Datum r2, int *sizep)
{
BOX *retval;
if ( ! (DatumGetPointer(r1) != NULL && DatumGetPointer(r2) != NULL) ) {
if ( DatumGetPointer(r1) != NULL ) {
retval = (BOX*) palloc( sizeof(BOX) );
memcpy( retval, DatumGetPointer(r1), sizeof(BOX) );
*sizep = sizeof(BOX);
} else if ( DatumGetPointer(r2) != NULL ) {
retval = (BOX*) palloc( sizeof(BOX) );
memcpy( retval, DatumGetPointer(r2), sizeof(BOX) );
*sizep = sizeof(BOX);
} else {
*sizep = 0;
retval = NULL;
}
} else {
retval = (BOX*) DatumGetPointer(
DirectFunctionCall2(rt_box_union, r1, r2));
*sizep = sizeof(BOX);
}
return PointerGetDatum(retval);
}
static float
size_box( Datum box ) {
if ( DatumGetPointer(box) != NULL ) {
float size;
DirectFunctionCall2( rt_box_size,
box, PointerGetDatum( &size ) );
return size;
} else
return 0.0;
}
/**************************************************
* Polygon ops
**************************************************/
GISTENTRY *
gpoly_compress(PG_FUNCTION_ARGS)
{
GISTENTRY *entry=(GISTENTRY*)PG_GETARG_POINTER(0);
GISTENTRY *retval;
if ( entry->leafkey) {
retval = palloc(sizeof(GISTENTRY));
if ( DatumGetPointer(entry->key) != NULL ) {
POLYGON *in;
POLYKEY *r;
in = (POLYGON *) PG_DETOAST_DATUM(entry->key);
r = (POLYKEY *) palloc( sizeof(POLYKEY) );
r->size = sizeof(POLYKEY);
memcpy( (void*)&(r->key), (void*)&(in->boundbox), sizeof(BOX) );
if ( in != (POLYGON *) DatumGetPointer(entry->key) )
pfree( in );
gistentryinit(*retval, PointerGetDatum(r),
entry->rel, entry->page,
entry->offset, sizeof(POLYKEY), FALSE);
} else {
gistentryinit(*retval, (Datum) 0,
entry->rel, entry->page,
entry->offset, 0,FALSE);
}
} else {
retval = entry;
}
return( retval );
}
bool
gpoly_consistent(PG_FUNCTION_ARGS)
{
GISTENTRY *entry = (GISTENTRY*) PG_GETARG_POINTER(0);
POLYGON *query = (POLYGON*)PG_DETOAST_DATUM( PG_GETARG_POINTER(1) );
StrategyNumber strategy = (StrategyNumber) PG_GETARG_UINT16(2);
bool result;
/*
** if entry is not leaf, use gbox_internal_consistent,
** else use gbox_leaf_consistent
*/
if ( ! (DatumGetPointer(entry->key) != NULL && query) )
return FALSE;
result = rtree_internal_consistent((BOX*)&( ((POLYKEY *) DatumGetPointer(entry->key))->key ),
&(query->boundbox), strategy);
PG_FREE_IF_COPY(query,1);
PG_RETURN_BOOL( result );
}
POLYKEY *
gpoly_union(PG_FUNCTION_ARGS)
{
return (POLYKEY*)
DatumGetPointer(rtree_union(
(bytea*) PG_GETARG_POINTER(0),
(int*) PG_GETARG_POINTER(1),
gpoly_binary_union
));
}
float *
gpoly_penalty(PG_FUNCTION_ARGS)
{
return rtree_penalty(
(GISTENTRY*) PG_GETARG_POINTER(0),
(GISTENTRY*) PG_GETARG_POINTER(1),
(float*) PG_GETARG_POINTER(2),
gpoly_binary_union,
size_polykey
);
}
GIST_SPLITVEC *
gpoly_picksplit(PG_FUNCTION_ARGS)
{
return rtree_picksplit(
(bytea*)PG_GETARG_POINTER(0),
(GIST_SPLITVEC*)PG_GETARG_POINTER(1),
sizeof(POLYKEY),
gpoly_binary_union,
gpoly_inter,
size_polykey
);
}
bool *
gpoly_same(PG_FUNCTION_ARGS)
{
POLYKEY *b1 = (POLYKEY*) PG_GETARG_POINTER(0);
POLYKEY *b2 = (POLYKEY*) PG_GETARG_POINTER(1);
bool *result = (bool*) PG_GETARG_POINTER(2);
if ( b1 && b2 )
*result = DatumGetBool( DirectFunctionCall2( box_same,
PointerGetDatum(&(b1->key)),
PointerGetDatum(&(b2->key))) );
else
*result = ( b1==NULL && b2==NULL ) ? TRUE : FALSE;
return(result);
}
/*
** SUPPORT ROUTINES for polygons
*/
Datum
gpoly_inter(PG_FUNCTION_ARGS)
{
POLYKEY *b1 = (POLYKEY*) PG_GETARG_POINTER(0);
POLYKEY *b2 = (POLYKEY*) PG_GETARG_POINTER(1);
Datum interd;
interd = DirectFunctionCall2(rt_box_inter,
PointerGetDatum( &(b1->key) ),
PointerGetDatum( &(b2->key) ));
if (DatumGetPointer(interd) != NULL) {
POLYKEY *tmp = (POLYKEY*) palloc( sizeof(POLYKEY) );
tmp->size = sizeof(POLYKEY);
memcpy( &(tmp->key), DatumGetPointer(interd), sizeof(BOX) );
pfree( DatumGetPointer(interd) );
PG_RETURN_POINTER( tmp );
} else
PG_RETURN_POINTER( NULL );
}
static Datum
gpoly_binary_union(Datum r1, Datum r2, int *sizep)
{
POLYKEY *retval;
if ( ! (DatumGetPointer(r1) != NULL && DatumGetPointer(r2) != NULL) ) {
if ( DatumGetPointer(r1) != NULL ) {
retval = (POLYKEY*)palloc( sizeof(POLYKEY) );
memcpy( (void*)retval, DatumGetPointer(r1), sizeof(POLYKEY) );
*sizep = sizeof(POLYKEY);
} else if ( DatumGetPointer(r2) != NULL ) {
retval = (POLYKEY*)palloc( sizeof(POLYKEY) );
memcpy( (void*)retval, DatumGetPointer(r2), sizeof(POLYKEY) );
*sizep = sizeof(POLYKEY);
} else {
*sizep = 0;
retval = NULL;
}
} else {
BOX *key = (BOX*)DatumGetPointer(
DirectFunctionCall2(
rt_box_union,
PointerGetDatum( &(((POLYKEY*) DatumGetPointer(r1))->key) ),
PointerGetDatum( &(((POLYKEY*) DatumGetPointer(r2))->key) )) );
retval = (POLYKEY*)palloc( sizeof(POLYKEY) );
memcpy( &(retval->key), key, sizeof(BOX) );
pfree( key );
*sizep = retval->size = sizeof(POLYKEY);
}
return PointerGetDatum(retval);
}
static float
size_polykey( Datum pk ) {
if ( DatumGetPointer(pk) != NULL ) {
float size;
DirectFunctionCall2( rt_box_size,
PointerGetDatum( &(((POLYKEY*) DatumGetPointer(pk))->key) ),
PointerGetDatum( &size ) );
return size;
} else
return 0.0;
}
/*
** Common rtree-function (for all ops)
*/
static Datum
rtree_union(bytea *entryvec, int *sizep, BINARY_UNION bu)
{
int numranges, i;
Datum out,
tmp;
numranges = (VARSIZE(entryvec) - VARHDRSZ)/sizeof(GISTENTRY);
tmp = ((GISTENTRY *) VARDATA(entryvec))[0].key;
out = (Datum) 0;
for (i = 1; i < numranges; i++) {
out = (*bu)(tmp,
((GISTENTRY *) VARDATA(entryvec))[i].key,
sizep);
if (i > 1 && DatumGetPointer(tmp) != NULL)
pfree(DatumGetPointer(tmp));
tmp = out;
}
return(out);
}
static float *
rtree_penalty(GISTENTRY *origentry, GISTENTRY *newentry, float *result, BINARY_UNION bu, SIZE_BOX sb)
{
Datum ud;
float tmp1;
int sizep;
ud = (*bu)( origentry->key, newentry->key, &sizep );
tmp1 = (*sb)( ud );
if (DatumGetPointer(ud) != NULL) pfree(DatumGetPointer(ud));
*result = tmp1 - (*sb)( origentry->key );
return(result);
}
/*
** The GiST PickSplit method
** We use Guttman's poly time split algorithm
*/
static GIST_SPLITVEC *
rtree_picksplit(bytea *entryvec, GIST_SPLITVEC *v, int keylen, BINARY_UNION bu, RDF interop, SIZE_BOX sb)
{
OffsetNumber i, j;
Datum datum_alpha, datum_beta;
Datum datum_l, datum_r;
Datum union_d, union_dl, union_dr;
Datum inter_d;
bool firsttime;
float size_alpha, size_beta, size_union, size_inter;
float size_waste, waste;
float size_l, size_r;
int nbytes;
int sizep;
OffsetNumber seed_1 = 0, seed_2 = 0;
OffsetNumber *left, *right;
OffsetNumber maxoff;
maxoff = ((VARSIZE(entryvec) - VARHDRSZ)/sizeof(GISTENTRY)) - 2;
nbytes = (maxoff + 2) * sizeof(OffsetNumber);
v->spl_left = (OffsetNumber *) palloc(nbytes);
v->spl_right = (OffsetNumber *) palloc(nbytes);
firsttime = true;
waste = 0.0;
for (i = FirstOffsetNumber; i < maxoff; i = OffsetNumberNext(i)) {
datum_alpha = ((GISTENTRY *) VARDATA(entryvec))[i].key;
for (j = OffsetNumberNext(i); j <= maxoff; j = OffsetNumberNext(j)) {
datum_beta = ((GISTENTRY *) VARDATA(entryvec))[j].key;
/* compute the wasted space by unioning these guys */
/* size_waste = size_union - size_inter; */
union_d = (*bu)( datum_alpha, datum_beta, &sizep );
if ( DatumGetPointer(union_d) != NULL ) {
size_union = (*sb)(union_d);
pfree(DatumGetPointer(union_d));
} else
size_union = 0.0;
if ( DatumGetPointer(datum_alpha) != NULL &&
DatumGetPointer(datum_beta) != NULL ) {
inter_d = DirectFunctionCall2(interop,
datum_alpha,
datum_beta);
if ( DatumGetPointer(inter_d) != NULL ) {
size_inter = (*sb)(inter_d);
pfree(DatumGetPointer(inter_d));
} else
size_inter = 0.0;
} else
size_inter = 0.0;
size_waste = size_union - size_inter;
/*
* are these a more promising split that what we've
* already seen?
*/
if (size_waste > waste || firsttime) {
waste = size_waste;
seed_1 = i;
seed_2 = j;
firsttime = false;
}
}
}
left = v->spl_left;
v->spl_nleft = 0;
right = v->spl_right;
v->spl_nright = 0;
if ( DatumGetPointer(((GISTENTRY *) VARDATA(entryvec))[seed_1].key) != NULL )
{
datum_l = PointerGetDatum(palloc( keylen ));
memcpy(DatumGetPointer(datum_l),
DatumGetPointer(((GISTENTRY *) VARDATA(entryvec))[seed_1].key),
keylen);
} else
datum_l = (Datum) 0;
size_l = (*sb)( datum_l );
if ( DatumGetPointer(((GISTENTRY *) VARDATA(entryvec))[seed_2].key) != NULL )
{
datum_r = PointerGetDatum(palloc( keylen ));
memcpy(DatumGetPointer(datum_r),
DatumGetPointer(((GISTENTRY *) VARDATA(entryvec))[seed_2].key),
keylen);
} else
datum_r = (Datum) 0;
size_r = (*sb)( datum_r );
/*
* Now split up the regions between the two seeds. An important
* property of this split algorithm is that the split vector v
* has the indices of items to be split in order in its left and
* right vectors. We exploit this property by doing a merge in
* the code that actually splits the page.
*
* For efficiency, we also place the new index tuple in this loop.
* This is handled at the very end, when we have placed all the
* existing tuples and i == maxoff + 1.
*/
maxoff = OffsetNumberNext(maxoff);
for (i = FirstOffsetNumber; i <= maxoff; i = OffsetNumberNext(i)) {
/*
* If we've already decided where to place this item, just
* put it on the right list. Otherwise, we need to figure
* out which page needs the least enlargement in order to
* store the item.
*/
if (i == seed_1) {
*left++ = i;
v->spl_nleft++;
continue;
} else if (i == seed_2) {
*right++ = i;
v->spl_nright++;
continue;
}
/* okay, which page needs least enlargement? */
datum_alpha = ((GISTENTRY *) VARDATA(entryvec))[i].key;
union_dl = (*bu)( datum_l, datum_alpha, &sizep );
union_dr = (*bu)( datum_r, datum_alpha, &sizep );
size_alpha = (*sb)( union_dl );
size_beta = (*sb)( union_dr );
/* pick which page to add it to */
if (size_alpha - size_l < size_beta - size_r) {
pfree(DatumGetPointer(datum_l));
pfree(DatumGetPointer(union_dr));
datum_l = union_dl;
size_l = size_alpha;
*left++ = i;
v->spl_nleft++;
} else {
pfree(DatumGetPointer(datum_r));
pfree(DatumGetPointer(union_dl));
datum_r = union_dr;
size_r = size_alpha;
*right++ = i;
v->spl_nright++;
}
}
*left = *right = FirstOffsetNumber; /* sentinel value, see dosplit() */
v->spl_ldatum = datum_l;
v->spl_rdatum = datum_r;
return( v );
}
static bool
rtree_internal_consistent(BOX *key,
BOX *query,
StrategyNumber strategy)
{
bool retval;
switch(strategy) {
case RTLeftStrategyNumber:
case RTOverLeftStrategyNumber:
retval = DatumGetBool( DirectFunctionCall2( box_overleft, PointerGetDatum(key), PointerGetDatum(query) ) );
break;
case RTOverlapStrategyNumber:
retval = DatumGetBool( DirectFunctionCall2( box_overlap, PointerGetDatum(key), PointerGetDatum(query) ) );
break;
case RTOverRightStrategyNumber:
case RTRightStrategyNumber:
retval = DatumGetBool( DirectFunctionCall2( box_right, PointerGetDatum(key), PointerGetDatum(query) ) );
break;
case RTSameStrategyNumber:
case RTContainsStrategyNumber:
retval = DatumGetBool( DirectFunctionCall2( box_contain, PointerGetDatum(key), PointerGetDatum(query) ) );
break;
case RTContainedByStrategyNumber:
retval = DatumGetBool( DirectFunctionCall2( box_overlap, PointerGetDatum(key), PointerGetDatum(query) ) );
break;
default:
retval = FALSE;
}
return(retval);
}
/*
** GiST DeCompress methods
** do not do anything.
*/
GISTENTRY *
rtree_decompress(PG_FUNCTION_ARGS)
{
return((GISTENTRY*)PG_GETARG_POINTER(0));
}
begin transaction;
--
--
--
-- BOX ops
--
--
--
-- define the GiST support methods
create function gbox_consistent(opaque,box,int4) returns bool as 'MODULE_PATHNAME' language 'C';
create function gbox_compress(opaque) returns opaque as 'MODULE_PATHNAME' language 'C';
create function rtree_decompress(opaque) returns opaque as 'MODULE_PATHNAME' language 'C';
create function gbox_penalty(opaque,opaque,opaque) returns opaque as 'MODULE_PATHNAME' language 'C';
create function gbox_picksplit(opaque, opaque) returns opaque as 'MODULE_PATHNAME' language 'C';
create function gbox_union(bytea, opaque) returns box as 'MODULE_PATHNAME' language 'C';
create function gbox_same(box, box, opaque) returns opaque as 'MODULE_PATHNAME' language 'C';
-- add a new opclass
--INSERT INTO pg_opclass (opcname, opcdeftype)
-- SELECT 'gist_box_ops', oid
-- FROM pg_type
-- WHERE typname = 'box';
INSERT INTO pg_opclass (opcname, opcdeftype) values ( 'gist_box_ops', 0 );
--SELECT oid, opcname FROM pg_opclass WHERE opcname = 'gist_box_ops';
-- get the comparators for boxes and store them in a tmp table
SELECT o.oid AS opoid, o.oprname
INTO TABLE rt_ops_tmp
FROM pg_operator o, pg_type t
WHERE o.oprleft = t.oid
and t.typname = 'box';
-- using the tmp table, generate the amop entries
-- box_left
INSERT INTO pg_amop (amopid, amopclaid, amopopr, amopstrategy)
SELECT am.oid, opcl.oid, c.opoid, 1
FROM pg_am am, pg_opclass opcl, rt_ops_tmp c
WHERE amname = 'gist' and opcname = 'gist_box_ops'
and c.oprname = '<<';
-- box_overleft
INSERT INTO pg_amop (amopid, amopclaid, amopopr, amopstrategy)
SELECT am.oid, opcl.oid, c.opoid, 2
FROM pg_am am, pg_opclass opcl, rt_ops_tmp c
WHERE amname = 'gist' and opcname = 'gist_box_ops'
and c.oprname = '&<';
-- box_overlap
INSERT INTO pg_amop (amopid, amopclaid, amopopr, amopstrategy)
SELECT am.oid, opcl.oid, c.opoid, 3
FROM pg_am am, pg_opclass opcl, rt_ops_tmp c
WHERE amname = 'gist' and opcname = 'gist_box_ops'
and c.oprname = '&&';
-- box_overright
INSERT INTO pg_amop (amopid, amopclaid, amopopr, amopstrategy)
SELECT am.oid, opcl.oid, c.opoid, 4
FROM pg_am am, pg_opclass opcl, rt_ops_tmp c
WHERE amname = 'gist' and opcname = 'gist_box_ops'
and c.oprname = '&>';
-- box_right
INSERT INTO pg_amop (amopid, amopclaid, amopopr, amopstrategy)
SELECT am.oid, opcl.oid, c.opoid, 5
FROM pg_am am, pg_opclass opcl, rt_ops_tmp c
WHERE amname = 'gist' and opcname = 'gist_box_ops'
and c.oprname = '>>';
-- box_same
INSERT INTO pg_amop (amopid, amopclaid, amopopr, amopstrategy)
SELECT am.oid, opcl.oid, c.opoid, 6
FROM pg_am am, pg_opclass opcl, rt_ops_tmp c
WHERE amname = 'gist' and opcname = 'gist_box_ops'
and c.oprname = '~=';
-- box_contains
INSERT INTO pg_amop (amopid, amopclaid, amopopr, amopstrategy)
SELECT am.oid, opcl.oid, c.opoid, 7
FROM pg_am am, pg_opclass opcl, rt_ops_tmp c
WHERE amname = 'gist' and opcname = 'gist_box_ops'
and c.oprname = '~';
-- box_contained
INSERT INTO pg_amop (amopid, amopclaid, amopopr, amopstrategy)
SELECT am.oid, opcl.oid, c.opoid, 8
FROM pg_am am, pg_opclass opcl, rt_ops_tmp c
WHERE amname = 'gist' and opcname = 'gist_box_ops'
and c.oprname = '@';
DROP table rt_ops_tmp;
-- add the entries to amproc for the support methods
-- note the amprocnum numbers associated with each are specific!
INSERT INTO pg_amproc (amid, amopclaid, amproc, amprocnum)
SELECT am.oid, opcl.oid, pro.oid, 1
FROM pg_am am, pg_opclass opcl, pg_proc pro
WHERE amname = 'gist' and opcname = 'gist_box_ops'
and proname = 'gbox_consistent';
INSERT INTO pg_amproc (amid, amopclaid, amproc, amprocnum)
SELECT am.oid, opcl.oid, pro.oid, 2
FROM pg_am am, pg_opclass opcl, pg_proc pro
WHERE amname = 'gist' and opcname = 'gist_box_ops'
and proname = 'gbox_union';
INSERT INTO pg_amproc (amid, amopclaid, amproc, amprocnum)
SELECT am.oid, opcl.oid, pro.oid, 3
FROM pg_am am, pg_opclass opcl, pg_proc pro
WHERE amname = 'gist' and opcname = 'gist_box_ops'
and proname = 'gbox_compress';
INSERT INTO pg_amproc (amid, amopclaid, amproc, amprocnum)
SELECT am.oid, opcl.oid, pro.oid, 4
FROM pg_am am, pg_opclass opcl, pg_proc pro
WHERE amname = 'gist' and opcname = 'gist_box_ops'
and proname = 'rtree_decompress';
INSERT INTO pg_amproc (amid, amopclaid, amproc, amprocnum)
SELECT am.oid, opcl.oid, pro.oid, 5
FROM pg_am am, pg_opclass opcl, pg_proc pro
WHERE amname = 'gist' and opcname = 'gist_box_ops'
and proname = 'gbox_penalty';
INSERT INTO pg_amproc (amid, amopclaid, amproc, amprocnum)
SELECT am.oid, opcl.oid, pro.oid, 6
FROM pg_am am, pg_opclass opcl, pg_proc pro
WHERE amname = 'gist' and opcname = 'gist_box_ops'
and proname = 'gbox_picksplit';
INSERT INTO pg_amproc (amid, amopclaid, amproc, amprocnum)
SELECT am.oid, opcl.oid, pro.oid, 7
FROM pg_am am, pg_opclass opcl, pg_proc pro
WHERE amname = 'gist' and opcname = 'gist_box_ops'
and proname = 'gbox_same';
--
--
--
-- POLYGON ops
--
--
--
-- define the GiST support methods
create function gpoly_consistent(opaque,polygon,int4) returns bool as 'MODULE_PATHNAME' language 'C';
create function gpoly_compress(opaque) returns opaque as 'MODULE_PATHNAME' language 'C';
create function gpoly_penalty(opaque,opaque,opaque) returns opaque as 'MODULE_PATHNAME' language 'C';
create function gpoly_picksplit(opaque, opaque) returns opaque as 'MODULE_PATHNAME' language 'C';
create function gpoly_union(bytea, opaque) returns opaque as 'MODULE_PATHNAME' language 'C';
create function gpoly_same(opaque, opaque, opaque) returns opaque as 'MODULE_PATHNAME' language 'C';
-- add a new opclass
--INSERT INTO pg_opclass (opcname, opcdeftype)
-- SELECT 'gist_poly_ops', oid
-- FROM pg_type
-- WHERE typname = 'polygon';
INSERT INTO pg_opclass (opcname, opcdeftype) values ( 'gist_poly_ops', 0 );
--SELECT oid, opcname FROM pg_opclass WHERE opcname = 'gist_poly_ops';
-- get the comparators for polygons and store them in a tmp table
-- hack for 757 (poly_contain_pt) Teodor
SELECT o.oid AS opoid, o.oprname
INTO TABLE rt_ops_tmp
FROM pg_operator o, pg_type t
WHERE o.oprleft = t.oid and o.oid <> 757
and t.typname = 'polygon';
-- using the tmp table, generate the amop entries
-- poly_left
INSERT INTO pg_amop (amopid, amopclaid, amopopr, amopstrategy)
SELECT am.oid, opcl.oid, c.opoid, 1
FROM pg_am am, pg_opclass opcl, rt_ops_tmp c
WHERE amname = 'gist' and opcname = 'gist_poly_ops'
and c.oprname = '<<';
-- poly_overleft
INSERT INTO pg_amop (amopid, amopclaid, amopopr, amopstrategy)
SELECT am.oid, opcl.oid, c.opoid, 2
FROM pg_am am, pg_opclass opcl, rt_ops_tmp c
WHERE amname = 'gist' and opcname = 'gist_poly_ops'
and c.oprname = '&<';
-- poly_overlap
INSERT INTO pg_amop (amopid, amopclaid, amopopr, amopstrategy)
SELECT am.oid, opcl.oid, c.opoid, 3
FROM pg_am am, pg_opclass opcl, rt_ops_tmp c
WHERE amname = 'gist' and opcname = 'gist_poly_ops'
and c.oprname = '&&';
-- poly_overright
INSERT INTO pg_amop (amopid, amopclaid, amopopr, amopstrategy)
SELECT am.oid, opcl.oid, c.opoid, 4
FROM pg_am am, pg_opclass opcl, rt_ops_tmp c
WHERE amname = 'gist' and opcname = 'gist_poly_ops'
and c.oprname = '&>';
-- poly_right
INSERT INTO pg_amop (amopid, amopclaid, amopopr, amopstrategy)
SELECT am.oid, opcl.oid, c.opoid, 5
FROM pg_am am, pg_opclass opcl, rt_ops_tmp c
WHERE amname = 'gist' and opcname = 'gist_poly_ops'
and c.oprname = '>>';
-- poly_same
INSERT INTO pg_amop (amopid, amopclaid, amopopr, amopstrategy)
SELECT am.oid, opcl.oid, c.opoid, 6
FROM pg_am am, pg_opclass opcl, rt_ops_tmp c
WHERE amname = 'gist' and opcname = 'gist_poly_ops'
and c.oprname = '~=';
-- poly_contains
INSERT INTO pg_amop (amopid, amopclaid, amopopr, amopstrategy)
SELECT am.oid, opcl.oid, c.opoid, 7
FROM pg_am am, pg_opclass opcl, rt_ops_tmp c
WHERE amname = 'gist' and opcname = 'gist_poly_ops'
and c.oprname = '~';
-- poly_contained
INSERT INTO pg_amop (amopid, amopclaid, amopopr, amopstrategy)
SELECT am.oid, opcl.oid, c.opoid, 8
FROM pg_am am, pg_opclass opcl, rt_ops_tmp c
WHERE amname = 'gist' and opcname = 'gist_poly_ops'
and c.oprname = '@';
DROP table rt_ops_tmp;
-- add the entries to amproc for the support methods
-- note the amprocnum numbers associated with each are specific!
INSERT INTO pg_amproc (amid, amopclaid, amproc, amprocnum)
SELECT am.oid, opcl.oid, pro.oid, 1
FROM pg_am am, pg_opclass opcl, pg_proc pro
WHERE amname = 'gist' and opcname = 'gist_poly_ops'
and proname = 'gpoly_consistent';
INSERT INTO pg_amproc (amid, amopclaid, amproc, amprocnum)
SELECT am.oid, opcl.oid, pro.oid, 2
FROM pg_am am, pg_opclass opcl, pg_proc pro
WHERE amname = 'gist' and opcname = 'gist_poly_ops'
and proname = 'gpoly_union';
INSERT INTO pg_amproc (amid, amopclaid, amproc, amprocnum)
SELECT am.oid, opcl.oid, pro.oid, 3
FROM pg_am am, pg_opclass opcl, pg_proc pro
WHERE amname = 'gist' and opcname = 'gist_poly_ops'
and proname = 'gpoly_compress';
INSERT INTO pg_amproc (amid, amopclaid, amproc, amprocnum)
SELECT am.oid, opcl.oid, pro.oid, 4
FROM pg_am am, pg_opclass opcl, pg_proc pro
WHERE amname = 'gist' and opcname = 'gist_poly_ops'
and proname = 'rtree_decompress';
INSERT INTO pg_amproc (amid, amopclaid, amproc, amprocnum)
SELECT am.oid, opcl.oid, pro.oid, 5
FROM pg_am am, pg_opclass opcl, pg_proc pro
WHERE amname = 'gist' and opcname = 'gist_poly_ops'
and proname = 'gpoly_penalty';
INSERT INTO pg_amproc (amid, amopclaid, amproc, amprocnum)
SELECT am.oid, opcl.oid, pro.oid, 6
FROM pg_am am, pg_opclass opcl, pg_proc pro
WHERE amname = 'gist' and opcname = 'gist_poly_ops'
and proname = 'gpoly_picksplit';
INSERT INTO pg_amproc (amid, amopclaid, amproc, amprocnum)
SELECT am.oid, opcl.oid, pro.oid, 7
FROM pg_am am, pg_opclass opcl, pg_proc pro
WHERE amname = 'gist' and opcname = 'gist_poly_ops'
and proname = 'gpoly_same';
end transaction;
--
-- first, define the datatype. Turn off echoing so that expected file
-- does not depend on contents of seg.sql.
--
\set ECHO none
\i rtree_gist.sql
\set ECHO all
create table boxtmp (b box);
\copy boxtmp from 'data/test_box.data'
select count(*) from boxtmp where b && '(1000,1000,0,0)'::box;
create index bix on boxtmp using rtree (b);
select count(*) from boxtmp where b && '(1000,1000,0,0)'::box;
drop index bix;
create index bix on boxtmp using gist (b gist_box_ops);
select count(*) from boxtmp where b && '(1000,1000,0,0)'::box;
create table polytmp (p polygon);
\copy polytmp from 'data/test_box.data'
create index pix on polytmp using rtree (p);
select count(*) from polytmp where p && '(1000,1000),(0,0)'::polygon;
drop index pix;
create index pix on polytmp using gist (p gist_poly_ops) with(islossy);
select count(*) from polytmp where p && '(1000,1000),(0,0)'::polygon;
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment