add ft tst func
authorxxx <xxx@rock-chips.com>
Thu, 26 Jun 2014 10:19:48 +0000 (18:19 +0800)
committerxxx <xxx@rock-chips.com>
Thu, 26 Jun 2014 10:19:48 +0000 (18:19 +0800)
16 files changed:
arch/arm/boot/dts/rk3288-ft.dts [new file with mode: 0755]
arch/arm/boot/dts/rk3288.dtsi
arch/arm/configs/rockchip_ft_defconfig [new file with mode: 0644]
arch/arm/mach-rockchip/Kconfig
arch/arm/mach-rockchip/Makefile
arch/arm/mach-rockchip/rk_pm_tests/ft/Makefile [new file with mode: 0755]
arch/arm/mach-rockchip/rk_pm_tests/ft/cpu_test.h [new file with mode: 0755]
arch/arm/mach-rockchip/rk_pm_tests/ft/fft/Makefile [new file with mode: 0755]
arch/arm/mach-rockchip/rk_pm_tests/ft/fft/fft_fixed_point.c [new file with mode: 0755]
arch/arm/mach-rockchip/rk_pm_tests/ft/fft/fft_fixed_point.h [new file with mode: 0755]
arch/arm/mach-rockchip/rk_pm_tests/ft/fft/test.c [new file with mode: 0644]
arch/arm/mach-rockchip/rk_pm_tests/ft/ft_cpu_tst.S [new file with mode: 0755]
arch/arm/mach-rockchip/rk_pm_tests/ft/ft_cpu_tst1.S [new file with mode: 0755]
arch/arm/mach-rockchip/rk_pm_tests/ft/ft_test.c [new file with mode: 0644]
arch/arm/mach-rockchip/rk_pm_tests/ft/rk_ft_io.c [new file with mode: 0755]
arch/arm/mach-rockchip/rk_pm_tests/ft/test_mem.c [new file with mode: 0755]

diff --git a/arch/arm/boot/dts/rk3288-ft.dts b/arch/arm/boot/dts/rk3288-ft.dts
new file mode 100755 (executable)
index 0000000..2aa85ad
--- /dev/null
@@ -0,0 +1,54 @@
+/dts-v1/;
+
+#include "rk3288.dtsi"
+
+/ {
+       memory {
+               device_type = "memory";
+               reg = <0x00000000 0x40000000>;
+       };
+
+       chosen {
+               bootargs = "console=ttyFIQ0 rockchip_jtag";
+       };
+
+       fiq-debugger {
+               status = "okay";
+       };
+    
+        rockchip_ft_test {
+            // rockchip,arm_rate =<408000000 504000000 600000000 792000000>;     
+            // firs rate is setting in rockchip_clocks_init,so it is set 0xfffffffff
+            rockchip,arm_rate =<0xffffffff (1728000000*1) (1752000000*1)  (1800000000*1)>;   
+            rockchip,l1_tst_cnt=<(5*10*2-10) (5*10*2-10) (5*10*2-10) (80-10+0) >;
+            rockchip,l2_cpy_cnt= <5 5 4 (4+0)>;
+            rockchip,ftt_tst_cnt= <1 1 1 (0+0)>;
+            rockchip,pi_tst_cnt= <12 12 10 (8+0)>;
+            rockchip,stp_task_num= <300 250 250 (80+0)>;
+            rockchip,ft_end_cnt= <0x23>;   
+        };
+};
+
+ &rockchip_clocks_init {
+                      rockchip,clocks-init-rate =
+                       <&clk_core 1704000000>, <&clk_gpll 297000000>,
+                       /*<&clk_cpll 47000000>,*/       <&clk_npll 1250000000>,
+                       <&aclk_bus_src 300000000>,      <&aclk_bus 300000000>,
+                       <&hclk_bus 150000000>,  <&pclk_bus 75000000>,
+                       <&clk_crypto 150000000>,        <&aclk_peri 300000000>,
+                       <&hclk_peri 150000000>, <&pclk_peri 75000000>,
+                       <&clk_gpu 200000000>,   <&aclk_vio0 300000000>,
+                       <&aclk_vio1 300000000>, <&hclk_vio 75000000>,
+                       <&pclk_pd_alive 100000000>,     <&pclk_pd_pmu 100000000>,
+                       <&aclk_hevc 400000000>, <&hclk_hevc 200000000>,
+                       <&clk_hevc_cabac 300000000>, <&clk_hevc_core 300000000>,
+                       <&aclk_rga 300000000>, <&clk_rga 300000000>,
+                       <&clk_vepu 300000000>, <&clk_vdpu 300000000>,
+                       <&clk_edp 200000000>, <&clk_isp 200000000>,
+                       <&clk_isp_jpe 400000000>, <&clk_tsp 80000000>,
+                       <&clk_tspout 80000000>, <&clk_mac 125000000>;
+};
+
+
+
+
index 3bc9fde70be8b0cb02e7e301760e3c47d4aca456..907348441034e239112e310cf9fee0b64544c63b 100755 (executable)
                status = "disabled";
        };
 
-       clocks-init{
+       rockchip_clocks_init: clocks-init{
                compatible = "rockchip,clocks-init";
                rockchip,clocks-init-parent =
                        <&clk_core &clk_apll>,  <&aclk_bus_src &clk_gpll>,
diff --git a/arch/arm/configs/rockchip_ft_defconfig b/arch/arm/configs/rockchip_ft_defconfig
new file mode 100644 (file)
index 0000000..7931ecf
--- /dev/null
@@ -0,0 +1,76 @@
+# CONFIG_LOCALVERSION_AUTO is not set
+CONFIG_KERNEL_LZO=y
+CONFIG_LOG_BUF_SHIFT=12
+CONFIG_CC_OPTIMIZE_FOR_SIZE=y
+# CONFIG_UID16 is not set
+# CONFIG_BASE_FULL is not set
+# CONFIG_FUTEX is not set
+# CONFIG_EPOLL is not set
+# CONFIG_SIGNALFD is not set
+# CONFIG_TIMERFD is not set
+# CONFIG_EVENTFD is not set
+# CONFIG_SHMEM is not set
+# CONFIG_AIO is not set
+CONFIG_EMBEDDED=y
+# CONFIG_VM_EVENT_COUNTERS is not set
+# CONFIG_BLOCK is not set
+CONFIG_ARCH_ROCKCHIP=y
+# CONFIG_RK_LAST_LOG is not set
+# CONFIG_RK_CONSOLE_THREAD is not set
+# CONFIG_DVFS is not set
+# CONFIG_RK_VCODEC is not set
+CONFIG_RK_FT_TEST=y
+# CONFIG_ARM_THUMB is not set
+# CONFIG_SWP_EMULATE is not set
+# CONFIG_KUSER_HELPERS is not set
+CONFIG_FIQ_DEBUGGER=y
+CONFIG_FIQ_DEBUGGER_NO_SLEEP=y
+CONFIG_FIQ_DEBUGGER_CONSOLE=y
+CONFIG_FIQ_DEBUGGER_CONSOLE_DEFAULT_ENABLE=y
+CONFIG_SMP=y
+CONFIG_PREEMPT=y
+CONFIG_AEABI=y
+# CONFIG_OABI_COMPAT is not set
+CONFIG_HIGHMEM=y
+# CONFIG_COMPACTION is not set
+CONFIG_DEFAULT_MMAP_MIN_ADDR=32768
+# CONFIG_CROSS_MEMORY_ATTACH is not set
+# CONFIG_ATAGS is not set
+CONFIG_ZBOOT_ROM_TEXT=0x0
+CONFIG_ZBOOT_ROM_BSS=0x0
+CONFIG_ARM_APPENDED_DTB=y
+CONFIG_CMDLINE="console=ttyFIQ0"
+CONFIG_CPU_IDLE=y
+CONFIG_VFP=y
+CONFIG_NEON=y
+# CONFIG_BINFMT_ELF is not set
+# CONFIG_BINFMT_SCRIPT is not set
+# CONFIG_COREDUMP is not set
+# CONFIG_SUSPEND is not set
+# CONFIG_FW_LOADER is not set
+CONFIG_SRAM=y
+# CONFIG_5V_EN is not set
+# CONFIG_INPUT is not set
+# CONFIG_SERIO is not set
+# CONFIG_VT is not set
+# CONFIG_UNIX98_PTYS is not set
+# CONFIG_LEGACY_PTYS is not set
+# CONFIG_DEVMEM is not set
+# CONFIG_DEVKMEM is not set
+# CONFIG_HW_RANDOM is not set
+# CONFIG_HWMON is not set
+# CONFIG_IEP is not set
+# CONFIG_USB_SUPPORT is not set
+# CONFIG_IOMMU_SUPPORT is not set
+# CONFIG_FILE_LOCKING is not set
+# CONFIG_DNOTIFY is not set
+# CONFIG_INOTIFY_USER is not set
+# CONFIG_PROC_FS is not set
+# CONFIG_SYSFS is not set
+# CONFIG_MISC_FILESYSTEMS is not set
+CONFIG_MAGIC_SYSRQ=y
+# CONFIG_DEBUG_PREEMPT is not set
+# CONFIG_STACKTRACE is not set
+# CONFIG_RCU_CPU_STALL_VERBOSE is not set
+# CONFIG_FTRACE is not set
+# CONFIG_CRC32 is not set
index eecbf138a36e154176314b09038342505057d4de..827f48ccd8f6559888bd34969b4fc7e8787a5429 100755 (executable)
@@ -80,4 +80,9 @@ config RK_VCODEC
 
 config RK_PL330_DMA_TEST
        bool "pl330 DMA memcpy test"
+
+config RK_FT_TEST
+       bool "ft test"
+       select LZO_COMPRESS
+       select LZO_DECOMPRESS
 endif
index 65060cd8a9969e0f5be94fb9f2f1844d9e5f77e8..367a0aea6d76c6afe71934707c13f244bac5bf9a 100755 (executable)
@@ -17,3 +17,4 @@ obj-$(CONFIG_RK_PL330_DMA_TEST) += dma_memcpy_test.o
 obj-$(CONFIG_BLOCK_RKNAND) += rknandbase.o
 obj-$(CONFIG_DDR_TEST) += ddr_test.o
 obj-$(CONFIG_RK_PM_TESTS) += rk_pm_tests/
+obj-$(CONFIG_RK_FT_TEST) += rk_pm_tests/ft/
diff --git a/arch/arm/mach-rockchip/rk_pm_tests/ft/Makefile b/arch/arm/mach-rockchip/rk_pm_tests/ft/Makefile
new file mode 100755 (executable)
index 0000000..215bb0f
--- /dev/null
@@ -0,0 +1,5 @@
+obj-y += ft_cpu_tst.o
+obj-y += ft_cpu_tst1.o
+obj-y += ft_test.o
+obj-y += test_mem.o
+obj-y += fft/
diff --git a/arch/arm/mach-rockchip/rk_pm_tests/ft/cpu_test.h b/arch/arm/mach-rockchip/rk_pm_tests/ft/cpu_test.h
new file mode 100755 (executable)
index 0000000..edae7e6
--- /dev/null
@@ -0,0 +1,228 @@
+
+#include <linux/linkage.h>
+#include <asm/assembler.h>
+//#include <asm/memory.h>
+//#include <mach/io.h>
+
+.text
+
+
+       //_off 12 -20
+       .macro  test_cpus_pc_jump , _base,_reg, _off
+       
+       ldr \_reg,[pc]
+       str \_reg,[\_base]
+       mov \_reg,pc
+       str \_reg,[\_base,#4]
+       add \_reg,\_off //pc offset
+       str \_reg,[\_base,#8]// offset start
+       ldr   pc, [\_base,#8]
+       ldr \_reg,[pc]
+       ldr \_reg,[pc]
+       .endm
+
+
+       .macro  cpus_tst_get_opcode, _reg0
+       lsl \_reg0,\_reg0,#4
+       lsr \_reg0,\_reg0,#24
+       .endm
+
+
+       .macro  cpus_tst_code_lsl , _reg0
+       ldr \_reg0,[pc]
+       mov \_reg0,\_reg0
+       lsl \_reg0,\_reg0,#4
+       lsr \_reg0,\_reg0,#24
+       cmp \_reg0,#0x1a
+       bne l1_test_error
+       .endm
+
+       .macro  cpus_tst_code_lsr , _reg0
+       ldr \_reg0,[pc]
+       lsl \_reg0,\_reg0,#4
+       lsr \_reg0,\_reg0,#24
+       cmp \_reg0,#0x1a
+       bne l1_test_error
+       .endm
+
+
+       .macro  cpus_tst_code_cmp, _reg0
+       ldr \_reg0,[pc,#4]
+       lsl \_reg0,\_reg0,#4
+       lsr \_reg0,\_reg0,#24
+       cmp \_reg0,#0x35
+       bne l1_test_error
+       .endm
+
+
+
+       .macro  cpus_tst_code_sub, _reg0,_reg1
+       mov \_reg0,pc
+       sub \_reg0,#4
+       ldr \_reg1,[\_reg0]
+       lsl \_reg1,\_reg1,#4
+       lsr \_reg1,\_reg1,#24
+       cmp \_reg1,#0x24
+       bne l1_test_error
+       .endm
+
+
+       .macro  cpus_tst_code_ldr, _reg0,_reg1
+       mov \_reg0,pc
+       sub \_reg0,#0
+       ldr \_reg1,[\_reg0]
+       lsl \_reg1,\_reg1,#4
+       lsr \_reg1,\_reg1,#24
+       cmp \_reg1,#0x59
+       bne l1_test_error
+       .endm
+
+       .macro  cpus_tst_code_bne, _reg0,_reg1
+       ldr \_reg0,[pc,#12]
+
+       lsr \_reg1,\_reg0,#28
+       lsl \_reg0,\_reg0,#4
+       lsr \_reg0,\_reg0,#24
+       
+       cmp \_reg1,#0x1 //
+       bne l1_test_error
+       cmp \_reg0,#0xa0
+       bne l1_test_error
+       
+       .endm
+
+
+
+.macro test_cpus_l1_code_base
+       //r4 base
+       mov r5,#16
+       test_cpus_pc_jump r4,r3,r5
+       
+       //add r4,#8
+       mov r6,r4
+       mov r8,#20
+       test_cpus_pc_jump r6,r7,r8 
+
+       //add r4,#8
+
+       cpus_tst_code_lsl r5
+       
+       cpus_tst_code_lsr r7
+       
+       cpus_tst_code_cmp r9
+
+
+       cpus_tst_code_sub r11,r10,
+       
+       cpus_tst_code_ldr r12,r8
+       
+       cpus_tst_code_bne r6,r7
+
+.endm
+
+
+.macro test_cpus_l1_code_base_
+
+       
+       mov r4,r4
+       mov r6,r6
+       mov r7,r7
+       mov r8,r8
+       mov r4,r4
+       mov r6,r6
+       mov r7,r7
+       mov r8,r8
+       mov r4,r4
+       mov r6,r6
+
+       mov r4,r4
+       mov r6,r6
+       mov r7,r7
+       mov r8,r8
+       mov r4,r4
+       mov r6,r6
+       mov r7,r7
+       mov r8,r8
+       mov r4,r4
+       mov r6,r6
+
+
+
+       mov r7,r7
+       mov r8,r8
+       mov r4,r4
+       mov r6,r6
+       mov r7,r7
+       
+       
+       
+       
+
+               
+.endm
+
+
+.macro test_cpus_l1_loop_100
+
+.endm
+
+.macro test_cpus_l1_loop_500
+#if 1
+       test_cpus_l1_code_base  
+       test_cpus_l1_code_base  
+#else
+
+       test_cpus_l1_code_base_
+       test_cpus_l1_code_base_
+
+       test_cpus_l1_code_base_
+       test_cpus_l1_code_base_
+       test_cpus_l1_code_base_
+       
+       test_cpus_l1_code_base_
+
+#endif
+.endm
+
+
+.macro test_cpus_l1_loop_1_k
+       test_cpus_l1_loop_500  
+       test_cpus_l1_loop_500  
+.endm
+
+
+.macro test_cpus_l1_loop_4_k
+       test_cpus_l1_loop_1_k  
+       test_cpus_l1_loop_1_k  
+       test_cpus_l1_loop_1_k   
+       test_cpus_l1_loop_1_k   
+       test_cpus_l1_loop_500   
+.endm
+
+
+
+.macro test_cpus_l1_loop_10_k
+       test_cpus_l1_loop_4_k  
+       test_cpus_l1_loop_4_k  
+       test_cpus_l1_loop_4_k
+       test_cpus_l1_loop_4_k
+.endm
+
+
+.macro test_cpus_l1_loop_50_k
+       test_cpus_l1_loop_10_k  
+       test_cpus_l1_loop_10_k  
+       test_cpus_l1_loop_10_k
+       test_cpus_l1_loop_10_k
+       test_cpus_l1_loop_10_k
+.endm
+
+.macro test_cpus_l1_loop_200_k
+       test_cpus_l1_loop_50_k  
+       test_cpus_l1_loop_50_k  
+       test_cpus_l1_loop_50_k
+       test_cpus_l1_loop_50_k
+       
+.endm
+
+
diff --git a/arch/arm/mach-rockchip/rk_pm_tests/ft/fft/Makefile b/arch/arm/mach-rockchip/rk_pm_tests/ft/fft/Makefile
new file mode 100755 (executable)
index 0000000..48fc2f1
--- /dev/null
@@ -0,0 +1,2 @@
+obj-y += fft_fixed_point.o
+obj-y += test.o
diff --git a/arch/arm/mach-rockchip/rk_pm_tests/ft/fft/fft_fixed_point.c b/arch/arm/mach-rockchip/rk_pm_tests/ft/fft/fft_fixed_point.c
new file mode 100755 (executable)
index 0000000..9308226
--- /dev/null
@@ -0,0 +1,401 @@
+#include "fft_fixed_point.h"\r
+#if 0\r
+#include <stdlib.h>\r
+#include <string.h>\r
+#else\r
+\r
+#include <linux/string.h>\r
+#include <linux/resume-trace.h>\r
+#include <linux/workqueue.h>\r
+#include <linux/mutex.h>\r
+#include <linux/module.h>\r
+#include <linux/syscalls.h>\r
+#include <linux/init.h>\r
+#include <linux/interrupt.h>\r
+#include <linux/delay.h>\r
+#include <linux/cpu.h>\r
+#include <linux/slab.h>\r
+#include <linux/freezer.h>\r
+#include <linux/vmalloc.h>\r
+#define free(ptr) kfree((ptr))\r
+//typedef unsigned char uchar;\r
+\r
+#define malloc(size) kmalloc((size), GFP_ATOMIC)\r
+\r
+#define printf(fmt, arg...) \\r
+               printk(KERN_EMERG fmt, ##arg)\r
+\r
+#endif\r
+\r
+\r
+/***********************************************²éÕÒ±í*************************************************************/\r
+\r
+//È«¾Ö±äÁ¿,2µÄÕûÊý´ÎÃݱí\r
+int Pow2_table[20] = {1,2,4,8,16,32,64,128,256,512,1024,2048,4096,8192,16384,32768,65536,131072,262144,524288};\r
+\r
+//¶ÔÉϱí½øÐж¨µã»¯ºóµÄsinº¯Êý²éÕÒ±í£¬¼´½«ÉϱíÖÐÿ¸öÊý³ËÉÏ 2^8 = 256\r
+int sin_table_INT_128[]={\r
+        0,  13,  25,  38,  50,  62,  74,  86,\r
+       98, 109, 121, 132, 142, 152, 162, 172,\r
+       181, 190, 198, 206, 213, 220, 226, 231,\r
+       237, 241, 245, 248, 251, 253, 255, 256,\r
+       256, 256, 255, 253, 251, 248, 245, 241,\r
+       237, 231, 226, 220, 213, 206, 198, 190,\r
+       181, 172, 162, 152, 142, 132, 121, 109,\r
+        98,  86,  74,  62,  50,  38,  25,  13\r
+       \r
+};\r
+//FFTÖеãÊýΪ128µÄcosº¯Êý¶¨µã»¯ºóµÄ²éÕÒ±í£¬\r
+int cos_table_INT_128[]={\r
+       256,  256,  255,  253,  251,  248,  245,  241,\r
+       237,  231,  226,  220,  213,  206,  198,  190,\r
+       181,  172,  162,  152,  142,  132,  121,  109,\r
+        98,   86,   74,   62,   50,   38,   25,   13,\r
+         0,  -13,  -25,  -38,  -50,  -62,  -74,  -86,\r
+        -98, -109, -121, -132, -142, -152, -162, -172,\r
+       -181, -190, -198, -206, -213, -220, -226, -231,\r
+       -237, -241, -245, -248, -251, -253, -255, -256\r
+};\r
+\r
+//FFTÖÐÊäÈëÐòÁÐÖØÐÂÅÅÐòºóµÄÐòÁРעÒâ:Ö»ÊÊÓÃÓÚµãÊýΪ128\r
+int reverse_matrix[]={\r
+       0,  64,  32,  96,  16,  80,  48, 112,   8,  72,  40, 104,  24,  88,  56, 120,\r
+       4,  68,  36, 100,  20,  84,  52, 116,  12,  76,  44, 108,  28,  92,  60, 124,\r
+       2,  66,  34,  98,  18,  82,  50, 114,  10,  74,  42, 106,  26,  90,  58, 122,\r
+       6,  70,  38, 102,  22,  86,  54, 118,  14,  78,  46, 110,  30,  94,  62, 126,\r
+       1,  65,  33,  97,  17,  81,  49, 113,   9,  73,  41, 105,  25,  89,  57, 121,\r
+       5,  69,  37, 101,  21,  85,  53, 117,  13,  77,  45, 109,  29,  93,  61, 125,\r
+       3,  67,  35,  99,  19,  83,  51, 115,  11,  75,  43, 107,  27,  91,  59, 123,\r
+       7,  71,  39, 103,  23,  87,  55, 119,  15,  79,  47, 111,  31,  95,  63, 127\r
+};\r
+\r
+/***********************************************²éÕÒ±í*************************************************************/\r
+\r
+\r
+\r
+/***************************************************************************************\r
+ÒÔÏÂΪʵÏÖ¶þά¾ØÕóµÄFFT±ä»»µÄÏà¹Øº¯Êý,ÉèAΪMÐÐNÁУ¬ÔòM,N±ØÐëÊÇ£²µÄÃÝ´ÎÊý\r
+Date:\r
+    2012-05-07\r
+****************************************************************************************/\r
+\r
+/*************************************************************************\r
+Function:\r
+       reverse\r
+Discription:\r
+       ÐòÁеÄÄæÐòÅÅÁÐ\r
+          ÓÉÓÚ x(n) ±»·´¸´µØ°´Æ桢ż·Ö×飬ËùÒÔÁ÷ͼÊäÈë¶ËµÄÅÅÁв»ÔÙÊÇ˳ÐòµÄ,\r
+          ÒÔÏÂÔòÊDZ£Ö¤Êä³ö¶ËÊÇ×ÔȻ˳Ðòʱ£¬ÊäÈë¶ËµÄ±äÖ·´¦Àí\r
+Input Arguments:\r
+       len: ÐòÁеij¤¶È\r
+          M:   ³¤¶È¶ÔÓ¦µÄ2µÄÖ¸Êý Èçlen = 8,M = 3\r
+Output Arguments:\r
+       b: ÊäÈë¶Ë±äÖ·´¦ÀíºóÊä³öµÄµ¹Î»Ðò\r
+Author:\r
+       Wu Lijuan\r
+Note:\r
+       N./A.\r
+Date:\r
+      $ID:  $ \r
+**************************************************************************/\r
+void reverse(int len, int M,int *b)\r
+{\r
+       //×¢ÒâbÔÚÍâÃæ±ØÐë³õʼ»¯Îª0\r
+    int i,j;\r
+\r
+       //aµÄ³õʼ»¯,aÓÃÓÚ´æ·Å±äÖ·ºóÐòÁеĶþ½øÖÆÊý\r
+       char *a = (char *)malloc(sizeof(char)*M); // aÓÃÓÚ´æ·ÅMλ¶þ½øÖÆÊý\r
+       memset(a,0,sizeof(char)*M);\r
+\r
+    for(i=1; i<len; i++)     // i = 0 Ê±£¬Ë³ÐòÒ»ÖÂ\r
+    {\r
+        j = 0;\r
+        while(a[j] != 0)\r
+        {\r
+            a[j] = 0;\r
+            j++;\r
+        }\r
+\r
+        a[j] = 1;\r
+        for(j=0; j<M; j++)\r
+        {\r
+            b[i] = b[i]+a[j] * Pow2_table[M-1-j];    //pow(2,M-1-j),½«¶þ½øÖÆaת»»Îª10½øÖÆb\r
+        }\r
+    }\r
+       free(a);\r
+}\r
+/*************************************************************************\r
+Function:\r
+       nexttopow2\r
+Discription:\r
+       y = nexttopow2(x);\r
+          Ôò2^yΪ´óÓÚµÈÓÚxµÄ×îСµÄ2µÄÕýÕûÊý´ÎÃݵÄÊý×Ö,\r
+          Èçx = 12£¬Ôòy = 4(2^4 = 16)\r
+Input Arguments:\r
+       x ±ØÐëÊÇÕýÕûÊý\r
+Output Arguments:\r
+       y: 2^yΪ´óÓÚµÈÓÚxµÄ×îСµÄ2µÄÕýÕûÊý´ÎÃݵÄÊý×Ö\r
+Author:\r
+       Wu Lijuan\r
+Note:\r
+       N./A.\r
+Date:\r
+      2013.02.23\r
+**************************************************************************/\r
+int nexttopow2(int x)\r
+{\r
+       int y;\r
+       int i = 0;\r
+       while(x > Pow2_table[i])\r
+       {\r
+               i++;\r
+       }\r
+       y = i;\r
+       return y;\r
+}\r
+/*************************************************************************\r
+Function:\r
+       fft_fixed_point\r
+Discription:\r
+       1D FFT±ä»» ²ÉÓö¨µã»¯¼ÆË㣬Ìá¸ßËÙ¶È\r
+Input Arguments:\r
+       A: ÓÃÓÚFFT±ä»»µÄÐòÁÐ,ͬʱҲÊÇÊä³ö½á¹û\r
+       fft_nLen:¡¡ÐòÁ㤶È\r
+          fft_M:     ³¤¶È¶ÔÓ¦µÄÖ¸Êý eg, 8 = 2^3\r
+Output Arguments:\r
+      \r
+Author:\r
+       Wu Lijuan\r
+Note:\r
+       ¾ßÌåÔ­Àí²Î¿¼±¾Îļþ¼ÐÏÂPPT:¿ìËÙ¸µÀïÒ¶±ä»»£¨µûÐÎÔËË㣩P31\r
+Date:\r
+      2013.06.25 \r
+**************************************************************************/\r
+void fft_fixed_point(int fft_nLen, int fft_M, RK_complex_INT * A)\r
+{\r
+       int i;\r
+       int L,dist,p,t;\r
+       int temp1,temp2;\r
+       RK_complex_INT *pr1,*pr2;\r
+       //double temp = 2*PI/fft_nLen;\r
+       \r
+        int WN_Re,WN_Im;       //WNµÄʵ²¿ºÍÐ鲿\r
+       int X1_Re,X1_Im;       //ÁÙʱ±äÁ¿µÄʵ²¿ºÍÐ鲿\r
+       int X2_Re,X2_Im;   \r
+        int WN_X2_Re;\r
+        int WN_X2_Im;\r
+\r
+       for(L = 1; L <= fft_M; L++)         //Íê³ÉM´ÎµûÐεĵü´ú¹ý³Ì\r
+       {\r
+               dist = Pow2_table[L-1];             //dist = pow(2,L-1); µûÐÎÔËËãÁ½½Úµã¼äµÄ¾àÀë\r
+               temp1 = Pow2_table[fft_M-L];        //temp1 = pow(2,fft_M-L);\r
+               temp2 = Pow2_table[L];              //temp2 = pow(2,L); \r
+               for(t=0; t<dist; t++)                 //Ñ­»·Íê³ÉÒò×ÓWNµÄ±ä»¯\r
+               {\r
+                       p = t * temp1;                    //p = t*pow(2,fft_M-L);  \r
+\r
+                       //WN_Re = int(cos(temp * p) * Pow2_table[Q_INPUT]);          //WµÄÈ·¶¨ (double)cos(2*PI*p/fft_nLen); \r
+       //              WN_Im = int(-1*sin(temp * p) * Pow2_table[Q_INPUT]);     //(double)(-1*sin(2*PI*p/fft_nLen));\r
+                       WN_Re = cos_table_INT_128[p];          //Q15\r
+                       WN_Im = -sin_table_INT_128[p];\r
+                       \r
+                       //Ñ­»·Íê³ÉÏàͬÒò×ÓWNµÄµûÐÎÔËËã\r
+                       for(i = t; i < fft_nLen; i = i + temp2)           //i=i+pow(2,L)   Note: i=i+pow(2,L) \r
+                       {\r
+                               //X(k) = X1(k) + WN * X2(k); Ç°°ë²¿·ÖX(k)¼ÆË㹫ʽ\r
+                               //X(k) = X1(k) - WN * X2(k); ºó°ë²¿·ÖX(k)¼ÆË㹫ʽ\r
+\r
+                               pr1 = A+i;\r
+                               pr2 = pr1+dist;\r
+\r
+                               X1_Re = pr1->real;        //X1_Re = A[i].real; \r
+                               X1_Im = pr1->image;       //X1_Im = A[i].image;\r
+                               X2_Re = pr2->real;        //X2_Re = A[i+dist].real;\r
+                               X2_Im = pr2->image;       //X2_Im = A[i+dist].image;\r
+\r
+\r
+                               //¼ÆËãWN * X2(k),ÊǸö¸´Êý\r
+                               WN_X2_Re = ((long long)WN_Re * X2_Re - (long long)WN_Im * X2_Im) >> Q_INPUT;\r
+                               WN_X2_Im = ((long long)WN_Im * X2_Re + (long long)WN_Re * X2_Im) >> Q_INPUT;\r
+                               \r
+                               //¼ÆËãX(k) = X1(k) + WN * X2(k);\r
+                               pr1->real = X1_Re + WN_X2_Re;     //A[i].real = X1_Re + WN_X2_Re;\r
+                               pr1->image = X1_Im + WN_X2_Im;    //A[i].image = X1_Im + WN_X2_Im; \r
+\r
+                               //¼ÆËãX(k) = X1(k) - WN * X2(k);\r
+                               pr2->real = X1_Re - WN_X2_Re;     //A[i+dist].real = X1_Re - WN_X2_Re;\r
+                               pr2->image = X1_Im - WN_X2_Im;    //A[i+dist].image = X1_Im - WN_X2_Im;\r
+                       }\r
+               }\r
+       }\r
+}\r
+\r
+/*************************************************************************\r
+Function:\r
+       fft_2D_fixed_point\r
+Discription:\r
+       2ά FFT±ä»» \r
+Input Arguments:\r
+       mLen,nLen ¾ØÕóµÄ¸ß¿í\r
+          M,N;     ³¤¶È¶ÔÓ¦µÄ2µÄ¶ÔÊý M = log2(mlen), N = log2(nlen);\r
+       A_In:¡¡¡¡Òª½øÐÐÄæ±ä»»µÄ¾ØÕó,ͬʱҲÊÇÊä³ö½á¹û\r
+          flag:    0¡¡ÕýÏòFFT±ä»»£¬¡¡1   ÄæÏòFFT±ä»»\r
+Output Arguments:\r
+       \r
+Author:\r
+       Wu Lijuan\r
+Note:\r
+       N./A.\r
+Date:\r
+      2013.06.25\r
+**************************************************************************/\r
+void fft_2D_fixed_point(int mLen,int nLen,int M,int N,RK_complex_INT *A_In,int flag)\r
+{\r
+       int i,j;\r
+       int len = mLen * nLen;\r
+\r
+       //int *b = NULL;\r
+       int *b = reverse_matrix;      //Ö±½Ó²éÄæÐò±í£¬Ìá¸ßЧÂÊ\r
+        RK_complex_INT *p;\r
+        //AÓÃÓÚ´¦Àí¾ØÕóÖÐÿÐеÄÊý¾Ý\r
+        RK_complex_INT * A;\r
+\r
+       if (flag == DXT_INVERSE)\r
+       {\r
+               p = A_In;\r
+\r
+               for(i=0; i<len; i++)\r
+               {\r
+                       //Äæ±ä»»¹«Ê½ÓëÕý±ä»»¹«Ê½Çø±ð,Æä½á¹ûµÄÐ鲿ÓëÕæʵ½á¹ûµÄÐ鲿²îÒ»¸ö¸ººÅ\r
+                       //µ«Í¨³£Çé¿öÏ£¬¶¼ÊǶÔifftµÄÄ£Öµ´¦Àí£¬¼´Re*Re + Im*Im,¹Ê¶Ô½á¹ûûʲôӰÏì\r
+                       //Ö»ÊÇÓ¦¸ÃÖªµÀÓÐÕâ¸öÇø±ð\r
+                       (p->image) *= -1;    //A_In[i].image = -A_In[i].image;\r
+                       p++;\r
+               }\r
+       }       \r
+\r
+       \r
+       A = (RK_complex_INT *)malloc(sizeof(RK_complex_INT)*nLen);\r
+\r
+\r
+       //ÏȶԾØÕóµÄÿһÐÐ×öFFT±ä»»\r
+       for(i=0; i<mLen; i++)\r
+       {\r
+               RK_complex_INT *pr1 = A;\r
+               RK_complex_INT *pr2 = NULL;\r
+               RK_complex_INT *pr3 = A_In + (i<<N);   //²»Í¬ÐеÄÆðʼµØÖ·\r
+\r
+               for(j=0; j<nLen; j++)\r
+               {\r
+                       //A[j].real = A_In[i*nLen+b[j]].real;        //È¡³öÖØÐÂÅÅÐòºóµÄÿÐÐÊý¾Ý\r
+                       //A[j].image = A_In[i*nLen+b[j]].image;\r
+\r
+                       pr2 = pr3 + b[j];                       //È¡´úÉ϶δúÂ룬Ìá¸ß·ÃÎÊЧÂÊ\r
+                       pr1->real = pr2->real;        \r
+                       pr1->image = pr2->image;\r
+                       pr1++;\r
+               }\r
+\r
+\r
+               fft_fixed_point(nLen,N,A);                                 //½øÐÐIFFT±ä»»£¬±ä»»ºóµÄ½á¹ûÈÔ´æÓÚA\r
+\r
+               if (flag == DXT_FORWARD)\r
+               {\r
+                       pr1 = A;             //ÖØÐÂÖÃλָÕë\r
+                       pr2 = pr3;\r
+\r
+                       for(j=0; j<nLen; j++)\r
+                       {\r
+                               //A_In[i*nLen+j].real = A[j].real;           //½«±ä»»½á¹û´æÈëÔ­¾ØÕóÖÐ\r
+                               //A_In[i*nLen+j].image = A[j].image;\r
+\r
+                               pr2->real = pr1->real;\r
+                               pr2->image = pr1->image;\r
+                               pr1++;\r
+                               pr2++;\r
+                       }\r
+               }\r
+               else\r
+               {\r
+                       pr1 = A;             //ÖØÐÂÖÃλָÕë\r
+                       pr2 = pr3;\r
+\r
+                       for(j=0; j<nLen; j++)\r
+                       {\r
+                               //A_In[i*nLen+j].real = A[j].real/nLen;           //½«±ä»»½á¹û´æÈëÔ­¾ØÕóÖÐ\r
+                               //A_In[i*nLen+j].image = A[j].image/nLen;\r
+\r
+                               pr2->real = pr1->real >> N;\r
+                               pr2->image = pr1->image >>N;\r
+                               pr1++;\r
+                               pr2++;\r
+                       }       \r
+               }\r
+\r
+       }\r
+       free(A);\r
+       //free(b);\r
+\r
+       //AÓÃÓÚ´¦Àí¾ØÕóÖÐÿÁеÄÊý¾Ý\r
+       A = (RK_complex_INT *)malloc(sizeof(RK_complex_INT)*mLen);\r
+\r
+\r
+       //ÏȶԾØÕóµÄÿһÁÐ×öFFT±ä»»\r
+       for(i=0; i<nLen; i++)\r
+       {\r
+               RK_complex_INT *pr1 = A;\r
+               RK_complex_INT *pr2 = NULL;\r
+               RK_complex_INT *pr3 = A_In + i;   //²»Í¬ÁеÄÆðʼµØÖ·\r
+\r
+               for(j=0; j<mLen; j++)\r
+               {\r
+                       //A[j].real = A_In[b[j]*nLen+i].real;        //È¡³öÖØÐÂÅÅÐòºóµÄÿÐÐÊý¾Ý\r
+                       //A[j].image = A_In[b[j]*nLen+i].image;\r
+\r
+                       pr2 = pr3 + (b[j]<<N);\r
+                       pr1->real = pr2->real;\r
+                       pr1->image = pr2->image;\r
+                       pr1++;\r
+               }\r
+\r
+               fft_fixed_point(mLen,M,A);                                 //½øÐÐIFFT±ä»»£¬±ä»»ºóµÄ½á¹ûÈÔ´æÓÚA\r
+\r
+               if (flag == DXT_FORWARD)\r
+               {\r
+                       pr1 = A;             //ÖØÐÂÖÃλָÕë\r
+                       pr2 = pr3;\r
+\r
+                       for(j=0; j<mLen; j++)\r
+                       {\r
+                               //A_In[j*nLen+i].real = A[j].real;           //½«±ä»»½á¹û´æÈëÔ­¾ØÕóÖÐ\r
+                               //A_In[j*nLen+i].image = A[j].image;\r
+\r
+                               pr2->real = pr1->real;\r
+                               pr2->image = pr1->image;\r
+                               pr1++;\r
+                               pr2 += nLen;\r
+                       }       \r
+               }\r
+               else\r
+               {\r
+                       pr1 = A;             //ÖØÐÂÖÃλָÕë\r
+                       pr2 = pr3;\r
+\r
+                       for(j=0; j<mLen; j++)\r
+                       {\r
+                               //A_In[j*nLen+i].real = A[j].real/mLen;           //½«±ä»»½á¹û´æÈëÔ­¾ØÕóÖÐ\r
+                               //A_In[j*nLen+i].image = A[j].image/mLen;\r
+\r
+                               pr2->real = pr1->real >> M;\r
+                               pr2->image = pr1->image >> M;\r
+                               pr1++;\r
+                               pr2 += nLen;\r
+                       }\r
+               }\r
+       }\r
+       free(A);\r
+}\r
+\r
+\r
+\r
+\r
diff --git a/arch/arm/mach-rockchip/rk_pm_tests/ft/fft/fft_fixed_point.h b/arch/arm/mach-rockchip/rk_pm_tests/ft/fft/fft_fixed_point.h
new file mode 100755 (executable)
index 0000000..e7fc32e
--- /dev/null
@@ -0,0 +1,25 @@
+#ifndef __FFT_FIXED_POINT_H_\r
+#define __FFT_FIXED_POINT_H_\r
+\r
+\r
+#define DXT_FORWARD  0\r
+#define DXT_INVERSE  1\r
+\r
+#define Q_INPUT 8\r
+\r
+typedef struct{\r
+       int real;\r
+       int image;\r
+}RK_complex_INT;\r
+\r
+\r
+//2012-05-07\r
+//ÒÔÏÂÊÇfft±ä»»µÄÏà¹Øº¯Êý\r
+//ÐòÁеÄÄæÐòÅÅÁÐ\r
+void reverse(int len, int M,int *b);\r
+int nexttopow2(int x);\r
+void fft_fixed_point(int fft_nLen, int fft_M, RK_complex_INT * A);\r
+void fft_2D_fixed_point(int mLen,int nLen,int M,int N,RK_complex_INT *A_In,int flag);\r
+\r
+\r
+#endif
\ No newline at end of file
diff --git a/arch/arm/mach-rockchip/rk_pm_tests/ft/fft/test.c b/arch/arm/mach-rockchip/rk_pm_tests/ft/fft/test.c
new file mode 100644 (file)
index 0000000..fd4be65
--- /dev/null
@@ -0,0 +1,353 @@
+#include "fft_fixed_point.h"\r
+#if 0\r
+#include <stdlib.h>\r
+#include <stdio.h>\r
+#else\r
+\r
+#include <linux/string.h>\r
+#include <linux/resume-trace.h>\r
+#include <linux/workqueue.h>\r
+#include <linux/mutex.h>\r
+#include <linux/module.h>\r
+#include <linux/syscalls.h>\r
+#include <linux/init.h>\r
+#include <linux/interrupt.h>\r
+#include <linux/delay.h>\r
+#include <linux/cpu.h>\r
+#include <linux/slab.h>\r
+#include <linux/freezer.h>\r
+#include <linux/vmalloc.h>\r
+#define free(ptr) kfree((ptr))\r
+//typedef unsigned char uchar;\r
+\r
+#define malloc(size) kmalloc((size), GFP_ATOMIC)\r
+\r
+#define printf(fmt, arg...) \\r
+               printk(KERN_EMERG fmt, ##arg)\r
+\r
+#endif\r
+\r
+\r
+extern int Pow2_table[20];\r
+typedef unsigned char uchar;\r
+\r
+uchar image[16384] = {\r
+       196,152,152,155,163,164,163,163,163,163,162,162,161,159,159,160,158,158,158,158,158,157,158,158,158,159,163,165,166,165,166,167,166,167,167,166,168,169,169,169,168,168,170,170,172,172,172,172,172,172,173,173,173,173,173,173,173,174,175,175,175,176,176,176,175,175,175,174,176,175,176,176,176,176,176,176,176,177,177,177,176,176,176,176,176,176,176,176,176,176,176,175,175,175,175,175,175,175,175,173,173,173,173,173,173,172,171,170,170,171,170,169,168,168,168,167,167,167,168,167,167,167,167,167,167,169,170,170,
+       197,153,152,155,163,164,163,163,163,163,162,162,161,159,159,160,158,158,158,158,158,159,158,158,158,159,163,165,166,165,166,167,166,167,167,166,168,168,169,169,170,170,170,170,172,172,172,172,172,172,173,173,173,173,173,173,173,174,175,175,175,176,175,176,175,175,175,174,176,175,176,176,176,176,176,176,176,177,177,177,176,176,176,176,176,176,176,176,176,176,176,175,175,175,175,175,175,175,175,173,173,173,173,173,172,172,171,171,170,171,170,169,168,168,168,167,167,167,168,167,167,167,167,167,167,168,167,167,
+       197,153,152,157,164,163,162,162,162,162,162,162,161,159,160,159,158,158,158,159,159,158,159,159,159,159,163,166,166,166,166,167,167,167,167,167,169,169,169,169,169,170,171,171,171,172,172,172,172,172,173,173,173,173,173,173,173,174,175,175,175,174,174,174,175,175,175,175,175,175,176,176,176,176,176,176,176,177,177,177,177,177,177,176,176,177,176,176,176,176,175,175,175,175,175,175,175,175,174,172,172,173,173,172,172,172,171,171,171,169,169,169,168,168,169,168,168,167,166,166,166,166,166,166,166,167,166,166,
+       197,153,152,159,164,163,162,162,162,162,161,160,160,159,160,159,158,158,158,159,159,159,160,159,159,159,163,166,166,166,166,167,167,167,167,167,169,169,169,169,169,170,171,171,171,172,172,172,172,172,173,173,173,173,173,173,173,174,175,175,175,174,174,174,175,175,175,175,175,175,176,176,176,176,176,176,176,177,177,177,177,177,177,176,176,177,176,176,176,176,175,175,175,175,175,175,175,175,174,172,172,173,173,172,172,171,170,170,170,169,169,169,168,168,169,167,167,167,166,166,166,166,166,166,166,167,166,166,
+       198,152,152,160,163,162,161,161,161,161,160,160,160,159,159,159,159,159,159,159,159,159,159,159,159,159,163,166,166,167,167,167,168,168,168,168,168,168,170,170,170,171,171,172,172,172,172,172,173,173,173,173,173,173,174,174,174,175,175,175,175,175,175,175,175,176,176,175,175,175,176,176,176,176,176,176,176,177,177,177,176,176,176,176,176,176,176,176,176,176,176,175,176,176,174,174,174,174,173,173,173,173,172,172,171,172,171,170,169,168,168,168,167,167,168,166,165,166,166,166,166,166,164,165,165,166,166,166,
+       197,152,152,161,163,162,161,161,161,160,160,160,160,159,159,159,159,159,159,159,159,159,159,159,159,159,163,165,166,167,167,167,168,168,168,168,168,167,170,171,171,171,171,172,172,172,172,172,173,174,173,173,173,173,174,174,174,175,175,175,175,175,175,175,175,176,176,175,175,175,176,176,176,176,176,176,176,177,177,177,176,175,176,176,176,176,176,176,176,176,176,175,177,176,174,173,173,173,173,173,173,173,172,172,171,171,171,170,169,168,168,168,167,167,167,166,166,166,166,166,165,165,163,165,165,166,166,166,
+       197,152,151,161,162,162,161,160,160,160,160,161,161,160,159,158,159,159,159,158,158,158,159,159,159,160,164,166,167,167,167,167,168,168,168,168,168,168,169,170,171,171,171,171,171,172,172,172,172,173,172,173,173,173,174,174,174,174,175,174,174,174,174,174,174,175,176,175,175,175,175,175,176,176,176,176,176,176,176,176,176,176,176,176,176,176,176,176,176,176,176,175,176,176,174,173,173,173,173,172,172,172,171,171,171,171,169,169,168,167,167,167,166,166,166,166,166,166,165,165,165,165,164,165,164,165,165,165,
+       197,152,152,161,162,162,161,160,160,160,160,160,160,159,158,158,159,159,159,158,158,158,159,159,159,160,166,168,168,167,167,167,168,168,168,168,168,168,169,170,171,171,171,171,171,172,172,172,172,172,172,173,173,173,173,173,173,173,173,173,173,174,174,174,174,174,175,175,175,175,175,175,176,176,176,176,176,176,176,176,176,176,176,176,176,176,176,176,176,176,176,175,175,176,174,173,173,173,173,172,172,172,171,171,171,171,169,169,168,167,167,167,166,166,166,165,165,165,165,165,165,165,164,164,163,164,164,164,
+       197,152,153,161,162,161,160,160,160,159,158,158,158,158,158,158,159,159,159,158,158,158,159,160,160,161,166,167,167,167,167,167,168,168,168,168,168,168,169,170,170,171,171,171,171,172,172,172,172,172,172,173,173,173,173,173,173,173,172,173,173,174,174,174,174,174,175,175,175,175,175,175,176,176,176,176,176,176,176,176,176,176,176,176,176,176,176,176,176,176,176,175,175,176,174,174,173,173,173,172,172,172,171,171,171,171,169,169,168,167,167,167,167,167,166,165,165,165,165,165,164,164,164,164,163,164,164,164,
+       196,152,155,161,160,159,159,159,159,158,157,157,157,159,159,159,159,159,159,159,159,159,159,160,161,162,166,167,167,167,167,167,168,168,168,168,168,168,169,169,169,170,171,171,171,171,172,172,172,172,172,173,173,173,174,174,174,174,174,174,174,174,174,174,175,175,175,175,175,175,176,176,176,176,175,176,176,176,176,176,176,176,176,176,176,176,175,175,175,175,175,175,175,175,175,175,173,173,172,171,171,171,171,171,171,170,168,168,168,168,168,166,166,166,166,165,165,165,164,163,163,163,163,164,164,163,163,163,
+       196,152,156,160,159,158,159,159,159,159,158,158,158,159,159,159,159,159,159,159,159,159,159,159,161,162,166,167,167,167,167,167,168,168,168,168,168,168,169,169,169,170,171,171,171,171,172,172,172,172,172,173,173,173,174,174,174,174,174,174,174,174,174,174,175,175,175,175,175,175,176,176,176,176,175,176,176,176,176,176,176,176,176,176,176,176,175,175,175,175,175,175,175,175,175,175,173,173,172,171,171,171,171,171,171,170,168,168,168,168,168,166,165,165,166,165,165,165,163,162,163,163,163,163,163,162,163,163,
+       196,151,156,160,159,159,159,159,159,159,158,158,158,158,158,158,158,160,160,159,159,159,158,159,161,162,165,167,167,167,167,167,167,167,167,169,169,169,169,169,169,170,171,171,171,171,171,172,172,172,172,173,173,173,173,173,174,174,173,174,174,173,173,173,175,175,174,175,175,175,175,175,175,175,175,175,175,175,175,175,176,176,175,177,177,176,176,175,175,175,176,175,175,175,175,175,173,173,172,171,171,171,171,171,171,169,167,167,167,168,167,167,165,165,165,165,164,164,163,163,163,163,163,163,163,163,163,163,
+       196,152,158,159,158,159,159,159,159,159,158,158,158,158,158,158,158,160,160,159,159,159,158,159,161,162,165,167,167,167,167,167,167,167,167,169,169,169,170,170,169,170,171,171,171,171,171,172,172,172,173,173,173,173,173,173,174,174,173,174,174,173,173,173,175,175,174,175,175,175,175,175,175,175,175,175,175,174,175,175,176,176,175,177,177,176,175,174,174,174,174,175,175,175,174,174,173,173,172,171,171,171,171,171,170,169,167,168,168,168,167,167,165,165,165,165,164,164,163,163,163,163,162,163,163,163,163,163,
+       196,153,158,158,158,159,158,157,157,157,157,157,157,158,158,159,158,158,158,160,160,160,160,160,160,164,167,167,167,168,168,168,168,168,168,169,169,169,169,169,169,170,171,171,171,171,172,172,172,172,174,172,172,172,173,173,173,173,173,173,173,173,173,173,173,173,173,173,173,173,174,174,174,174,175,176,176,174,176,177,176,176,177,176,176,176,175,174,174,174,173,173,173,173,172,172,171,171,171,171,171,171,171,171,170,170,169,169,169,167,166,166,165,165,165,164,164,164,164,163,162,162,163,164,164,162,163,163,
+       196,154,159,158,158,159,158,158,155,156,157,157,157,156,157,159,158,158,158,160,160,160,160,160,160,164,167,167,167,168,168,168,168,168,168,169,169,169,169,169,169,170,171,171,171,171,172,172,172,172,172,172,172,172,174,174,174,174,174,173,173,174,174,173,173,174,174,173,173,173,174,174,174,175,175,174,174,174,175,175,176,176,175,176,176,176,175,174,174,174,173,173,173,173,172,172,171,171,171,171,171,171,170,169,170,168,167,167,167,165,164,164,166,166,166,164,164,164,164,163,162,162,163,162,162,162,161,161,
+       197,155,158,158,158,158,158,158,157,157,157,157,157,157,158,159,157,157,158,158,158,158,158,159,161,165,167,167,167,167,167,167,167,167,167,168,168,168,169,169,169,169,169,170,170,171,171,171,172,173,172,172,172,172,174,174,174,174,174,173,173,174,173,172,173,173,173,172,174,173,174,174,174,175,174,174,174,173,175,176,175,175,175,174,174,174,174,174,174,174,173,174,174,172,173,173,171,171,169,169,170,170,169,168,168,168,168,168,167,165,166,166,165,163,164,164,164,163,163,163,163,163,163,162,162,161,161,161,
+       197,155,157,156,156,158,158,158,157,157,157,157,157,157,156,157,157,157,158,158,158,158,158,159,161,165,167,167,167,167,167,167,167,167,167,168,168,168,169,169,169,169,169,170,170,171,171,171,172,173,172,173,173,173,175,175,175,175,175,173,173,172,172,172,172,171,171,172,174,173,175,175,174,175,174,174,174,173,174,174,175,175,175,174,174,174,174,174,174,174,173,172,172,172,171,171,171,171,169,169,170,170,169,168,168,168,168,168,167,165,164,164,165,164,165,163,162,163,162,161,161,161,161,162,161,159,159,159,
+       196,156,157,157,157,157,157,157,157,157,157,157,157,156,156,156,157,157,157,158,159,159,159,159,161,165,167,167,168,167,167,167,168,168,168,167,167,168,168,168,168,167,168,169,170,170,171,171,172,172,172,172,172,172,172,172,172,172,172,171,171,171,171,172,172,170,168,169,173,171,171,169,170,174,174,173,172,173,173,172,173,173,173,173,173,173,173,173,173,173,172,172,172,171,171,171,171,170,169,170,170,169,168,168,168,168,166,165,165,164,164,164,163,163,164,163,162,162,162,161,161,161,160,160,160,161,160,161,
+       196,157,156,157,157,156,155,155,155,156,157,157,157,156,156,156,155,156,157,158,159,159,159,159,161,165,167,167,168,167,167,167,168,168,168,167,167,168,168,168,168,167,168,168,168,170,171,171,170,171,172,172,172,172,172,172,172,172,170,169,164,162,169,169,169,169,164,155,154,163,162,158,170,172,171,174,174,173,172,172,173,173,173,173,173,173,173,173,173,173,172,172,172,171,171,171,171,170,169,168,168,167,168,168,168,168,166,165,165,164,164,164,163,163,162,162,162,162,162,161,161,161,160,160,160,161,161,161,
+       198,159,156,157,157,156,155,155,155,156,156,157,157,156,156,157,156,157,158,159,159,159,158,158,159,165,167,166,166,166,166,166,167,167,167,168,168,168,168,168,168,168,168,169,169,171,171,171,171,171,171,171,171,171,171,171,172,172,170,167,148,120,121,142,152,142,126,119,85,101,108,98,136,167,170,172,173,172,173,173,172,172,172,172,172,173,172,172,172,172,173,171,171,171,170,170,170,170,170,168,168,168,168,168,168,167,165,166,166,165,164,164,163,163,162,161,161,160,161,162,161,161,161,161,160,162,161,159,
+       199,159,156,156,156,156,154,153,153,154,156,156,156,156,156,157,156,156,157,159,160,160,159,159,161,166,167,166,166,166,166,166,167,167,167,168,168,168,168,168,168,168,168,168,169,171,171,171,171,171,171,171,171,171,171,171,171,171,169,162,139,87,55,63,78,65,56,57,50,54,49,42,54,96,135,155,163,170,173,174,172,172,172,172,172,172,172,172,172,172,172,171,171,171,170,170,170,170,170,168,168,168,168,168,167,166,165,165,165,165,164,164,164,164,163,162,162,161,161,161,160,160,160,160,159,161,161,159,
+       199,157,156,155,155,155,155,155,154,154,156,155,155,156,156,156,156,157,156,157,159,159,159,159,161,166,167,166,166,166,166,166,167,167,167,167,167,167,168,168,168,168,168,168,168,170,169,170,170,170,170,170,170,170,171,171,170,170,168,158,144,100,82,49,38,41,43,37,35,40,42,43,46,45,53,66,93,138,170,173,172,172,171,171,171,171,171,171,172,172,171,171,171,171,170,170,169,169,168,167,167,167,167,167,165,165,165,164,164,165,164,164,164,164,164,162,162,162,161,161,160,160,159,160,158,159,159,158,
+       199,155,154,155,154,154,155,155,154,154,155,155,155,156,156,156,157,158,158,157,158,158,159,159,162,166,167,166,166,166,166,166,167,167,167,167,167,167,168,168,168,168,168,168,168,168,167,168,169,169,169,170,170,170,170,169,168,167,161,157,131,96,86,51,38,38,36,33,33,36,40,45,46,45,42,40,44,64,129,169,169,171,173,172,171,171,171,171,171,171,171,171,171,170,170,170,168,168,167,167,167,167,166,166,165,165,165,164,164,164,164,164,163,163,163,162,162,161,160,160,160,160,159,160,158,159,159,159,
+       198,155,153,155,155,153,155,155,155,155,154,154,154,155,156,156,156,156,156,157,159,158,158,158,162,166,166,166,166,167,167,167,166,166,167,168,168,168,168,168,168,168,167,167,167,167,167,168,168,168,169,170,170,170,170,166,163,148,121,97,59,41,41,40,37,35,32,34,36,35,37,36,35,37,39,38,38,38,56,119,165,171,171,171,171,171,171,171,170,170,170,170,170,168,169,169,168,168,168,168,167,167,166,165,165,164,164,164,164,165,164,162,163,163,163,162,162,161,160,158,159,159,159,159,159,159,160,159,
+       198,154,152,154,154,154,154,154,154,154,154,154,154,155,156,156,156,156,156,157,159,158,158,158,162,165,166,166,166,167,167,167,166,166,167,167,167,168,168,168,168,168,167,167,167,168,168,168,168,168,169,168,168,168,168,163,141,97,57,45,33,30,35,35,32,31,30,37,40,39,36,32,31,31,34,36,37,36,37,48,105,158,170,169,169,170,170,170,170,170,170,170,170,168,168,168,167,167,167,168,166,166,165,165,165,165,164,164,164,163,164,162,162,162,162,162,162,161,160,159,159,159,159,159,159,158,159,158,
+       198,154,151,154,154,154,153,153,153,153,154,154,154,155,156,156,156,156,156,157,159,158,158,158,163,166,166,166,166,167,167,167,166,166,167,167,167,167,168,168,168,168,167,167,167,168,168,168,168,168,169,168,168,168,164,144,102,58,41,43,38,31,31,31,28,29,30,32,35,37,38,34,30,27,28,32,35,37,35,37,48,87,153,168,169,169,168,169,169,169,170,170,169,168,168,168,167,167,166,166,165,165,165,165,165,165,165,164,164,163,163,162,161,161,161,162,161,160,159,159,159,159,159,158,158,158,159,158,
+       199,155,152,154,154,154,154,154,154,154,154,155,155,154,154,156,156,156,156,157,159,158,157,159,164,167,167,166,166,166,166,166,167,167,167,167,167,167,167,167,167,167,167,167,167,167,167,167,167,167,168,168,168,166,153,114,68,41,39,37,35,33,30,28,27,26,31,36,35,34,36,39,33,30,29,30,32,35,34,35,38,42,94,164,170,167,168,168,168,168,169,168,168,167,168,169,168,168,167,165,165,165,165,165,165,165,165,164,164,164,164,162,162,162,162,161,160,160,160,160,159,159,158,157,157,158,158,158,
+       199,155,152,153,153,153,154,154,154,154,154,155,155,154,153,155,156,156,156,157,158,158,157,159,164,167,167,166,166,166,166,166,167,167,166,166,166,167,167,167,167,167,166,167,167,168,167,167,168,168,167,168,165,163,140,78,44,38,36,33,31,29,26,24,30,29,28,30,28,29,30,32,34,36,34,30,30,34,33,35,37,36,40,98,158,169,168,167,168,168,169,168,167,167,167,168,168,168,166,165,165,165,165,165,165,165,165,163,163,163,164,163,162,162,162,160,159,160,160,160,159,159,159,158,158,157,157,157,
+       198,154,153,153,152,153,154,154,154,154,155,156,156,155,155,156,156,157,156,157,157,158,158,159,164,166,166,166,166,167,167,167,166,166,166,167,167,167,166,166,166,166,167,167,167,169,167,167,169,169,167,168,164,163,106,49,38,30,30,26,26,27,30,33,37,37,31,26,22,24,24,26,29,34,34,29,29,28,31,30,35,38,30,35,84,154,171,167,167,167,168,169,167,166,167,168,167,167,165,164,165,165,165,165,164,164,164,164,164,164,163,163,162,162,161,159,159,159,160,159,158,158,159,158,158,157,158,156,
+       198,154,153,152,151,153,153,154,154,154,155,154,154,155,156,157,155,156,156,157,157,158,158,159,164,166,166,166,166,167,167,167,166,166,165,167,167,167,166,166,166,166,167,167,167,167,167,167,167,167,166,166,164,129,68,42,29,26,31,24,22,28,35,39,41,38,34,30,26,24,23,20,22,25,28,27,24,24,27,24,28,32,32,29,36,81,152,168,166,166,167,167,167,165,167,168,167,166,166,165,165,165,165,164,164,164,164,164,164,164,162,161,162,162,161,159,158,158,159,158,158,158,158,158,158,156,157,157,
+       197,152,151,154,153,154,153,153,153,153,153,153,153,155,155,155,155,155,156,156,158,157,158,158,164,167,166,165,165,166,166,166,166,166,166,167,166,167,166,166,166,166,167,166,166,167,167,167,166,167,168,164,140,76,45,31,23,30,34,29,23,25,32,36,39,36,33,30,28,26,25,25,25,24,26,25,23,22,23,25,25,25,29,30,29,32,92,161,168,167,168,168,168,166,166,166,165,165,165,165,165,166,166,165,165,164,163,163,163,163,162,162,163,161,161,160,159,158,158,158,158,158,156,156,156,155,155,155,
+       198,152,150,152,151,152,151,151,153,153,153,153,153,153,154,155,155,155,156,156,158,157,157,159,165,167,166,165,165,166,166,166,166,166,166,167,166,167,166,166,166,166,165,166,166,167,167,167,168,168,166,159,107,47,36,24,24,34,33,31,26,25,27,29,33,34,33,32,31,30,26,27,28,27,25,26,25,24,21,24,25,23,25,26,28,24,39,124,168,167,167,167,167,166,166,166,165,165,165,165,165,164,165,164,163,163,163,163,163,163,162,162,162,161,161,160,160,158,157,158,158,158,156,156,156,155,155,155,
+       197,152,150,152,152,151,151,151,153,153,154,153,154,155,155,154,156,156,156,156,158,157,157,159,166,167,166,166,166,166,166,166,166,166,166,167,167,167,167,166,166,166,166,166,166,166,166,166,167,167,165,148,66,32,31,25,27,33,28,31,28,28,28,26,29,31,31,32,30,32,34,31,29,30,27,28,29,29,27,25,22,22,23,24,26,26,26,67,156,168,166,166,165,165,165,165,165,165,164,165,165,165,164,163,163,164,165,163,163,163,162,162,161,160,160,160,160,159,159,157,158,158,157,156,155,155,156,155,
+       197,152,149,150,150,151,151,151,151,151,152,152,153,153,154,154,154,155,156,156,158,157,157,159,166,167,166,166,166,166,166,166,166,166,166,167,167,167,166,165,166,166,166,166,166,166,166,166,167,166,162,110,34,30,29,29,29,31,26,29,26,25,31,29,32,31,31,30,28,33,39,40,35,32,27,30,36,37,38,35,26,20,20,23,25,26,24,37,125,169,166,166,164,165,165,165,165,165,164,165,165,165,164,163,163,163,163,161,161,163,163,163,162,162,162,159,158,157,158,157,156,156,155,155,154,155,156,155,
+       196,151,150,150,150,151,151,151,151,151,151,151,153,153,153,153,155,155,156,157,158,159,158,159,166,167,168,168,168,167,167,167,166,165,166,166,166,166,166,166,165,165,165,166,166,167,167,165,168,166,154,71,28,29,30,29,29,31,27,29,29,29,35,34,38,32,32,34,33,34,42,48,45,42,34,33,40,46,53,54,44,28,20,21,24,25,23,29,93,166,165,167,165,165,166,166,166,165,166,165,165,164,165,165,164,164,163,163,163,162,162,161,160,159,159,160,159,159,159,158,158,158,156,156,156,156,157,156,
+       196,151,150,150,150,151,151,151,151,151,151,151,153,153,153,153,154,154,156,156,157,158,158,159,166,167,168,168,168,167,167,167,166,165,164,166,166,166,166,166,165,165,165,164,164,166,166,165,166,167,131,59,29,26,31,27,27,33,32,32,33,34,39,34,36,34,37,38,36,38,44,55,58,53,43,37,40,52,59,71,69,49,32,22,23,23,22,24,49,140,165,165,164,165,165,165,165,165,166,164,163,164,165,165,164,164,163,163,163,162,161,160,160,158,158,158,159,159,159,159,158,158,155,154,155,156,156,155,
+       197,151,151,150,150,151,151,151,151,151,151,151,152,153,154,152,154,155,156,156,157,156,157,160,165,167,169,167,167,167,168,168,165,165,165,165,165,165,165,166,166,166,165,165,165,166,167,166,166,161,124,72,27,29,33,29,29,35,38,38,41,41,43,36,36,38,42,40,37,41,41,54,69,59,54,44,43,58,66,70,82,74,58,36,26,24,22,23,27,95,162,164,165,165,165,165,166,165,166,165,164,163,164,164,163,163,163,164,164,162,162,161,159,159,159,159,159,160,159,158,158,157,156,155,156,155,155,155,
+       197,151,149,149,149,150,151,151,151,151,151,151,151,153,153,153,155,155,156,156,157,156,157,161,165,166,168,167,167,167,167,167,165,165,165,165,165,165,165,166,166,166,165,165,165,166,166,166,165,154,121,58,28,29,32,30,33,41,40,44,50,50,50,40,39,48,52,49,43,43,46,48,70,71,65,58,49,57,65,68,82,89,84,66,41,30,23,22,24,51,152,167,165,165,165,165,164,164,166,165,164,163,163,163,163,163,163,164,164,162,163,161,159,159,159,160,160,158,158,158,158,157,155,155,155,155,155,155,
+       196,149,149,149,149,150,150,151,151,151,151,151,152,152,152,153,154,154,155,156,156,157,157,160,166,166,167,167,167,166,165,165,165,165,165,165,165,165,166,166,165,166,166,166,166,166,166,166,166,152,93,37,31,29,32,31,39,47,41,50,60,59,58,50,49,54,65,65,59,46,55,55,63,81,75,70,59,60,68,77,88,93,95,89,70,48,31,23,23,27,112,170,166,165,164,164,164,165,166,164,164,165,164,164,163,163,162,163,163,162,162,161,160,160,160,159,159,158,158,157,157,156,155,155,155,155,155,154,
+       195,149,149,150,150,150,151,151,151,151,151,151,152,151,151,153,153,153,156,156,156,157,158,161,165,166,167,167,167,166,165,165,165,165,165,165,166,166,166,166,165,166,166,166,166,166,166,167,168,140,60,34,33,31,32,34,45,50,42,53,63,67,64,60,61,62,80,80,78,65,60,67,75,90,88,83,76,69,79,82,86,93,94,95,90,80,56,29,25,24,84,165,166,164,164,164,164,164,164,164,164,165,164,164,163,163,162,162,160,161,160,161,160,160,160,158,158,159,158,157,156,155,155,155,155,154,155,154,
+       196,149,149,150,149,149,151,151,151,150,151,151,151,151,151,153,154,154,156,156,156,157,158,162,164,165,167,167,167,166,165,165,165,165,166,166,167,166,165,165,166,166,166,166,166,166,166,165,168,119,44,33,31,30,29,39,49,52,40,53,64,69,72,70,81,86,95,95,95,87,79,80,90,96,100,100,94,85,79,92,90,92,91,92,95,96,84,41,27,26,85,159,163,163,164,164,164,164,164,163,163,164,163,162,163,163,163,161,161,161,160,161,160,160,160,158,158,157,158,157,155,155,155,155,155,154,154,154,
+       195,149,148,149,149,150,150,150,150,150,151,151,152,152,152,153,154,154,155,156,156,157,158,162,165,166,167,168,166,166,165,165,165,165,165,166,166,165,165,165,165,165,165,167,165,166,166,165,163,89,34,33,28,26,30,43,50,52,43,55,70,72,79,80,92,98,103,105,103,100,97,94,96,99,106,109,108,105,97,95,96,97,94,93,96,98,94,55,27,28,77,161,165,164,164,164,164,164,164,164,164,164,163,162,163,163,162,162,162,161,160,160,160,160,160,158,158,157,158,157,155,155,155,155,155,155,155,153,
+       195,148,148,149,149,150,150,150,150,150,151,151,152,152,152,153,154,154,155,156,156,157,158,162,165,165,167,167,166,166,165,165,165,165,165,166,166,165,166,166,164,165,166,166,165,166,166,166,157,69,30,31,27,26,33,49,56,55,52,62,78,81,84,86,93,95,101,104,107,106,103,97,97,102,106,108,104,102,99,97,98,98,97,96,95,95,94,66,28,27,58,159,166,165,164,164,165,165,165,164,164,164,163,162,163,162,161,162,162,161,160,160,160,160,160,159,159,158,157,156,155,155,155,155,155,155,155,153,
+       195,148,149,149,150,151,151,151,151,151,151,151,152,151,152,152,153,153,154,155,157,157,157,163,165,165,166,165,165,165,165,165,165,165,165,165,165,165,166,166,165,166,166,165,165,166,166,166,147,55,29,27,25,26,33,59,65,65,66,74,80,83,86,88,89,88,92,98,100,100,97,93,94,99,100,98,96,93,93,93,97,102,102,100,96,92,90,76,33,24,47,147,167,166,165,166,165,165,165,164,164,164,164,163,163,163,162,161,161,160,160,160,160,159,159,159,159,158,157,156,155,155,155,155,155,154,153,153,
+       194,148,149,149,150,151,151,151,151,151,151,151,152,151,152,152,153,153,154,155,157,157,157,163,165,165,166,165,165,165,165,165,165,165,164,165,165,165,166,166,165,165,165,165,165,166,166,167,142,51,26,25,23,30,34,70,76,78,78,78,80,85,88,89,89,84,85,89,91,95,93,89,89,91,90,86,85,85,88,91,95,95,93,94,97,92,88,82,40,21,44,149,168,166,165,166,164,164,164,164,164,164,163,162,163,163,162,160,160,159,160,160,160,159,158,159,159,158,157,156,154,154,155,155,155,154,152,152,
+       195,148,148,147,149,150,150,150,150,150,151,151,151,151,151,152,153,153,156,156,157,156,156,164,165,165,166,165,165,165,165,165,165,164,164,164,164,164,165,165,165,165,165,165,165,166,165,166,136,47,25,25,25,39,45,74,78,80,80,83,87,92,93,92,90,88,85,84,84,86,86,84,82,84,83,78,79,78,77,80,83,76,71,77,87,90,90,84,47,22,43,154,167,166,166,165,164,164,164,164,164,164,164,162,163,163,162,162,161,159,160,160,159,159,159,158,158,157,156,156,155,155,155,154,154,154,153,153,
+       195,148,149,149,150,150,150,150,150,150,150,151,151,151,151,152,153,153,156,155,156,156,157,163,165,164,164,165,165,165,165,165,165,164,164,164,164,164,165,165,165,165,165,165,165,166,166,164,135,48,25,24,26,46,64,77,77,78,84,88,84,88,89,87,85,83,79,76,74,74,77,76,73,75,74,71,67,62,56,55,56,58,60,66,75,85,90,86,59,24,40,152,167,163,165,165,164,164,164,164,164,164,163,162,163,163,162,162,161,159,160,160,159,159,159,158,158,157,156,155,155,155,154,154,154,154,153,153,
+       195,149,149,148,149,149,150,150,150,150,150,151,151,151,151,153,152,154,156,155,157,156,158,163,165,165,164,164,164,164,164,164,165,165,164,163,163,164,165,165,164,164,165,166,165,165,164,163,145,50,25,23,26,53,73,78,75,80,84,80,72,72,70,66,60,55,56,60,62,68,69,69,68,68,66,59,47,38,35,34,37,45,54,63,72,85,90,88,68,29,30,132,167,163,163,163,163,164,165,163,162,163,163,162,161,161,161,160,160,160,160,160,158,158,158,157,157,157,155,155,155,155,153,152,153,153,152,152,
+       195,149,149,147,148,148,149,150,150,150,150,151,151,151,151,153,152,153,155,155,157,156,158,163,165,165,166,166,166,164,164,164,165,165,164,164,164,165,166,165,164,164,165,166,165,166,164,164,153,58,24,23,27,57,76,78,77,80,75,71,64,59,50,44,38,35,37,42,48,58,63,66,64,61,59,49,39,34,35,35,38,47,55,62,71,80,86,85,75,37,28,118,168,163,163,163,162,163,163,162,162,163,163,162,161,161,161,160,160,160,160,160,158,158,158,157,157,155,155,155,155,155,153,152,152,151,152,152,
+       194,147,146,148,148,149,149,149,150,151,151,151,151,151,151,151,153,153,154,156,156,156,157,163,164,164,164,164,164,165,166,165,165,165,164,165,165,166,166,165,165,165,165,165,165,166,164,166,160,67,21,21,31,61,78,80,81,81,72,68,61,57,49,42,39,39,40,42,45,51,59,63,64,59,52,46,44,41,43,44,51,58,62,64,66,72,79,84,86,46,32,123,167,163,164,164,162,162,162,162,162,163,162,162,161,161,163,162,161,161,160,159,159,159,158,158,156,155,155,155,154,154,152,151,152,153,152,152,
+       194,147,146,148,148,147,148,149,150,151,151,150,150,151,151,151,151,152,154,156,156,156,159,164,164,164,164,164,164,165,166,165,165,165,164,164,164,165,166,165,165,165,165,165,165,165,165,166,162,73,20,21,37,67,77,82,83,81,70,64,62,63,59,54,48,44,42,43,44,47,54,62,65,58,49,45,43,44,49,54,60,64,65,62,61,66,73,83,91,55,32,128,168,163,162,162,162,162,162,162,162,163,161,160,161,160,161,161,160,159,159,159,159,159,158,156,155,155,154,153,152,153,153,150,150,152,150,150,
+       193,145,146,147,147,147,148,149,149,149,150,151,151,151,151,152,153,154,155,155,155,154,160,164,164,164,165,165,163,164,165,164,163,164,163,164,164,165,164,163,165,164,165,165,165,165,164,165,163,85,22,23,41,70,75,81,81,76,68,65,65,67,70,66,58,49,44,42,42,45,54,66,69,61,49,44,44,47,51,53,56,59,61,61,63,62,66,80,94,62,34,127,167,162,161,162,161,161,161,160,160,160,160,161,160,160,161,161,160,161,160,159,157,157,156,156,155,154,153,153,152,152,152,152,152,152,151,148,
+       193,145,146,148,147,147,148,150,149,149,150,151,151,151,151,153,153,154,155,156,156,155,160,164,164,164,165,165,163,164,164,163,163,164,163,165,165,164,163,163,165,164,165,165,165,165,164,164,165,99,25,23,41,73,76,76,76,70,64,64,64,65,70,71,65,56,53,46,43,44,54,69,77,65,49,44,46,47,51,53,52,53,54,56,63,69,69,73,94,63,40,122,163,161,161,161,161,161,161,160,160,160,160,160,160,160,161,161,159,161,158,158,156,156,154,155,154,152,153,153,151,152,153,151,150,149,150,150,
+       193,146,147,148,148,148,149,149,149,149,149,150,150,150,151,153,153,153,154,155,155,154,160,164,163,163,164,165,163,163,163,164,162,163,163,164,164,164,164,163,165,164,165,165,165,165,165,165,166,113,26,23,39,76,78,74,72,66,59,61,69,69,69,71,65,56,54,52,49,47,50,73,90,72,48,44,51,50,46,36,30,37,43,46,56,68,81,75,79,60,72,144,162,160,159,159,161,161,161,159,159,160,160,160,159,159,160,160,158,159,157,157,156,156,155,155,154,153,152,151,152,151,149,151,153,153,150,139,
+       194,146,146,147,147,147,147,147,147,148,148,149,149,150,151,153,153,153,154,155,155,154,160,164,163,163,164,165,163,163,163,164,162,162,163,164,163,165,164,163,165,164,165,165,165,165,165,165,167,133,33,24,36,76,79,75,70,61,61,68,63,55,58,53,45,47,49,49,47,51,45,52,80,70,41,46,53,46,38,26,27,38,43,40,52,68,83,72,35,53,117,160,162,161,160,160,160,160,160,159,159,159,159,159,158,158,158,158,156,156,155,155,155,155,155,154,154,153,151,153,155,154,148,137,116,89,60,42,
+       195,145,146,146,146,146,146,146,147,147,148,149,150,151,151,152,152,153,155,155,156,156,161,163,162,163,163,163,163,162,163,163,163,163,164,163,164,164,163,163,164,164,164,164,164,165,165,165,168,146,50,26,31,71,80,76,64,56,64,62,53,47,41,27,25,30,44,45,47,54,47,63,105,99,57,47,56,47,51,37,34,52,63,68,70,76,84,74,58,61,126,163,164,162,161,161,159,159,158,159,159,158,158,157,156,156,156,156,156,157,156,155,155,155,156,155,158,157,143,123,99,71,53,38,29,24,24,27,
+       194,145,146,146,146,146,147,147,148,148,148,148,150,151,151,152,151,153,155,155,156,157,162,163,162,163,163,162,162,162,163,163,163,164,163,162,164,163,163,163,163,163,163,163,163,165,165,165,166,160,68,23,27,64,77,67,40,41,66,58,46,39,49,32,24,39,52,46,56,54,47,68,100,96,64,40,59,66,63,62,65,74,79,85,90,92,90,80,76,71,98,157,162,160,159,159,158,158,158,157,157,157,156,156,158,158,157,156,154,153,155,159,159,155,141,126,99,70,45,31,28,26,24,24,25,24,24,26,
+       194,144,146,145,145,146,147,147,147,148,149,149,150,151,151,152,151,153,155,155,155,157,163,164,163,163,163,162,162,162,163,163,162,163,162,162,164,163,163,163,163,163,163,164,164,165,165,164,163,167,92,22,37,68,57,65,67,64,70,60,58,60,64,56,51,61,62,63,66,52,39,70,101,102,72,44,67,71,65,67,70,73,76,83,93,98,99,85,82,72,85,162,161,159,158,158,158,158,158,156,156,156,155,154,154,154,156,157,159,153,139,115,88,63,42,30,24,23,25,25,26,25,26,27,28,31,34,33,
+       195,143,144,145,145,146,146,146,146,148,150,149,149,150,151,152,152,154,154,154,155,157,163,164,164,163,162,162,162,162,162,162,162,162,162,163,163,164,164,164,164,163,163,164,164,165,165,164,165,167,130,42,29,53,60,71,73,70,75,77,78,74,73,70,65,60,61,65,74,70,49,76,109,121,96,58,74,74,67,73,75,78,83,89,98,105,103,88,89,70,75,160,158,157,157,157,158,157,156,155,155,156,157,159,159,156,147,122,92,62,39,26,22,22,23,24,23,23,26,26,27,28,28,28,31,32,29,26,
+       194,142,143,145,145,145,144,145,145,146,147,148,148,149,150,151,152,153,153,154,155,157,163,164,163,163,162,162,162,162,161,161,162,162,162,163,163,164,164,164,164,163,163,163,163,164,164,164,166,162,123,62,30,69,72,70,76,80,77,89,87,78,72,68,65,65,64,70,80,70,59,82,111,127,110,73,64,73,70,71,73,79,86,93,98,97,89,94,94,76,76,158,156,157,156,156,156,156,156,158,161,156,142,116,90,65,41,28,24,25,25,23,22,24,26,27,30,31,29,24,24,25,24,26,25,26,29,30,
+       193,143,144,145,145,145,145,145,146,147,147,148,149,150,150,150,151,152,154,154,155,158,164,164,163,163,164,163,163,162,162,162,162,162,163,163,163,164,164,163,163,163,163,163,163,163,163,163,165,149,81,69,44,72,77,73,77,86,79,78,90,87,79,73,70,65,62,74,70,63,67,86,111,130,120,87,73,72,70,64,62,66,73,80,88,91,95,96,94,78,88,158,156,155,157,159,159,160,148,124,91,60,37,25,21,22,23,25,25,26,26,29,29,29,32,34,33,28,26,27,29,28,29,32,33,36,37,35,
+       193,144,145,145,145,145,145,145,146,147,147,147,148,150,150,150,151,152,154,154,155,158,164,164,163,164,165,163,163,163,163,163,162,162,163,163,163,164,164,163,163,163,163,163,163,163,163,163,164,154,60,64,49,70,81,79,80,86,91,86,79,76,74,72,68,60,62,72,72,65,71,89,110,130,125,97,77,81,87,81,73,70,77,86,92,94,92,92,93,82,121,161,163,158,147,124,96,65,40,26,24,25,24,24,24,24,26,30,30,29,26,25,25,23,24,24,24,27,30,31,33,34,37,39,34,29,29,28,
+       193,143,144,146,145,145,145,145,145,147,147,147,147,150,151,151,151,151,153,154,154,157,164,164,163,163,163,163,161,162,162,162,162,164,163,163,163,164,163,162,162,162,162,162,162,164,164,164,163,161,67,52,46,70,80,82,85,88,92,97,96,85,75,67,64,67,76,80,72,62,77,91,109,127,127,107,79,75,86,94,94,88,88,88,87,84,83,85,86,80,129,137,100,66,41,26,20,20,23,24,25,26,27,29,29,33,34,34,27,25,26,27,28,27,31,35,35,36,38,40,41,41,35,28,28,31,33,33,
+       193,143,145,145,144,145,145,145,145,147,147,147,147,150,151,151,151,151,153,153,154,158,165,164,163,163,163,163,161,162,162,162,163,164,162,163,163,164,163,162,162,162,162,162,162,162,162,162,163,162,73,54,48,72,78,83,87,90,90,88,88,84,80,75,74,79,81,76,67,65,78,92,107,125,129,114,87,71,78,92,95,93,91,88,81,78,77,78,80,79,57,29,23,22,22,23,24,24,28,30,28,25,25,24,24,25,25,25,28,29,29,33,37,39,35,33,29,28,31,32,28,28,31,33,34,35,35,34,
+       193,143,143,144,144,145,145,145,146,146,147,147,148,148,149,150,151,152,154,153,153,159,166,166,164,163,163,163,163,162,163,163,162,162,162,162,162,161,161,161,162,162,162,162,163,163,162,162,164,164,78,61,53,73,81,84,87,87,83,81,79,78,78,78,80,82,80,73,66,68,82,95,103,113,113,108,96,81,72,84,92,92,89,88,84,78,74,74,74,76,39,21,26,25,28,29,32,34,32,27,23,26,27,28,31,32,36,38,38,41,42,42,38,29,28,29,31,30,33,31,34,39,40,40,39,40,39,42,
+       193,142,141,143,144,145,145,145,144,144,146,146,148,148,148,148,151,151,153,153,154,159,164,164,164,164,163,163,163,162,162,161,162,162,162,162,162,161,161,161,162,162,163,163,163,163,162,162,163,166,104,64,53,73,82,82,82,80,78,76,75,73,74,79,80,81,76,70,63,81,93,91,90,91,92,88,88,99,77,79,88,91,88,86,82,78,75,72,71,71,42,29,26,25,24,24,26,27,25,28,27,31,31,37,40,35,32,30,31,33,30,29,27,31,34,32,35,36,38,39,38,35,35,35,35,36,34,29,
+       193,142,142,143,143,145,146,146,145,146,146,146,148,148,148,149,149,151,153,155,154,161,164,163,163,163,163,164,164,162,162,162,161,161,161,161,161,161,162,162,161,162,163,164,164,162,162,162,162,165,139,67,52,73,81,79,78,78,78,77,78,78,82,85,85,85,81,73,70,92,85,73,71,73,73,66,65,81,80,81,90,92,90,88,82,79,76,70,68,68,41,25,25,27,26,29,31,35,38,37,37,39,42,35,29,28,29,30,31,32,31,36,39,38,39,38,39,41,39,30,27,29,32,33,29,29,31,32,
+       193,142,142,143,143,144,145,145,145,146,145,146,148,148,148,149,148,150,153,155,154,161,164,163,163,163,163,163,163,162,162,162,161,161,161,161,161,161,160,161,161,161,162,163,163,160,160,160,163,162,158,83,54,74,78,75,74,75,77,78,82,86,89,92,93,92,85,74,73,79,63,53,57,54,54,51,40,54,71,83,90,92,89,87,84,80,75,69,67,65,40,29,32,32,37,39,35,30,29,31,32,30,28,29,34,34,35,36,38,42,39,36,34,32,35,35,31,29,28,29,32,34,36,36,34,34,37,39,
+       192,140,141,143,143,143,144,144,144,145,145,145,147,147,148,149,150,151,153,154,155,162,164,163,163,162,162,161,161,161,161,161,161,161,162,162,162,162,161,161,161,161,161,162,162,161,161,161,162,162,164,104,59,71,74,72,70,72,74,78,83,86,88,91,93,91,84,78,69,60,32,19,39,47,48,33,16,43,68,86,87,88,88,85,83,81,75,70,66,63,45,41,42,39,36,30,29,31,33,33,32,35,38,41,42,41,42,42,41,35,27,25,28,30,29,29,29,31,35,33,32,31,34,39,39,37,35,34,
+       191,139,140,142,142,143,144,144,144,145,145,145,146,147,149,149,150,151,152,153,156,163,163,163,163,162,161,161,161,161,160,160,161,161,161,162,162,161,161,161,160,160,161,161,161,161,161,161,161,161,164,125,76,74,73,71,69,70,72,76,81,84,87,88,88,88,82,84,65,51,30,13,35,46,50,38,33,48,72,92,88,87,88,85,81,80,76,72,68,65,37,30,31,28,28,36,36,34,36,39,41,40,35,32,33,35,33,32,28,30,33,33,35,37,36,36,37,36,38,36,35,37,40,40,31,26,27,28,
+       191,140,140,142,142,143,143,144,145,145,145,145,146,147,149,149,150,151,152,154,155,162,164,163,163,163,162,162,162,161,160,160,161,161,160,160,160,161,161,161,160,161,161,161,161,161,161,161,161,160,162,143,82,80,71,69,68,69,70,74,78,81,85,85,84,84,81,87,69,47,43,38,48,48,53,55,57,58,85,98,94,90,88,84,80,78,76,73,68,63,38,32,36,41,42,42,42,41,42,39,30,28,30,31,31,29,28,31,35,36,35,35,36,39,39,38,36,32,34,35,34,32,30,26,31,33,33,34,
+       191,139,140,143,143,143,143,144,144,144,144,144,145,146,147,148,150,151,153,154,156,161,164,163,163,163,162,161,161,160,159,160,160,160,160,160,160,160,160,160,160,160,160,160,160,160,160,160,160,160,159,158,87,67,71,67,68,68,69,73,76,78,80,82,83,83,86,91,84,61,57,63,60,57,56,64,74,83,95,99,97,92,85,80,79,78,77,74,67,62,42,40,38,34,31,31,33,34,29,27,30,36,38,37,36,36,39,41,40,38,39,41,41,36,29,26,28,29,28,28,28,29,30,32,35,34,34,34,
+       190,137,140,142,142,142,143,144,143,144,144,144,145,146,146,147,149,151,153,153,156,161,162,162,162,162,161,161,161,160,160,161,160,160,161,160,159,158,158,159,159,159,159,159,159,158,157,159,160,163,165,159,116,72,72,67,66,68,69,71,73,75,77,79,81,84,91,94,91,84,75,75,74,70,70,80,87,90,94,97,96,89,85,79,77,78,76,74,68,61,42,30,28,30,33,32,30,30,32,36,37,37,35,38,40,37,34,33,32,33,35,32,28,26,32,33,35,35,36,37,38,39,39,38,39,39,40,40,
+       190,136,140,140,140,141,142,143,143,144,144,144,144,146,146,147,148,150,152,153,156,161,161,162,162,160,161,161,161,161,161,161,160,160,161,161,160,157,158,159,159,159,159,160,162,164,166,161,143,116,83,47,43,89,74,67,64,67,70,70,71,75,77,78,81,86,92,94,90,85,82,83,78,76,85,94,95,93,91,92,92,89,84,79,77,77,74,71,67,57,33,35,37,36,36,35,38,41,40,42,43,43,42,39,32,27,29,32,31,31,29,29,32,36,36,35,34,37,39,40,39,34,28,27,28,30,31,29,
+       190,136,139,139,139,140,141,142,142,143,144,143,144,145,146,147,148,149,150,152,156,161,161,162,162,160,160,160,160,160,159,159,158,158,159,159,158,157,156,157,161,161,162,155,141,114,84,52,30,23,22,26,30,61,74,68,65,67,70,70,69,72,75,76,80,86,90,90,83,85,90,88,86,91,97,101,100,100,96,86,83,87,84,80,78,76,71,70,67,55,38,39,35,34,36,39,38,31,30,30,33,31,28,27,30,34,36,37,35,37,39,42,40,39,39,41,42,41,37,31,27,31,32,32,34,32,29,31,
+       190,136,139,139,139,140,141,142,142,142,143,144,145,146,146,147,147,149,150,152,156,162,161,163,160,161,160,159,159,158,159,158,157,157,157,156,161,164,164,154,133,100,70,44,32,24,25,30,29,25,25,25,25,33,66,70,66,66,67,69,68,70,71,74,80,84,85,80,80,88,85,71,73,87,88,71,64,76,94,91,79,81,82,79,77,75,70,70,65,53,38,40,39,39,35,28,29,31,32,31,28,29,33,37,40,38,36,39,39,41,37,28,26,27,28,29,31,29,26,30,35,37,36,35,34,34,34,37,
+       189,134,138,138,138,140,140,141,141,141,142,142,144,146,146,147,148,149,150,151,156,161,160,160,159,158,157,157,156,157,158,157,157,159,160,153,129,99,70,43,27,24,24,25,26,25,27,28,27,26,28,33,36,36,57,71,65,66,67,67,68,69,71,74,78,79,79,78,85,73,51,48,51,54,55,49,49,48,55,76,75,76,79,75,76,74,68,67,64,47,27,28,29,28,28,33,35,37,36,35,36,39,40,39,38,37,37,38,35,30,27,31,34,34,34,33,31,33,35,34,34,36,36,37,40,38,35,33,
+       188,133,140,140,140,141,141,141,141,141,142,142,143,143,145,146,147,149,151,150,158,162,159,158,158,156,156,156,157,159,160,150,126,101,71,45,28,22,24,25,27,29,26,24,26,27,29,34,36,39,39,35,33,26,48,71,66,67,67,65,67,70,72,73,76,75,77,78,70,44,40,46,45,44,43,44,43,39,36,46,57,69,75,76,76,69,64,62,60,45,32,31,35,36,37,39,38,38,40,41,37,30,27,27,29,29,28,28,30,32,35,36,36,35,35,36,38,38,39,39,38,40,40,40,35,29,27,28,
+       188,133,140,140,140,138,139,139,139,139,140,142,142,142,143,145,147,148,149,151,158,158,155,156,160,161,162,157,138,109,76,50,33,28,25,27,29,25,25,25,27,27,29,34,36,35,32,33,36,36,32,34,37,37,51,68,65,67,68,65,64,67,69,72,72,73,68,60,43,32,34,38,39,36,36,40,49,47,39,36,44,61,69,75,76,66,61,59,55,42,35,36,38,35,30,33,34,34,30,29,27,29,34,34,36,36,35,35,36,36,37,36,37,37,40,38,33,31,29,30,32,32,32,28,27,32,37,34,
+       188,133,138,138,138,138,138,139,141,141,141,142,142,143,143,143,146,148,149,150,160,166,161,149,135,111,83,56,38,30,29,29,28,25,25,26,26,31,35,35,33,33,39,39,32,31,35,34,35,39,42,43,43,43,45,64,64,64,66,65,63,64,67,72,71,70,56,39,31,36,53,64,70,65,62,69,82,80,72,65,65,60,64,73,73,63,60,57,52,41,37,33,27,25,26,30,30,29,29,29,32,35,36,35,36,39,40,39,39,38,40,39,37,37,30,27,29,30,32,32,31,30,32,33,35,36,37,36,
+       187,130,137,136,136,137,136,138,139,139,138,140,141,141,141,141,144,150,156,148,133,105,75,52,36,30,28,26,28,29,28,28,29,30,31,33,35,37,34,30,30,29,31,34,37,39,41,42,41,38,37,38,41,39,39,60,61,57,59,62,61,60,64,70,69,66,57,51,55,59,67,72,74,72,70,69,70,68,71,71,65,61,63,68,65,58,56,52,47,30,26,29,33,34,35,35,39,37,38,37,36,36,38,39,39,35,30,28,28,29,29,27,27,27,30,36,36,35,35,37,37,38,39,38,38,37,39,38,
+       187,129,134,136,137,137,135,136,138,138,137,138,138,140,141,143,147,134,95,53,34,30,29,31,32,30,27,27,32,32,32,36,37,33,29,29,30,30,32,37,38,40,42,40,40,43,40,35,33,37,38,38,40,37,29,49,60,52,54,59,61,60,60,65,68,63,59,62,64,57,60,60,59,63,65,62,57,54,56,52,53,58,60,61,57,55,50,49,42,32,34,35,36,35,34,36,39,36,35,34,36,37,36,33,27,27,33,35,33,35,34,33,35,38,38,37,36,35,37,39,39,37,33,29,28,28,30,32,
+       187,130,135,135,137,137,136,136,136,136,136,134,135,138,140,132,87,45,33,32,31,30,29,30,31,31,33,31,29,29,26,30,30,34,35,35,37,38,39,40,41,38,32,29,29,31,33,35,41,44,42,40,36,24,13,29,54,50,47,53,57,57,57,58,62,61,61,62,55,44,44,49,48,49,47,50,47,45,40,42,50,56,55,54,52,51,48,46,40,37,37,37,36,36,37,34,30,26,27,29,27,26,25,27,33,37,38,36,35,36,37,41,40,36,36,37,39,37,36,35,30,26,30,34,32,34,32,29,
+       187,128,134,134,135,136,136,136,136,136,135,132,131,134,119,54,33,31,33,31,29,30,32,32,30,31,30,28,31,33,35,36,38,39,38,37,31,31,31,30,29,30,36,36,35,39,42,40,39,39,41,39,25,14,11,22,44,48,47,47,50,52,52,53,57,59,62,63,56,45,38,38,36,33,32,34,38,39,41,46,51,54,52,49,48,47,46,45,32,28,27,28,30,28,27,30,32,33,35,37,35,35,36,37,37,37,39,37,37,39,36,31,27,26,28,29,28,28,27,27,32,36,35,35,34,37,38,36,
+       187,128,134,133,133,134,133,133,134,133,132,129,129,126,64,33,34,32,31,32,34,32,30,30,30,32,35,38,39,39,34,31,31,31,30,30,35,36,38,39,37,37,41,41,39,36,36,38,35,35,40,34,17,18,31,37,37,43,48,45,44,48,50,50,50,54,57,58,56,51,45,41,39,38,39,42,46,48,48,49,50,50,50,47,46,46,44,39,28,33,33,29,28,30,32,38,39,34,35,35,37,38,36,32,31,33,37,37,33,30,27,33,36,37,37,37,32,34,34,35,35,34,34,36,35,38,39,37,
+       187,129,133,132,131,131,132,131,131,131,130,127,127,94,37,36,38,32,29,28,31,31,34,38,40,37,34,34,32,33,32,36,36,33,36,39,40,39,40,37,36,37,40,37,26,22,30,38,37,37,38,27,27,34,33,27,31,37,45,45,41,44,47,46,47,49,53,53,52,50,49,48,50,55,58,59,59,57,56,56,51,50,49,46,45,43,40,27,25,32,35,35,36,36,36,38,38,37,36,37,36,28,28,28,29,30,29,28,29,31,34,37,38,36,35,36,37,37,37,37,38,38,37,38,38,36,29,26,
+       186,127,131,130,130,129,130,129,128,128,125,121,119,60,37,38,32,31,35,34,36,37,39,38,35,33,35,36,35,36,38,39,41,37,36,40,41,36,32,32,33,29,23,16,14,29,37,38,36,39,39,34,33,29,25,28,28,34,37,42,39,40,42,42,44,47,48,51,52,52,57,61,63,68,69,69,69,71,69,63,55,52,51,45,42,40,30,15,17,21,26,32,33,28,25,26,27,27,28,26,27,32,37,36,36,36,36,36,37,37,36,36,36,38,38,38,35,31,29,29,31,34,33,31,27,27,31,34,
+       186,127,130,129,129,128,126,126,125,124,121,117,105,45,33,34,37,41,42,36,32,31,32,33,38,41,41,41,38,38,39,37,32,31,31,30,31,33,38,40,37,21,13,20,32,39,33,31,35,40,38,30,24,29,38,38,32,32,34,34,37,36,37,37,41,46,46,46,52,60,67,71,69,68,72,70,67,74,73,67,61,55,50,42,38,34,18,15,15,15,15,18,21,26,32,32,33,31,32,31,33,34,36,35,34,37,38,36,35,36,38,39,39,39,33,28,26,27,31,32,31,29,28,30,31,33,35,35,
+       186,126,129,128,128,127,124,125,125,123,119,115,86,39,38,34,35,37,35,37,38,40,41,42,42,41,36,32,30,29,28,30,33,34,38,41,40,39,42,41,26,22,22,31,37,35,24,24,33,36,27,27,37,40,39,38,33,29,32,29,31,32,31,33,35,40,46,46,52,64,70,74,72,68,64,60,65,72,72,69,62,53,45,37,33,20,10,15,18,17,15,18,18,23,29,35,37,35,34,35,35,34,34,35,36,35,29,25,25,25,27,26,27,28,29,34,34,34,36,35,34,34,36,37,36,36,36,34,
+       185,123,128,127,127,126,124,123,124,121,115,108,70,37,36,32,33,37,38,39,40,36,33,33,35,33,32,35,37,37,38,38,40,38,39,34,30,30,35,29,16,23,26,39,35,27,32,40,25,27,34,38,40,38,36,37,34,30,28,32,27,26,28,29,30,34,42,47,52,60,68,69,67,63,55,51,60,67,68,66,58,45,36,35,30,11,9,11,19,18,17,27,27,32,33,32,36,30,26,26,28,30,30,28,25,26,30,33,34,34,34,33,33,35,37,35,33,35,37,36,36,37,39,37,34,33,35,35,
+       184,123,128,127,126,125,123,123,122,118,112,99,56,33,35,37,39,39,39,38,35,34,35,36,40,41,39,38,38,38,30,29,30,30,30,31,36,40,34,18,17,15,33,36,25,28,35,43,24,26,37,40,38,37,40,41,39,36,29,30,32,24,25,26,26,27,34,42,49,55,59,61,61,59,56,52,56,61,61,58,52,39,32,35,35,10,9,11,18,17,19,21,20,24,31,33,31,28,27,28,27,27,25,26,29,34,35,35,33,36,35,35,38,37,39,38,39,39,38,38,40,35,27,27,28,27,28,27,
+       184,123,128,127,126,125,123,122,119,116,109,90,45,36,40,37,34,34,32,33,38,39,40,40,37,33,31,33,31,31,34,35,34,35,37,36,38,42,24,17,16,19,29,26,25,29,27,24,32,25,28,35,38,37,37,37,38,38,32,26,31,25,23,25,25,24,25,32,41,47,52,55,53,51,53,52,51,53,54,50,40,33,32,45,39,13,8,12,16,15,13,17,20,19,24,33,35,35,36,33,32,33,34,33,34,36,34,33,34,36,37,37,34,28,27,27,30,32,32,29,28,28,32,37,38,37,36,34,
+       184,121,128,126,126,125,124,121,118,114,106,84,43,37,35,36,37,39,39,37,35,37,37,34,34,34,33,35,33,37,36,38,36,34,37,40,41,39,18,11,11,21,22,27,28,24,27,21,28,34,21,23,27,34,35,31,30,32,35,28,25,28,21,23,25,24,23,26,32,37,41,44,44,43,45,46,44,44,45,39,31,30,38,50,38,16,13,11,15,14,16,21,26,26,26,31,39,36,35,34,33,34,34,32,31,32,35,36,34,34,31,28,28,31,31,33,32,29,30,30,32,37,38,37,36,36,35,34,
+       184,122,126,125,124,123,122,120,116,110,101,73,38,36,38,41,41,39,33,31,30,31,33,37,38,37,36,40,43,48,49,45,37,33,29,30,35,36,19,9,23,34,37,42,39,25,22,21,15,31,32,17,14,17,20,25,28,28,31,31,24,30,26,17,21,23,24,23,24,27,30,34,35,36,37,38,36,35,33,30,29,35,43,51,35,12,18,12,13,17,17,18,20,24,29,28,37,38,38,37,34,29,23,24,25,26,26,26,25,26,29,33,36,35,35,37,37,35,36,37,36,37,37,37,35,35,37,37,
+       184,120,123,124,123,122,121,119,115,109,97,66,39,37,34,34,34,33,34,38,38,38,40,41,42,36,34,39,44,55,58,55,53,53,46,40,32,31,28,27,35,41,45,44,44,37,22,22,17,17,32,26,19,21,21,19,21,24,28,32,28,24,29,18,14,19,22,22,23,23,24,26,28,31,30,31,30,29,28,30,35,41,45,53,35,10,18,20,10,12,13,15,17,22,29,27,30,35,38,37,30,25,28,31,32,34,34,31,31,33,35,34,33,34,35,35,37,37,37,37,36,39,38,37,34,36,33,27,
+       182,117,123,122,121,119,118,116,114,108,94,60,37,35,35,38,39,40,39,39,36,31,30,31,33,36,35,37,43,45,48,51,53,54,53,48,30,22,27,35,34,32,38,40,43,40,33,26,16,14,18,31,27,23,31,33,31,30,32,33,32,25,23,25,14,15,17,19,21,22,22,22,24,25,25,26,26,26,27,34,41,45,48,51,33,8,14,25,13,8,12,15,19,22,25,30,31,34,38,34,32,33,33,31,31,31,32,33,34,35,35,36,35,35,36,38,33,28,25,24,25,27,30,29,26,25,26,32,
+       182,118,122,121,120,119,118,115,113,106,89,50,35,38,39,38,35,35,36,31,32,36,35,35,46,49,41,41,46,48,46,46,48,50,52,51,44,25,15,18,27,37,30,37,41,38,30,23,17,13,13,20,30,20,19,28,30,29,28,28,32,31,21,26,23,16,15,17,18,20,21,23,24,26,25,25,27,28,33,38,41,46,48,51,35,10,11,24,19,6,11,16,21,21,21,26,34,37,36,33,32,32,33,32,32,34,33,29,27,27,27,31,33,30,31,26,27,31,33,31,32,35,33,32,32,32,34,35,
+       182,117,121,120,118,116,116,114,111,102,82,45,40,39,36,32,32,31,33,36,38,39,40,47,50,50,42,37,40,46,47,45,43,43,45,47,46,43,25,22,25,35,32,32,38,38,20,19,19,13,13,13,20,30,20,16,21,28,31,31,31,30,26,21,27,21,16,18,17,17,19,21,23,23,25,25,29,34,39,39,44,46,48,50,34,9,9,21,25,10,14,18,19,22,25,25,28,33,34,38,36,33,36,32,29,27,23,24,26,29,27,27,26,26,29,33,35,35,35,34,34,38,36,36,36,36,37,37,
+       182,116,120,119,118,115,115,113,111,100,77,42,36,35,34,36,36,38,36,36,36,39,44,48,48,47,39,39,44,43,45,45,44,43,43,43,44,46,41,32,24,31,36,32,35,37,27,17,19,14,14,13,12,21,29,21,14,16,24,31,29,29,30,22,24,26,22,23,20,19,20,21,23,25,27,29,35,38,42,43,45,45,49,49,33,8,11,19,29,16,14,18,23,23,24,29,26,30,31,37,40,38,39,35,27,26,29,30,33,33,32,32,32,35,36,36,36,33,33,34,35,37,37,37,34,34,34,33,
+       181,114,119,118,117,114,113,111,110,97,70,37,36,37,37,39,37,32,30,33,31,37,48,48,45,43,40,39,45,44,43,44,44,44,42,41,41,40,39,32,22,23,37,29,31,34,31,17,16,19,15,16,12,11,16,26,23,12,11,21,26,32,32,25,19,25,26,23,24,24,25,26,26,29,33,36,39,41,44,45,45,46,49,46,31,9,11,17,30,21,13,16,21,22,24,29,28,30,32,34,37,39,41,39,36,35,35,33,32,33,34,33,34,35,35,35,37,36,35,35,36,36,32,26,25,28,31,28,
+       181,113,119,118,116,114,112,111,108,95,64,39,36,35,34,32,33,35,37,36,40,47,50,47,45,44,39,37,43,43,42,42,41,42,42,40,40,40,38,33,30,30,42,30,39,38,29,16,12,16,17,16,15,12,11,13,19,20,15,11,17,26,29,31,23,22,26,24,24,26,29,30,30,33,37,40,41,44,45,45,46,47,48,45,31,9,12,17,27,25,12,16,20,21,25,28,28,30,32,33,36,37,38,38,36,33,33,32,33,33,33,34,32,25,26,25,27,27,27,26,27,26,27,31,34,36,36,34,
+       181,112,118,117,115,114,113,110,106,92,58,36,33,35,36,37,38,38,37,35,41,46,47,46,44,43,41,40,41,41,41,41,40,42,43,41,41,41,39,35,31,37,37,30,41,38,30,21,14,11,16,19,15,14,10,10,10,16,21,14,9,16,23,27,26,17,23,29,26,26,29,31,33,37,40,42,43,44,44,46,47,47,45,43,34,13,14,18,25,28,12,15,19,18,20,26,29,29,30,32,35,37,37,37,36,34,34,32,35,35,31,26,26,28,32,34,35,33,29,29,29,32,35,35,35,35,35,35,
+       181,112,116,115,114,114,113,108,103,87,48,34,38,38,38,34,33,32,32,34,36,35,40,42,42,41,36,40,39,39,39,41,41,40,41,40,40,40,39,35,33,37,30,35,39,35,31,23,16,11,11,14,17,15,13,10,12,13,14,19,17,12,17,24,26,20,19,28,29,28,29,33,35,38,41,42,43,44,44,44,44,46,44,41,32,13,11,18,23,31,13,14,18,20,24,26,28,28,28,29,32,34,35,34,34,32,32,33,35,21,20,31,34,34,34,35,34,34,35,34,35,37,36,35,36,35,36,35,
+       180,112,116,115,113,113,110,106,100,82,47,38,36,36,35,36,34,32,34,40,33,23,27,35,37,37,37,39,40,39,39,41,41,38,37,38,38,39,38,36,35,33,29,39,39,36,33,27,22,17,11,10,11,10,13,12,11,12,12,16,23,19,14,19,26,25,16,20,28,30,31,33,35,38,40,41,43,42,43,43,43,46,44,41,33,15,10,19,23,31,17,14,15,16,20,24,25,26,29,32,35,37,35,33,28,26,30,32,25,16,30,35,33,33,34,34,32,35,36,36,37,38,36,36,37,36,37,36,
+       179,111,116,115,113,111,109,106,99,76,40,34,34,36,38,37,35,39,39,39,40,28,19,29,34,33,36,38,41,39,39,40,40,37,35,36,37,38,37,36,37,27,36,37,37,36,31,30,26,23,19,11,10,11,13,12,14,12,13,17,20,25,21,14,15,23,20,16,25,32,32,33,35,37,39,42,42,41,42,44,44,46,44,41,32,17,12,22,24,30,20,8,11,14,20,24,25,27,28,30,36,34,30,22,24,27,27,29,22,28,34,34,33,33,33,33,36,33,27,25,25,25,26,30,31,30,26,26,
+       179,111,115,114,113,111,109,105,97,69,36,35,37,40,40,39,40,41,40,39,41,35,18,21,30,29,32,37,40,40,39,39,39,39,35,35,36,36,36,35,31,30,38,34,36,34,33,32,28,23,22,18,11,12,13,14,14,14,13,16,16,27,26,23,12,14,21,18,21,29,33,34,35,38,39,41,41,39,42,44,44,44,44,41,31,19,13,25,27,31,21,6,12,16,21,22,21,25,27,26,30,26,22,21,29,25,28,29,30,28,25,27,30,31,30,27,26,27,28,30,33,34,31,28,28,30,31,33,
+       179,110,115,113,112,111,110,105,94,63,36,36,34,35,34,38,41,43,43,41,41,37,21,16,25,27,29,35,39,40,39,39,38,38,35,33,35,34,38,31,26,33,33,30,33,34,35,34,29,26,21,19,18,12,7,13,16,18,16,13,17,23,29,25,23,12,19,20,17,26,33,34,34,36,38,40,39,40,42,42,43,44,44,41,31,21,17,26,15,18,24,10,14,15,19,23,26,26,27,25,28,30,29,25,23,24,27,27,29,29,29,29,27,25,24,27,31,33,34,33,33,34,34,34,34,35,36,36,
+       180,110,116,112,110,110,109,103,91,57,35,35,35,37,42,42,40,42,43,43,42,36,25,16,22,27,27,32,36,39,40,40,38,36,36,32,32,35,38,28,29,30,30,30,34,36,35,34,31,29,25,21,21,17,13,8,10,11,13,13,15,19,28,23,30,21,16,22,16,21,33,35,33,34,36,37,37,39,41,41,43,45,46,40,30,23,22,26,6,6,16,10,14,16,20,22,24,26,28,26,26,28,26,21,24,26,26,28,30,32,32,31,32,32,33,32,33,34,33,32,34,35,35,37,37,37,35,33,
+       180,110,114,112,110,109,107,101,87,48,35,36,39,43,42,41,40,41,41,42,40,37,30,16,18,24,22,26,31,38,40,40,37,36,38,34,31,36,31,29,29,29,29,33,34,35,36,36,32,31,28,23,25,23,18,12,10,12,14,15,12,15,23,24,25,29,17,17,15,16,28,34,34,35,35,36,35,36,40,41,43,45,46,40,31,24,25,22,7,8,15,9,15,15,20,22,23,23,25,28,26,25,21,20,24,26,27,25,27,30,31,31,32,28,31,35,35,33,33,34,35,35,34,33,26,25,27,28,
+       179,108,112,112,111,110,107,99,81,46,36,39,43,41,41,41,39,40,41,43,41,38,33,21,17,23,21,23,28,36,39,39,39,37,38,36,35,33,29,32,62,105,82,41,32,31,30,29,27,30,28,24,25,27,22,17,13,15,19,19,13,14,21,29,21,29,25,14,20,15,20,32,33,34,34,36,35,36,40,41,43,45,45,39,31,22,24,14,7,13,14,11,16,16,20,22,23,23,24,26,25,24,24,22,23,26,27,25,28,29,31,32,34,32,31,30,26,23,26,27,28,29,26,27,30,34,36,36,
+       179,107,111,110,109,108,104,96,75,39,35,43,42,39,41,41,40,41,41,43,41,38,34,27,16,20,20,22,26,31,38,39,40,36,37,37,34,28,36,29,114,86,80,72,32,32,27,23,23,27,29,25,25,28,26,20,18,12,14,16,14,14,17,30,19,23,30,18,18,14,12,26,33,33,33,34,34,35,40,42,43,45,45,40,28,21,23,10,7,13,10,10,11,14,18,23,23,23,24,25,22,21,24,23,25,27,27,26,26,29,30,32,32,33,34,32,30,29,29,29,31,29,32,34,35,36,37,37,
+       178,108,112,111,107,106,106,94,67,39,44,44,42,40,42,43,43,42,39,40,39,39,37,32,18,18,20,19,22,27,35,39,39,36,39,37,28,30,35,31,68,128,125,61,35,40,38,33,27,24,27,25,25,28,28,22,19,14,9,8,12,15,12,25,25,19,28,28,12,14,15,20,31,34,33,33,33,34,39,43,46,45,46,39,24,24,21,7,8,11,7,9,11,12,19,26,26,21,21,22,23,22,25,23,26,28,27,25,24,27,27,32,32,32,33,32,31,31,31,33,35,36,36,36,36,35,36,35,
+       178,108,112,109,108,105,84,64,52,44,45,47,45,44,43,43,43,42,40,38,37,38,36,33,24,16,20,18,20,23,28,32,33,33,32,33,32,30,29,31,32,43,46,33,36,37,38,36,31,27,26,24,25,28,27,25,19,19,20,11,10,11,10,14,27,18,20,27,20,18,16,15,28,36,33,33,36,38,39,41,45,46,39,29,25,26,15,6,12,11,6,9,12,14,18,24,28,23,20,22,27,26,24,25,27,28,27,25,26,27,25,27,31,31,33,33,30,30,31,29,33,35,36,36,36,36,35,36,
+       179,106,112,112,102,67,47,45,46,46,45,45,44,44,43,42,43,41,41,39,37,37,33,33,26,15,19,21,32,26,31,34,33,26,16,19,25,28,29,29,32,32,33,32,31,31,32,32,33,29,26,22,26,29,29,25,19,20,21,9,7,10,12,12,22,20,16,19,22,21,19,12,20,37,36,35,38,39,40,43,44,33,22,21,23,26,11,7,11,9,6,9,13,14,18,24,28,27,22,23,26,25,24,26,26,27,26,23,26,29,29,29,31,34,33,33,33,32,32,31,31,29,26,27,29,28,28,27,
+       177,105,112,99,61,45,46,44,47,47,45,45,44,45,43,42,43,41,41,41,40,37,30,30,31,17,14,20,27,27,30,35,37,29,17,10,10,12,14,17,19,23,29,31,29,31,34,33,32,30,26,31,33,27,29,24,20,26,15,6,6,8,11,12,16,27,19,20,27,33,27,14,11,31,42,40,42,43,44,40,30,24,24,24,21,18,8,8,10,10,9,8,12,15,18,22,26,29,29,26,24,21,27,24,23,24,25,24,24,28,30,30,32,35,33,32,33,32,32,31,31,32,31,32,30,28,30,30,
+       178,103,84,55,43,46,46,45,45,45,44,44,43,43,42,42,43,41,40,40,39,36,32,27,29,21,16,17,21,26,29,32,32,28,25,17,11,10,9,8,9,11,13,16,17,17,21,24,28,31,34,45,41,25,27,23,25,24,9,4,6,7,13,15,12,21,27,17,20,27,30,19,12,25,45,46,47,43,31,21,18,19,21,21,14,8,5,8,13,15,16,11,10,12,11,19,24,23,30,27,25,25,27,20,21,23,24,24,24,27,29,32,33,34,34,32,33,31,31,32,31,31,33,35,34,33,35,37,
+       172,59,45,45,46,46,44,44,44,44,43,43,43,43,40,41,42,40,39,37,37,36,36,34,30,24,18,19,17,21,23,28,31,30,30,26,20,14,11,8,9,9,8,9,9,7,8,8,13,24,35,35,31,30,28,25,28,18,6,4,8,13,11,9,12,12,24,26,18,20,27,27,17,18,42,53,46,28,24,23,22,21,22,15,7,7,6,10,13,17,23,18,11,10,13,15,23,21,22,26,25,28,22,18,20,23,24,24,24,26,29,32,32,33,33,30,32,32,32,32,33,33,33,34,31,36,36,35,
+       165,40,45,45,44,42,42,42,41,42,41,41,44,43,41,41,42,40,39,37,35,35,34,34,36,27,18,18,13,17,23,20,17,23,28,30,28,19,13,11,9,10,10,11,8,8,8,8,8,12,23,34,34,29,27,31,23,13,6,7,7,15,14,7,9,12,13,24,26,18,25,31,26,15,32,47,27,24,26,27,26,27,22,9,6,7,7,9,11,17,25,23,17,16,21,18,17,22,22,23,26,25,18,19,19,22,24,24,25,24,28,32,31,32,33,32,32,33,32,30,31,33,33,33,34,33,26,26,
+       166,39,43,43,43,43,40,40,41,40,40,41,42,40,40,39,41,39,39,38,35,34,32,32,35,30,19,17,20,25,35,33,27,20,18,21,27,26,18,14,11,10,11,11,8,8,8,8,7,8,11,21,32,32,32,28,14,8,9,7,10,15,18,17,12,10,9,12,23,26,18,23,34,23,19,24,22,23,23,24,25,25,17,6,7,7,7,9,12,21,25,25,28,29,29,27,20,22,26,25,27,20,16,19,18,21,24,24,26,24,26,30,33,32,32,34,32,33,32,31,32,32,32,31,34,33,30,33,
+       165,35,41,42,42,43,42,43,42,41,40,40,38,37,38,39,39,38,38,38,35,33,31,31,32,30,27,18,21,24,30,36,36,37,34,28,23,21,24,21,16,9,9,11,10,10,8,8,6,6,9,11,19,28,24,17,10,8,12,13,12,16,19,20,16,11,12,11,11,23,26,16,26,31,18,22,26,26,28,29,26,24,13,6,8,8,8,13,23,27,27,28,30,29,30,30,26,25,26,29,24,16,17,18,18,21,24,25,25,23,24,28,33,32,31,32,33,33,32,32,31,34,32,31,32,33,32,33,
+       163,29,33,32,33,35,39,42,42,41,40,40,39,38,38,38,38,38,37,36,34,32,29,29,30,29,30,21,18,21,29,36,36,37,38,36,32,29,22,22,21,13,11,10,10,10,10,9,9,6,8,10,12,14,13,12,9,13,14,16,13,14,16,17,19,17,14,11,10,10,20,23,18,29,25,19,26,21,19,16,11,10,9,8,10,10,9,21,28,28,29,29,29,28,29,30,27,29,31,28,17,16,19,19,20,22,24,25,25,23,25,25,30,32,31,31,33,32,31,31,33,35,32,33,29,30,33,31,
+       165,38,43,42,35,28,26,34,37,35,37,40,40,39,39,38,38,37,33,32,32,31,29,28,29,28,27,24,14,17,26,35,36,37,37,34,34,35,33,19,17,19,16,12,13,11,10,10,12,8,9,10,13,12,11,9,12,15,14,14,15,12,14,17,18,17,14,12,13,12,11,21,19,19,28,23,25,21,21,15,10,9,11,11,12,11,15,25,29,30,29,29,27,26,26,27,33,35,25,16,15,18,18,19,20,22,24,24,25,24,24,24,28,32,33,32,32,31,31,30,32,32,32,32,30,30,31,31,
+       166,35,38,40,39,39,31,24,26,33,37,39,39,38,39,38,39,38,35,31,32,31,31,29,27,28,28,27,19,15,25,33,36,38,37,34,33,34,37,27,22,17,17,15,12,13,10,11,12,12,10,10,11,12,12,12,14,16,16,16,17,14,12,16,18,21,17,13,15,14,11,15,21,18,25,27,20,14,16,18,12,8,12,13,13,18,28,33,33,31,31,32,31,30,32,32,29,18,14,13,13,17,19,19,20,22,23,24,25,24,24,26,27,30,30,31,31,31,31,31,33,33,32,32,31,31,29,31,
+       165,29,34,32,33,35,38,37,25,22,31,36,37,38,37,36,38,36,32,31,34,32,30,28,27,28,29,26,24,16,23,30,35,37,36,35,34,35,38,30,32,28,19,16,16,12,11,12,11,11,9,10,14,13,11,13,16,19,17,17,20,19,12,13,14,18,22,21,16,14,13,13,19,21,20,28,24,18,17,19,14,10,15,14,20,32,30,21,16,14,16,19,19,18,17,14,10,7,11,14,15,18,21,21,20,22,23,23,23,22,22,26,26,28,28,30,31,30,31,31,34,33,32,32,30,30,28,28,
+       166,30,31,31,33,33,33,35,35,26,20,27,35,39,38,36,35,29,25,27,28,29,30,28,27,28,28,26,25,19,18,26,32,35,36,35,34,34,38,31,33,35,26,16,16,15,15,16,13,14,11,10,12,14,11,13,17,19,19,19,20,21,15,11,12,13,19,21,18,17,16,14,16,22,17,25,28,21,20,20,16,18,20,20,33,23,10,9,8,9,10,9,9,8,7,11,7,6,11,15,14,19,21,20,20,22,22,25,23,21,20,23,26,28,28,29,29,29,33,34,33,32,31,33,31,29,29,27,
+       170,43,33,30,31,33,31,30,32,32,19,16,26,36,37,35,34,26,23,26,23,22,24,25,25,26,26,27,24,22,16,24,28,32,34,34,33,33,36,34,34,36,29,18,15,17,17,15,13,16,15,10,11,12,11,14,16,17,18,19,19,19,18,10,13,11,12,15,17,18,18,15,14,19,19,19,28,23,19,17,17,19,21,31,22,10,9,10,9,9,12,9,9,10,11,12,7,5,8,15,15,19,20,20,20,22,23,23,21,19,21,40,30,25,27,29,28,29,34,35,34,32,30,32,31,30,31,26,
+       176,63,46,35,31,32,30,30,30,34,30,19,17,23,33,36,36,32,25,23,19,19,20,23,25,25,25,26,23,22,16,23,27,30,35,36,35,33,32,36,34,36,37,27,25,24,17,15,16,16,13,12,11,11,11,15,16,16,18,19,16,17,20,14,10,9,8,8,9,11,16,16,13,14,21,17,25,26,21,20,20,21,27,27,8,10,7,6,9,9,9,11,10,13,14,12,7,5,7,14,19,19,18,18,19,20,23,22,16,15,61,92,55,32,27,28,29,30,32,31,32,33,31,32,32,31,29,25,
+       133,64,54,38,30,31,30,29,27,29,29,25,18,19,25,31,33,31,26,24,21,20,21,21,22,24,23,23,23,23,19,22,26,27,31,32,32,31,29,29,29,30,34,29,27,27,20,17,18,21,17,16,15,15,15,19,19,19,19,19,18,18,19,20,15,16,16,15,16,16,15,17,17,18,20,21,23,25,23,24,24,26,31,20,14,14,12,14,16,15,15,16,16,18,18,17,14,12,13,16,22,21,23,22,21,20,22,19,20,22,67,74,38,42,26,26,28,29,28,28,30,31,31,30,30,28,28,27};\r
+\r
+\r
+int result[16384] = {\r
+50175,38900,38903,39673,41726,41983,41725,41726,41721,41725,41462,41457,41209,40694,40695,40945,40447,40436,40447,40440,40445,40177,40440,40447,40440,40696,41723,42234,42493,42233,42492,42743,42495,42746,42748,42492,43008,43257,43264,43260,43006,43004,43521,43520,44033,44030,44032,44031,44031,44030,44290,44288,44289,44287,44290,44285,44291,44544,44805,44806,44801,45056,45061,45062,44799,44804,44805,44545,45055,44797,45057,45056,45061,45058,45064,45070,45058,45317,45315,45323,45055,45061,45056,45058,45055,45066,45059,45053,45062,45064,45059,44807,44801,44805,44803,44808,44799,44806,44801,44288,44287,44295,44288,44291,44288,44035,43772,43520,43519,43780,43517,43264,43007,43011,43007,42751,42752,42752,43006,42751,42747,42751,42745,42748,42748,43264,43516,43514,
+50432,39157,38904,39678,41732,41982,41719,41721,41718,41726,41467,41466,41217,40702,40707,40957,40459,40446,40451,40440,40441,40686,40433,40438,40429,40682,41713,42225,42487,42225,42487,42738,42487,42736,42735,42477,42995,42989,43253,43250,43510,43505,43511,43512,44025,44021,44026,44025,44030,44051,44345,44359,44353,44339,44353,44349,44353,44604,44863,44863,44856,45109,44855,45111,44848,44851,44851,44592,45104,44844,45105,45098,45098,45098,45111,45122,45102,45361,45366,45375,45104,45110,45108,45109,45104,45113,45095,45095,45113,45121,45113,44825,44817,44822,44819,44825,44816,44822,44814,44299,44298,44305,44296,44300,44040,44045,43783,43787,43529,43792,43533,43283,43032,43040,43039,42785,42788,42787,43040,42786,42781,42785,42780,42782,42782,43040,42777,42777,
+50437,39172,38916,40203,42007,41754,41496,41496,41491,41493,41480,41470,41226,40711,40970,40707,40473,40469,40481,40731,40736,40469,40733,40740,40733,40735,41763,42531,42534,42531,42535,42785,42791,42787,42791,42791,43308,43300,43306,43302,43303,43558,43819,43817,43820,44074,44074,44071,44073,44073,44338,44348,44351,44349,44352,44339,44335,44585,44851,44853,44848,44590,44594,44591,44840,44844,44846,44844,44841,44835,45098,45100,45103,45096,45102,45109,45099,45358,45357,45366,45354,45364,45366,45122,45116,45382,45120,45118,45126,45126,44868,44868,44863,44867,44862,44869,44858,44868,44605,44093,44094,44358,44350,44097,44090,44092,43830,43835,43837,43326,43321,43322,43064,43067,43322,43064,43065,42808,42549,42554,42550,42555,42549,42550,42552,42810,42550,42547,
+50434,39151,38898,40696,41981,41726,41469,41469,41465,41470,41205,40942,40959,40702,40961,40699,40459,40447,40459,40709,40712,40702,40963,40709,40705,40703,41731,42497,42500,42496,42500,42752,42759,42754,42755,42754,43268,43259,43269,43268,43271,43525,43790,43794,43796,44049,44052,44050,44052,44073,44341,44337,44339,44333,44329,44314,44313,44563,44824,44822,44821,44568,44576,44573,44821,44827,44828,44820,44816,44813,45076,45073,45067,45055,45065,45080,45072,45332,45333,45348,45339,45341,45330,45080,45082,45352,45090,45077,45091,45111,44843,44805,44796,44801,44798,44805,44794,44801,44538,44026,44024,44289,44284,44036,44038,43792,43535,43541,43543,43290,43280,43284,43031,43033,43288,42773,42774,42772,42514,42520,42515,42520,42516,42518,42519,42778,42517,42515,
+50692,38913,38916,40971,41743,41487,41230,41230,41227,41235,40976,40974,40985,40725,40727,40719,40734,40722,40733,40727,40731,40720,40726,40732,40724,40724,41754,42523,42525,42777,42781,42774,43039,43034,43037,43037,43041,43034,43555,43553,43556,43809,43812,44071,44068,44065,44070,44071,44327,44325,44338,44341,44346,44348,44608,44604,44608,44861,44866,44868,44860,44853,44857,44861,44855,45116,45112,44852,44848,44844,45098,45091,45097,45099,45108,45113,45100,45362,45361,45371,45098,45108,45106,45116,45119,45133,45125,45118,45122,45105,45080,44825,45078,45083,44563,44568,44558,44564,44302,44303,44301,44309,44046,44050,43789,44050,43788,43534,43277,43024,43016,43019,42764,42767,43020,42507,42252,42507,42506,42510,42504,42508,41990,42248,42249,42507,42502,42500,
+50425,38875,38879,41191,41706,41450,41193,41195,41192,40945,40941,40936,40950,40687,40683,40675,40690,40677,40684,40677,40680,40671,40678,40682,40675,40672,41697,42208,42465,42719,42723,42719,42986,42985,42992,42995,42998,42733,43508,43761,43762,43756,43763,44020,44017,44014,44018,44017,44278,44555,44336,44347,44348,44340,44589,44577,44580,44828,44832,44836,44836,44840,44849,44858,44855,45117,45112,44854,44845,44839,45089,45077,45077,45072,45090,45100,45086,45343,45341,45354,45080,44831,45083,45085,45084,45101,45097,45090,45090,45087,45038,44745,45248,44996,44480,44235,44233,44245,44243,44250,44249,44256,43991,43993,43731,43737,43729,43475,43217,42966,42963,42969,42716,42720,42719,42461,42465,42465,42463,42466,42202,42206,41689,42203,42203,42460,42454,42452,
+50438,38924,38670,41240,41499,41500,41242,40985,40980,40984,40975,41230,41244,40992,40741,40476,40748,40737,40748,40485,40490,40480,40744,40748,40740,40998,42029,42540,42800,42795,42799,42794,43058,43055,43059,43058,43061,43055,43320,43574,43835,43831,43837,43837,43835,44088,44092,44091,44089,44349,44109,44371,44357,44346,44600,44592,44599,44596,44857,44599,44597,44601,44610,44621,44616,44876,45131,44871,44864,44861,44861,44858,45117,45111,45121,45127,45118,45118,45111,45115,45100,45105,45098,45103,45103,45119,45123,45129,45139,45138,45109,44842,45096,45103,44586,44336,44329,44334,44328,44073,44070,44077,43814,43816,43811,43815,43293,43298,43042,42790,42781,42782,42528,42531,42529,42527,42530,42529,42270,42275,42267,42271,42010,42267,42014,42273,42271,42269,
+50427,38889,38891,41203,41460,41462,41203,40946,40941,40947,40940,40936,40953,40692,40438,40428,40698,40687,40696,40434,40437,40427,40692,40698,40691,40950,42495,43008,43013,42752,42755,42750,43013,43010,43013,43013,43019,43012,43275,43526,43785,43781,43784,43779,43776,44030,44033,44032,44034,44053,44086,44351,44330,44308,44305,44295,44300,44297,44304,44308,44309,44566,44573,44579,44576,44583,44840,44838,44831,44826,44820,44812,45074,45068,45074,45088,45087,45095,45094,45092,45071,45078,45071,45072,45071,45086,45080,45082,45095,45096,45087,44813,44802,45064,44545,44292,44282,44288,44280,44026,44027,44035,43773,43779,43775,43781,43263,43267,43010,42757,42754,42756,42498,42498,42495,42236,42242,42241,42238,42242,42237,42240,41976,41975,41720,41977,41973,41970,
+50436,38910,39168,41223,41481,41226,40967,40966,40963,40713,40449,40443,40455,40451,40453,40446,40716,40704,40714,40451,40454,40441,40706,40967,40960,41217,42504,42759,42762,42758,42762,42755,43019,43015,43017,43017,43021,43014,43278,43531,43534,43788,43792,43792,43791,44045,44049,44047,44046,44040,44035,44294,44313,44333,44335,44334,44342,44338,44084,44340,44335,44590,44596,44597,44589,44593,44848,44844,44841,44842,44849,44849,45112,45109,45118,45118,45103,45104,45098,45106,45093,45105,45099,45107,45104,45116,45109,45101,45109,45114,45123,44884,44881,45142,44622,44629,44363,44370,44363,44106,44103,44111,43848,43850,43845,43851,43332,43335,43080,42826,42819,42823,42825,42827,42565,42306,42309,42307,42305,42309,42045,42050,42045,42045,41789,42047,42043,42040,
+50169,38875,39649,41186,40931,40675,40674,40675,40672,40421,40153,40146,40156,40663,40664,40656,40669,40659,40672,40669,40676,40665,40670,40930,41182,41436,42467,42720,42720,42716,42720,42717,42979,42975,42977,42973,42975,42968,43233,43226,43224,43475,43737,43739,43738,43734,43996,43995,43990,43996,44024,44287,44284,44274,44528,44525,44538,44540,44548,44547,44544,44542,44550,44551,44800,44806,44809,44808,44803,44798,45052,45050,45051,45048,44805,45071,45066,45070,45071,45080,45060,45060,45044,45048,45046,45057,44794,44789,44795,44805,44790,44785,44775,44782,44777,44786,44265,44273,44011,43756,43755,43762,43755,43756,43752,43501,42982,42985,42987,42990,42984,42475,42477,42477,42475,42217,42220,42220,41961,41709,41701,41706,41703,41960,41960,41703,41700,41698,
+50183,38928,39957,40985,40730,40473,40729,40729,40723,40724,40456,40456,40475,40732,40734,40726,40742,40732,40743,40737,40740,40726,40730,40733,41243,41500,42533,42790,42794,42790,42794,42788,43050,43046,43048,43046,43049,43042,43307,43308,43310,43565,43824,43827,43833,43841,44115,44127,44135,44139,44140,44388,44381,44364,44619,44620,44636,44635,44637,44635,44631,44632,44639,44640,44887,44887,44885,44881,44878,44878,45139,45138,45144,45139,44885,45147,45137,45141,45139,45152,45139,45146,45132,45130,45126,45143,44878,44874,44886,44907,44909,44906,44904,44910,44903,44909,44387,44398,44137,43880,43879,43886,43880,43883,43877,43622,43100,43103,43106,43109,43103,42590,42334,42334,42588,42330,42333,42333,41818,41569,41819,41823,41820,41819,41821,41566,41820,41816,
+50179,38658,39944,40972,40717,40718,40715,40715,40713,40723,40460,40460,40472,40466,40470,40468,40487,40990,40999,40734,40738,40721,40468,40731,41243,41496,42271,42783,42784,42779,42781,42775,42783,42778,42782,43297,43300,43291,43299,43302,43313,43579,43852,43870,43875,43874,43877,44134,44131,44143,44169,44435,44430,44414,44404,44402,44672,44672,44417,44671,44662,44399,44405,44407,44911,44917,44657,44917,44917,44911,44919,44926,44924,44912,44922,44936,44933,44930,44912,44925,45174,45177,44910,45422,45416,45175,45171,44913,44923,44949,45210,44944,44933,44940,44936,44944,44419,44425,44161,43906,43907,43915,43906,43905,43897,43387,42864,42865,42866,43124,42868,42877,42362,42362,42357,42355,42101,42102,41846,41850,41841,41845,41844,41844,41849,41850,41846,41842,
+50182,38919,40463,40721,40467,40724,40723,40723,40722,40734,40472,40464,40473,40468,40472,40464,40479,40982,40991,40728,40732,40721,40472,40735,41243,41498,42272,42784,42786,42780,42784,42778,42784,42781,42783,43297,43303,43298,43565,43564,43312,43566,43830,43830,43828,43825,43827,44084,44089,44115,44379,44380,44377,44376,44373,44353,44616,44610,44357,44613,44609,44349,44353,44354,44859,44866,44608,44862,44857,44857,44861,44862,44867,44862,44870,44874,44864,44611,44862,44867,45110,45118,44854,45373,45372,45135,44874,44615,44635,44645,44634,44868,44857,44864,44602,44607,44341,44347,44083,43828,43827,43833,43826,43829,43567,43314,42794,43053,43055,43059,42796,42798,42285,42286,42283,42281,42027,42026,41768,41772,41765,41770,41510,41765,41768,41770,41767,41764,
+50176,39155,40439,40439,40443,40701,40444,40187,40184,40189,40176,40168,40175,40433,40440,40689,40446,40434,40440,40945,40948,40934,40939,40941,40938,41965,42741,42740,42743,42995,42998,42994,43000,42999,43003,43259,43264,43260,43264,43262,43263,43515,43776,43780,43778,43775,44035,44033,44033,44059,44599,44095,44097,44093,44359,44351,44358,44350,44353,44353,44346,44343,44345,44344,44335,44341,44343,44343,44345,44351,44616,44614,44611,44605,44872,45139,45126,44611,45119,45381,45110,45122,45378,45129,45118,45118,44844,44592,44608,44619,44355,44323,44315,44324,44061,44065,43800,43806,43800,43803,43803,43810,43803,43806,43545,43548,43288,43290,43292,42780,42519,42526,42271,42271,42273,42016,42018,42018,42017,41765,41502,41505,41756,42013,42016,41504,41759,41757,
+50181,39433,40722,40467,40470,40726,40469,40470,39695,39957,40211,40213,40228,39968,40228,40734,40492,40482,40492,41000,41004,40994,40999,41001,40995,42021,42797,42793,42797,43047,43055,43050,43054,43050,43054,43309,43314,43309,43314,43311,43317,43567,43829,43832,43827,43824,44088,44090,44091,44101,44099,44095,44098,44104,44624,44618,44627,44621,44622,44363,44361,44614,44617,44358,44349,44612,44613,44354,44350,44349,44608,44609,44617,44868,44872,44621,44609,44616,44877,44891,45135,45144,44877,45136,45134,45146,44869,44598,44606,44613,44360,44357,44352,44358,44095,44103,43838,43843,43836,43838,43838,43844,43578,43324,43577,43067,42804,42806,42810,42301,42032,42026,42548,42551,42550,42035,42037,42036,42034,41782,41518,41521,41776,41519,41521,41522,41262,41259,
+50439,39690,40464,40467,40470,40469,40469,40468,40208,40214,40209,40204,40215,40218,40479,40724,40222,40210,40476,40469,40472,40460,40465,40726,41238,42271,42796,42800,42806,42803,42809,42803,42808,42805,42810,43064,43069,43062,43322,43318,43321,43318,43320,43577,43576,43831,43835,43836,44095,44379,44132,44140,44163,44160,44668,44653,44662,44659,44662,44403,44401,44659,44408,44151,44402,44406,44407,44150,44657,44400,44661,44656,44656,44912,44668,44674,44659,44405,44917,45185,44914,44920,44910,44659,44658,44672,44661,44649,44661,44675,44416,44648,44638,44131,44381,44387,43864,43870,43350,43352,43606,43611,43346,43096,43094,43101,43095,43098,42844,42334,42583,42583,42328,41814,42074,42073,42076,41817,41817,41826,41821,41830,41830,41574,41576,41320,41316,41311,
+50431,39670,40185,39930,39933,40444,40444,40444,40184,40188,40180,40173,40185,40182,39929,40176,40192,40181,40447,40441,40446,40434,40440,40700,41207,42231,42747,42745,42748,42743,42749,42744,42748,42743,42748,43002,43006,43001,43261,43259,43261,43258,43261,43520,43518,43771,43776,43776,44028,44286,44030,44287,44285,44286,44801,44794,44804,44799,44804,44291,44287,44030,44036,44038,44029,43777,43776,44031,44539,44284,44798,44800,44548,44800,44550,44556,44545,44292,44545,44553,44796,44804,44798,44545,44541,44552,44544,44541,44545,44547,44286,44033,44030,44036,43774,43780,43772,43778,43258,43261,43518,43523,43260,43007,43002,43007,43001,43002,42748,42238,41977,41977,42236,41978,42235,41721,41468,41723,41464,41214,41205,41209,41205,41462,41207,40694,40691,40685,
+50183,39953,40210,40211,40217,40219,40222,40221,40217,40218,40211,40202,40214,39953,39953,39945,40215,40205,40216,40472,40734,40723,40731,40734,41242,42271,42789,42787,43046,42785,42790,42784,43043,43039,43043,42784,42786,43039,43040,43038,43045,42790,43046,43302,43556,43553,43811,43812,44073,44085,44105,44127,44135,44120,44123,44110,44119,44117,44123,43865,43861,43858,43861,44120,44110,43601,43085,43336,44358,43847,43851,43336,43594,44612,44622,44376,44107,44369,44366,44116,44363,44375,44371,44372,44365,44378,44381,44377,44374,44378,44120,44105,44102,43851,43846,43853,43846,43598,43338,43598,43596,43344,43082,43085,43082,43087,42567,42309,42311,42058,42056,42061,41808,41808,42068,41812,41559,41557,41555,41301,41295,41300,41040,41041,41043,41301,41040,41293,
+50175,40182,39930,40180,40175,39916,39659,39658,39661,39927,40183,40185,40196,39939,39940,39934,39684,39926,40188,40440,40700,40690,40696,40701,41206,42229,42746,42748,43004,42743,42746,42745,43004,43001,43002,42740,42750,42998,43003,43004,43007,42746,43004,43007,43004,43517,43777,43779,43522,43772,44027,44043,44042,44040,44039,44031,44038,44031,43520,43260,41976,41463,43262,43267,43262,43269,41986,39677,39419,41724,41465,40439,43520,44032,43784,44560,44549,44290,44030,44035,44281,44289,44290,44299,44301,44314,44308,44297,44287,44284,44028,44043,44040,43789,43782,43788,43779,43528,43264,43011,43012,42760,43008,43010,43010,43014,42496,42242,42241,41988,41986,41989,41733,41731,41476,41473,41474,41473,41474,41220,41213,41214,40957,40956,40957,41214,41212,41209,
+50694,40709,39947,40189,40187,39926,39678,39674,39701,39954,39949,40211,40213,39966,39954,40213,39953,40208,40465,40719,40716,40715,40459,40466,40711,42256,42766,42520,42514,42518,42514,42520,42764,42765,42766,43015,43044,43035,43025,43028,43030,43047,43046,43312,43307,43825,43824,43817,43820,43810,43824,43838,43835,43837,43828,43828,44090,44077,43568,42792,37915,30728,30971,36373,38952,36380,32277,30482,21751,25851,27649,25087,34831,42781,43559,44086,44339,44068,44340,44329,44077,44077,44089,44077,44084,44337,44093,44084,44088,44088,44338,43836,43839,43846,43573,43587,43576,43583,43566,43063,43080,43080,43067,43069,43073,42820,42312,42569,42568,42314,42065,42068,41801,41797,41551,41288,41284,41021,41292,41544,41278,41277,41280,41277,41021,41537,41287,40769,
+50948,40724,39964,39927,39931,39914,39419,39167,39232,39450,39959,39968,39946,39967,39943,40209,39937,39954,40196,40716,40960,40977,40706,40724,41214,42517,42761,42525,42513,42527,42515,42535,42755,42767,42759,43008,43050,43054,43023,43019,43007,43015,43005,43017,43275,43797,43789,43782,43783,43801,43826,43814,43793,43798,43804,43801,43801,43799,43291,41488,35588,22232,13987,16032,19968,16595,14289,14570,12762,13783,12497,10704,13769,24576,34563,39690,41752,43538,44347,44573,44065,44043,44084,44056,44070,44049,44076,44055,44051,44065,44074,43805,43790,43806,43517,43539,43525,43545,43501,43006,43037,43033,43019,43021,42761,42500,42247,42253,42244,42239,42011,42006,41987,41980,41752,41480,41470,41208,41234,41226,40956,40952,40966,40957,40703,41213,41238,40725,
+50955,40232,39977,39691,39684,39659,39680,39685,39499,39455,40002,39764,39702,39979,39964,39975,39962,40244,39966,40246,40724,40749,40725,40750,41231,42542,42782,42543,42530,42551,42532,42567,42785,42808,42797,42806,42840,42851,43073,43064,43050,43052,43045,43055,43085,43597,43319,43582,43573,43602,43644,43644,43639,43633,43893,43880,43627,43626,43115,40540,36927,25605,20957,12440,9730,10441,10959,9453,8924,10211,10722,10973,11722,11511,13508,16816,23823,35391,43657,44409,44147,44116,43910,43873,43897,43868,43897,43874,44146,44146,43900,43869,43861,43871,43586,43613,43342,43373,43063,42827,42867,42865,42846,42856,42333,42328,42319,42090,42083,42324,42092,42080,42065,42058,42098,41571,41547,41541,41309,41304,41029,41024,40797,41036,40517,40770,40798,40547,
+50948,39691,39440,39667,39400,39385,39661,39666,39483,39441,39746,39778,39709,39987,39963,39975,40219,40499,40482,40253,40470,40510,40731,40759,41496,42553,42786,42544,42533,42555,42537,42571,42781,42807,42785,42794,42824,42843,43051,43041,43021,43027,43021,43025,43072,43066,42785,43050,43293,43305,43313,43570,43570,43569,43576,43311,43058,42799,41264,40232,33532,24550,21957,12925,9703,9634,9125,8393,8377,9148,10169,11460,11696,11475,10646,10077,11172,16299,33039,43301,43299,43789,44351,44054,43827,43805,43852,43831,43847,43821,43834,43828,43838,43571,43561,43578,43045,43075,42759,42778,42818,42817,42533,42547,42281,42280,42269,42044,42042,42022,42044,42034,41765,41750,41799,41531,41504,41246,41016,41007,40990,40987,40759,41000,40481,40728,40771,40771,
+50687,39687,39180,39670,39649,39133,39653,39665,39736,39688,39473,39495,39425,39674,39899,39922,39892,39924,39901,40191,40651,40445,40401,40442,41430,42488,42473,42494,42505,42774,42753,42786,42472,42505,42726,42993,43026,43048,42993,42993,42970,42959,42698,42719,42764,42774,42751,43006,42992,43007,43270,43527,43515,43519,43517,42482,41721,37872,30953,24790,15034,10437,10412,10121,9465,8890,8132,8681,9196,8935,9438,9189,8909,9448,9916,9595,9672,9650,14311,30422,42235,43751,43791,43750,43779,43756,43792,43757,43515,43506,43534,43539,43562,43027,43285,43288,43021,43049,42996,43021,42792,42798,42513,42268,42256,41999,41992,42020,42013,42242,42017,41492,41736,41719,41771,41502,41473,41222,40998,40472,40710,40705,40740,40715,40712,40690,41009,40732,
+50695,39459,38945,39446,39418,39420,39425,39444,39502,39450,39498,39515,39468,39772,40002,40048,39996,40038,40006,40306,40764,40560,40515,40565,41530,42329,42570,42587,42604,42873,42846,42882,42564,42598,42809,42827,42856,43136,43081,43091,43071,43084,42826,42826,42849,43113,43103,43107,43079,43092,43319,43072,43056,43064,43060,41766,36137,24843,14561,11466,8376,7627,8884,8860,8190,7866,7619,9432,10200,9927,9148,8130,7869,7894,8631,9065,9405,9115,9421,12178,26903,40477,43605,43319,43339,43579,43622,43580,43594,43573,43595,43609,43635,43085,43089,43083,42819,42848,42801,43088,42591,42599,42312,42324,42313,42310,42062,42089,42077,41790,42078,41553,41541,41529,41579,41561,41529,41271,41047,40781,40764,40755,40797,40759,40755,40471,40796,40516,
+50690,39445,38679,39433,39403,39405,39157,39177,39233,39185,39483,39500,39438,39722,39950,39999,39941,39999,39959,40267,40729,40522,40480,40533,41748,42560,42533,42552,42560,42839,42805,42838,42525,42560,42776,42799,42829,42855,43052,43064,43033,43056,42787,42786,42816,43072,43064,43071,43043,43064,43306,43065,43055,43065,42022,36880,26111,14827,10451,10953,9660,7893,7862,7838,7172,7370,7635,8162,8935,9428,9688,8673,7640,6889,7116,8060,8911,9387,8925,9363,12299,22239,39232,43045,43317,43306,43083,43306,43326,43306,43593,43598,43367,43068,43085,43083,42814,42832,42544,42568,42328,42336,42304,42316,42307,42301,42311,42080,42067,41780,41809,41539,41269,41267,41302,41543,41258,40994,40781,40765,40774,40757,40798,40498,40494,40468,40794,40519,
+50948,39699,38939,39433,39404,39404,39412,39433,39490,39456,39504,39782,39723,39495,39468,40033,39972,40035,39976,40285,40745,40537,40245,40812,42020,42844,42810,42571,42576,42601,42563,42597,42799,42833,42793,42813,42841,42871,42810,42826,42793,42825,42798,42796,42829,42826,42816,42826,42803,42816,43044,43049,43036,42545,39187,29177,17362,10434,9901,9387,8861,8371,7567,7025,6878,6562,7854,9139,8885,8601,9126,9905,8358,7600,7308,7487,8087,8823,8621,8804,9693,10641,24068,42004,43561,42789,43075,43041,43074,43048,43322,43071,43105,42797,43076,43334,43061,43070,42799,42302,42317,42330,42298,42311,42299,42291,42301,42070,42058,42032,42059,41536,41522,41532,41549,41289,41004,40992,41043,41012,40794,40766,40552,40243,40240,40472,40539,40526,
+50956,39721,38976,39213,39176,39174,39439,39457,39517,39480,39521,39809,39728,39493,39211,39795,39991,40054,39991,40301,40505,40553,40259,40826,42037,42867,42836,42618,42606,42633,42589,42623,42823,42855,42559,42575,42612,42900,42838,42860,42817,42847,42562,42829,42853,43105,42837,42849,43078,43089,42806,43073,42316,41822,35891,19988,11228,9697,9158,8390,7873,7383,6577,6032,7689,7371,7128,7645,7131,7365,7641,8155,8670,9203,8653,7550,7630,8622,8418,8845,9476,9138,10244,25099,40527,43343,43108,42829,43114,43079,43357,43107,42891,42841,42866,43122,43107,43117,42598,42348,42365,42374,42341,42354,42336,42327,42336,41844,41834,41807,42092,41834,41555,41565,41572,41061,40783,41026,41080,41027,40843,40806,40855,40553,40542,40260,40327,40323,
+50695,39466,39232,39209,38905,39162,39430,39439,39507,39461,39764,40059,39971,39742,39712,40030,39960,40280,39957,40265,40206,40524,40490,40789,42001,42569,42536,42593,42565,42853,42800,42839,42526,42558,42516,42781,42823,42854,42540,42567,42514,42551,42784,42799,42813,43327,42796,42809,43294,43305,42788,43042,42050,41783,27110,12526,9662,7613,7584,6555,6554,6840,7582,8323,9461,9409,7884,6600,5564,6059,6076,6574,7340,8632,8603,7258,7343,7054,7879,7537,8953,9643,7665,8901,21495,39452,43850,42795,42804,42777,43057,43290,42824,42515,42803,43054,42785,42793,42281,42038,42327,42336,42315,42328,42058,42049,42060,42079,42072,42048,41821,41825,41536,41546,41292,40778,40776,40750,41060,40725,40562,40520,40827,40529,40507,40225,40550,40038,
+50691,39454,39217,38938,38632,39137,39160,39420,39491,39436,39740,39519,39428,39727,39954,40274,39686,40005,39943,40253,40188,40528,40483,40773,41983,42557,42523,42581,42554,42854,42784,42827,42515,42553,42249,42771,42810,42849,42536,42555,42503,42550,42777,42789,42803,42812,42778,42795,42774,42786,42525,42534,42068,33062,17355,10737,7361,6615,7854,6054,5543,7107,8877,9875,10494,9669,8660,7631,6595,6073,5845,5056,5564,6340,7080,6766,6078,6047,6860,5999,7162,8104,8179,7363,9173,20673,38961,43051,42558,42534,42813,42783,42832,42265,42819,43068,42802,42552,42560,42312,42336,42322,42316,42062,42050,42042,42051,42070,42067,42046,41557,41308,41536,41554,41299,40786,40536,40511,40809,40472,40574,40519,40577,40539,40517,39990,40309,40309,
+50441,38958,38713,39463,39158,39402,39183,39180,39255,39216,39265,39307,39222,39788,39741,39803,39729,39785,39982,40030,40485,40330,40529,40559,42028,42864,42568,42368,42338,42642,42560,42606,42554,42595,42550,42818,42597,42903,42591,42609,42539,42584,42814,42570,42591,42867,42829,42849,42570,42839,43072,42034,35913,19441,11438,7898,5797,7629,8603,7315,5775,6312,8083,9084,9968,9135,8388,7622,7096,6564,6344,6310,6313,6064,6544,6222,5794,5508,5798,6229,6364,6281,7390,7602,7366,8070,23543,41272,43085,42804,43097,43075,43125,42561,42604,42592,42320,42317,42324,42333,42359,42591,42609,42341,42327,42061,41812,41834,41830,41815,41576,41585,41806,41305,41304,41046,40795,40527,40554,40472,40582,40518,40068,40027,40002,39740,39793,39786,
+50701,38966,38465,38959,38653,38898,38686,38676,39258,39217,39266,39309,39226,39291,39502,39825,39747,39800,39999,40046,40517,40361,40296,40818,42302,42882,42582,42381,42351,42655,42577,42627,42565,42604,42558,42832,42608,42919,42603,42613,42563,42609,42329,42599,42611,42889,42849,42877,43100,43104,42584,40776,27467,11995,9145,6124,6076,8682,8368,7851,6556,6317,6798,7282,8434,8623,8393,8136,7866,7588,6608,6827,7087,6833,6283,6480,6314,6027,5286,5984,6379,5782,6380,6591,7129,6028,9932,31810,43128,42850,42895,42879,42924,42615,42658,42638,42366,42362,42371,42375,42403,42116,42408,42135,41866,41862,41871,41894,41889,41877,41630,41636,41604,41360,41363,41107,41110,40593,40355,40526,40636,40567,40116,40071,40049,39795,39852,39847,
+50432,38938,38441,38933,38881,38613,38660,38650,39226,39188,39485,39263,39431,39742,39691,39502,39932,39986,39929,39971,40453,40288,40224,40719,42488,42808,42507,42559,42535,42582,42506,42570,42496,42532,42487,42763,42784,42840,42783,42535,42485,42523,42498,42518,42526,42545,42504,42539,42751,42755,42229,37862,16896,8136,7874,6396,6855,8431,7092,7869,7092,7109,7075,6523,7424,7868,7898,8154,7637,8120,8689,7887,7369,7634,6834,7018,7363,7337,6853,6272,5635,5551,5888,6096,6644,6562,6594,17146,39939,42986,42500,42496,42278,42229,42272,42248,42240,42238,41992,42245,42277,42244,42027,41749,41734,41983,42250,41761,41750,41740,41495,41502,41216,40971,40973,40975,40968,40716,40733,40138,40508,40444,40255,39953,39678,39686,39996,39747,
+50436,38949,38190,38426,38372,38616,38662,38655,38711,38679,38977,39020,39190,39253,39463,39530,39450,39759,39958,39992,40489,40315,40252,40734,42531,42837,42537,42583,42567,42612,42534,42604,42527,42566,42522,42805,42823,42881,42568,42315,42534,42571,42545,42565,42565,42587,42546,42581,42795,42551,41523,28154,8687,7608,7350,7417,7358,7910,6568,7346,6567,6323,7829,7268,8175,7852,7878,7618,7102,8354,9951,10173,8884,8121,6807,7504,9142,9376,9662,8833,6645,5023,5104,5825,6385,6555,6063,9448,32015,43299,42560,42561,42070,42280,42321,42301,42296,42296,42054,42309,42343,42311,42094,41814,41801,41796,41805,41317,41301,41805,41817,41821,41535,41546,41545,40778,40509,40258,40531,40191,40051,39986,39797,39757,39478,39745,40053,39808,
+50185,38707,38465,38447,38403,38652,38699,38696,38745,38719,38756,38807,39224,39280,39228,39290,39730,39782,39984,40271,40520,40856,40539,40761,42578,42868,43084,43123,43111,42901,42823,42892,42555,42334,42551,42579,42590,42645,42590,42588,42297,42334,42305,42590,42589,42865,42824,42346,43075,42567,39477,18147,7160,7368,7623,7439,7373,7931,6850,7374,7369,7385,8890,8585,9751,8144,8179,8692,8434,8655,10765,12279,11503,10736,8652,8314,10205,11719,13542,13751,11287,7105,5126,5327,6150,6323,5836,7423,23813,42537,42311,42849,42348,42304,42597,42573,42565,42308,42578,42314,42346,42055,42354,42329,42060,42051,41805,41834,41809,41539,41547,41287,41008,40762,40761,41022,40753,40762,40776,40433,40559,40492,40046,40015,39983,39994,40298,40057,
+50179,38707,38458,38442,38389,38635,38680,38675,38721,38697,38726,38774,39199,39258,39205,39262,39449,39495,39958,39984,40232,40566,40509,40729,42571,42842,43055,43097,43091,42879,42804,42873,42539,42318,42021,42560,42566,42620,42569,42566,42282,42324,42304,42078,42077,42612,42568,42342,42557,42824,33583,15075,7427,6613,7896,6935,6854,8438,8128,8137,8381,8656,9899,8562,9216,8634,9442,9696,9181,9658,11245,14039,14805,13525,10935,9316,10176,13232,15057,18106,17685,12496,8203,5594,5903,5817,5589,6148,12519,35856,42296,42323,42080,42298,42331,42311,42301,42302,42569,42048,41825,42042,42343,42313,42039,42022,41782,41811,41788,41524,41281,41021,40999,40497,40496,40502,40748,40759,40767,40676,40553,40481,39774,39487,39710,39975,40025,39789,
+50429,38682,38685,38411,38360,38610,38651,38647,38685,38656,38685,38730,38894,39204,39409,38953,39397,39701,39909,39932,40184,39997,40198,40927,42268,42781,43247,42773,42763,42804,42993,43062,42215,42258,42210,42233,42242,42295,42244,42492,42462,42490,42211,42237,42240,42518,42727,42505,42464,41189,31677,18331,6852,7310,8343,7383,7308,8891,9614,9617,10379,10408,10880,9033,9169,9610,10682,10164,9397,10396,10462,13773,17609,15043,13736,11096,10927,14743,16818,17799,20957,18852,14800,9117,6598,5989,5511,5826,6814,24256,41466,42003,42277,42247,42273,42256,42504,42251,42516,42251,42028,41726,42035,42008,41737,41717,41747,42032,42012,41494,41505,41247,40713,40719,40721,40725,40714,40987,40734,40391,40522,40193,39995,39711,39940,39690,39739,39760,
+50432,38688,38172,38156,38110,38359,38654,38650,38687,38655,38678,38716,38641,39208,39159,39219,39660,39707,39923,39941,40202,40009,40213,41200,42286,42543,43007,42780,42776,42811,42749,42816,42233,42289,42239,42267,42275,42330,42280,42526,42493,42516,42240,42265,42273,42556,42505,42537,42240,39425,30944,14783,7149,7351,8125,7670,8359,10452,10157,11181,12706,12730,12679,10057,9934,12166,13242,12467,10938,10914,11745,12236,17866,18132,16579,14716,12482,14511,16596,17324,20998,22750,21507,16854,10486,7570,5809,5610,6080,13007,38947,42829,42324,42299,42320,42301,42031,42036,42555,42291,42065,41765,41823,41795,41782,41753,41794,42075,42058,41540,41798,41279,40745,40753,40757,41014,41002,40510,40509,40426,40554,40227,39766,39742,39721,39726,39775,39800,
+50180,38197,38198,38182,38143,38393,38430,38684,38722,38685,38718,38749,38934,38983,38938,39252,39440,39485,39702,39967,39979,40291,40244,40974,42574,42576,42786,42805,42809,42583,42269,42331,42264,42317,42264,42286,42286,42345,42552,42535,42255,42538,42519,42546,42548,42571,42527,42558,42518,38940,23778,9389,7894,7332,8122,7935,9916,12007,10439,12753,15313,15091,14785,12689,12564,13774,16642,16631,15102,11733,14104,14083,16116,20757,19207,17847,15083,15307,17379,19654,22546,23787,24321,22766,17937,12203,7862,5858,5823,6845,28665,43595,42563,42282,42041,42022,42012,42273,42543,42018,42042,42260,42058,42028,41763,41730,41524,41796,41776,41516,41513,41251,40983,40993,40992,40739,40726,40494,40497,40158,40282,39958,39744,39730,39708,39713,39768,39539,
+49932,38195,38196,38437,38401,38400,38680,38684,38717,38688,38715,38745,38942,38724,38687,39258,39196,39237,39971,39977,39991,40292,40508,41241,42334,42594,42811,42828,42838,42602,42299,42355,42292,42342,42293,42314,42570,42628,42582,42563,42283,42553,42531,42567,42567,42588,42548,42839,43061,35891,15345,8664,8444,7872,8145,8720,11467,12780,10701,13526,16086,17144,16329,15255,15632,15821,20484,20468,19973,16601,15378,17154,19188,23056,22540,21185,19442,17613,20208,20948,22038,23799,24059,24314,23081,20453,14308,7435,6379,6117,21500,42324,42580,42052,42064,42049,42038,42043,42068,42049,42063,42287,42077,42049,41782,41749,41549,41556,41021,41275,41019,41271,41008,41016,41009,40503,40489,40772,40521,40187,40052,39739,39769,39761,39739,39484,39786,39553,
+50183,38188,38190,38432,38139,38148,38676,38682,38708,38444,38746,38768,38710,38744,38712,39276,39477,39516,39994,39996,40016,40321,40542,41524,42097,42350,42820,42831,42841,42599,42300,42354,42298,42350,42561,42577,42825,42616,42326,42309,42551,42573,42555,42583,42571,42592,42560,42330,43073,30507,11249,8408,7919,7601,7359,10001,12496,13295,10191,13524,16345,17653,18392,17835,20761,21982,24345,24321,24330,22234,20235,20473,23031,24583,25603,25546,24054,21718,20201,23513,23058,23544,23278,23538,24353,24564,21494,10487,6855,6595,21748,40788,41811,41798,42064,42047,42036,42033,42069,41786,41794,42023,41811,41536,41783,41752,41809,41294,41280,41278,41024,41277,41007,41018,40999,40497,40484,40252,40517,40182,39787,39734,39758,39749,39725,39470,39510,39536,
+49920,38169,37915,38158,38123,38393,38403,38409,38426,38411,38691,38713,38919,38954,38931,39232,39437,39479,39707,39968,39995,40291,40505,41493,42322,42569,42791,43059,42555,42567,42271,42319,42270,42323,42278,42545,42535,42318,42295,42280,42266,42283,42268,42806,42278,42556,42527,42295,41755,22778,8680,8417,7165,6593,7631,11031,12743,13275,10940,14013,17858,18393,20164,20385,23547,25028,26364,26848,26351,25547,24840,24051,24550,25325,27119,27843,27627,26832,24793,24253,24563,24808,24031,23785,24599,25068,24059,14100,6889,7151,19703,41272,42295,42031,42037,42023,42010,42004,42046,42017,42025,41995,41785,41507,41757,41728,41529,41526,41508,41250,41002,40999,40992,41010,40987,40485,40476,40243,40501,40170,39775,39727,39743,39740,39716,39723,39751,39262,
+49931,37952,37955,38195,38162,38434,38443,38450,38473,38476,38767,38781,38987,39018,38991,39288,39497,39532,39761,40016,40041,40343,40563,41564,42384,42373,42854,42858,42610,42622,42326,42369,42324,42378,42334,42604,42592,42374,42605,42596,42071,42343,42585,42613,42344,42619,42589,42614,40280,17656,7640,7882,6885,6575,8393,12580,14318,14092,13301,15865,19975,20765,21510,21989,23872,24328,25922,26666,27450,27157,26440,24871,24863,26158,27183,27648,26658,26127,25372,24835,25147,25134,24841,24585,24367,24317,24070,16904,7108,6851,14805,40803,42602,42338,42084,42071,42316,42312,42356,42074,42079,42045,41838,41557,41808,41525,41322,41578,41563,41309,41059,41057,41048,41065,41038,40791,40780,40546,40288,39960,39813,39773,39782,39784,39757,39763,39791,39298,
+49934,37949,38211,38194,38421,38691,38701,38706,38733,38741,38783,38807,39007,38782,39013,39060,39278,39313,39543,39800,40339,40388,40343,41867,42425,42416,42641,42383,42389,42399,42362,42403,42361,42419,42373,42391,42378,42418,42648,42638,42368,42643,42621,42385,42374,42647,42615,42644,37735,14054,7375,6834,6349,6558,8374,15128,16603,16630,16874,18921,20459,21248,21994,22476,22817,22490,23577,25096,25626,25599,24873,23820,24075,25374,25616,25047,24573,23781,23799,23773,24855,26125,26097,25599,24619,23539,23044,19473,8395,6096,12006,37761,42905,42641,42389,42633,42359,42351,42389,42108,42116,42078,42130,41855,41853,41832,41630,41372,41354,41094,41097,41093,41084,40842,40818,40829,40818,40585,40327,40006,39845,39807,39809,39818,39786,39534,39304,39323,
+49665,37916,38183,38157,38383,38650,38661,38662,38690,38682,38714,38732,38922,38693,38923,38959,39174,39207,39436,39693,40228,40273,40226,41749,42302,42300,42521,42268,42275,42286,42249,42292,42250,42309,42005,42277,42271,42319,42557,42549,42280,42298,42279,42305,42304,42577,42542,42823,36385,13007,6592,6312,5819,7569,8606,17916,19404,19942,19927,19917,20428,21731,22479,22697,22778,21422,21738,22751,23275,24272,23813,22768,22761,23296,23018,21934,21726,21708,22493,23233,24317,24301,23765,24031,24840,23488,22478,20964,10156,5284,11197,38199,43080,42560,42310,42554,42026,42018,42058,42032,42041,42002,41793,41510,41763,41741,41534,41022,41004,40746,41008,41001,40992,40752,40471,40738,40727,40495,40239,39922,39498,39467,39728,39742,39707,39457,38972,38985,
+49924,37918,37932,37643,38126,38389,38402,38404,38437,38427,38725,38751,38685,38721,38702,38997,39209,39245,39982,39983,40258,40045,39997,42037,42329,42334,42554,42307,42308,42322,42284,42329,42289,42095,42050,42066,42059,42102,42330,42319,42299,42315,42293,42323,42322,42596,42305,42587,34868,12008,6379,6344,6366,9908,11444,18947,19921,20455,20439,21200,22227,23530,23768,23471,23039,22458,21741,21465,21474,21960,22004,21470,20951,21490,21200,19855,20170,19895,19654,20395,21234,19412,18110,19657,22271,22987,23022,21512,11994,5580,10976,39506,42846,42581,42593,42323,42057,42049,42099,42070,42085,42052,42096,41560,41815,41793,41584,41589,41312,40795,41052,41048,40779,40796,40762,40518,40506,40269,40020,39960,39789,39760,39766,39523,39484,39492,39267,39279,
+49924,37925,38207,38170,38398,38403,38421,38424,38459,38440,38482,38762,38688,38715,38687,38986,39193,39235,39969,39715,39991,40041,40246,41781,42314,42071,42030,42300,42299,42322,42287,42330,42285,42091,42044,42058,42047,42093,42312,42307,42279,42304,42277,42307,42311,42582,42551,42062,34595,12243,6356,6078,6618,11725,16348,19738,19688,19962,21487,22505,21477,22525,22762,22210,21779,21197,20224,19441,18935,18904,19712,19442,18663,19200,18907,18081,17115,15812,14293,13994,14342,14818,15321,16863,19221,21722,23031,22009,15051,6065,10175,38956,42821,41784,42317,42305,42038,42025,42074,42042,42059,42023,41803,41527,41780,41758,41546,41556,41282,40765,41016,41010,40746,40772,40737,40492,40487,40246,40000,39687,39765,39739,39489,39506,39461,39463,39236,39245,
+49930,38192,38217,37917,38142,38149,38424,38436,38473,38465,38514,38800,38723,38762,38735,39293,38980,39539,40014,39765,40291,40092,40547,41837,42365,42394,42096,42116,42115,42137,42097,42136,42344,42411,42104,41868,41852,42157,42365,42369,42079,42099,42331,42632,42389,42402,42114,41885,37229,12801,6381,5836,6617,13517,18653,19985,19174,20480,21491,20451,18391,18415,17879,16803,15355,13990,14307,15328,15854,17356,17641,17634,17361,17384,16836,14968,11945,9606,8863,8554,9443,11452,13768,16082,18443,21721,23044,22542,17386,7358,7632,33890,42908,41856,41872,41855,41843,42090,42390,41846,41611,41833,41866,41588,41326,41305,41346,41099,41080,41076,41074,41071,40543,40561,40534,40289,40284,40298,39797,39744,39817,39793,39286,39048,39258,39258,39045,39050,
+49919,38165,38194,37642,37860,37865,38140,38409,38438,38421,38463,38751,38659,38690,38660,39217,38905,39207,39681,39694,40212,40017,40467,41755,42269,42299,42509,42524,42521,42032,41992,42035,42236,42302,41996,42020,42005,42311,42513,42259,41973,41999,42225,42511,42266,42537,41993,42019,39164,14823,6118,5842,6889,14544,19408,19969,19669,20455,19156,18119,16324,15065,12729,11146,9727,8870,9438,10711,12256,14791,16106,16865,16344,15603,15057,12419,9927,8616,8908,8848,9731,11984,14043,15832,18182,20413,21985,21741,19164,9422,7122,30202,43036,41732,41748,41738,41468,41715,41753,41468,41497,41717,41752,41476,41216,41200,41244,40999,40976,40968,40974,40969,40444,40458,40438,40189,40182,39684,39700,39648,39718,39693,39191,38953,38903,38646,38954,38956,
+49670,37668,37436,37911,37874,38137,38164,38175,38465,38707,38764,38798,38712,38749,38715,38759,39218,39262,39481,40008,40009,40071,40271,41816,42071,42103,42057,42071,42069,42352,42565,42355,42298,42367,42067,42351,42340,42644,42591,42336,42312,42337,42304,42331,42340,42617,42069,42609,41036,17163,5353,5323,7900,15574,19931,20488,20703,20723,18394,17356,15560,14554,12470,10626,9979,9893,10203,10710,11485,12996,15084,16101,16352,15093,13259,11643,11198,10397,10947,11147,13044,14791,15827,16336,16894,18359,20195,21491,22008,11746,8176,31558,42871,41820,42089,42077,41545,41535,41572,41543,41572,41793,41572,41552,41287,41273,41830,41587,41304,41296,41048,40785,40773,40786,40512,40521,39998,39754,39771,39717,39528,39502,39003,38764,38974,39230,39022,39026,
+49663,37652,37424,37897,37853,37593,37877,38142,38433,38669,38723,38495,38402,38692,38660,38702,38651,38951,39430,39955,39957,40017,40729,42011,42012,42040,41997,42012,42013,42297,42508,42298,42240,42301,41997,42017,42007,42311,42516,42258,42233,42259,42230,42260,42266,42279,42241,42520,41456,18640,5057,5307,9447,17117,19676,20999,21211,20715,17874,16320,15803,16082,15027,13703,12274,11170,10708,10957,11219,11964,13789,15831,16596,14831,12488,11386,10947,11177,12486,13720,15352,16333,16601,15829,15619,16817,18654,21236,23264,14013,8121,32764,43039,41733,41491,41480,41470,41462,41498,41472,41499,41719,41239,40963,41216,40946,41243,41254,40970,40702,40709,40700,40689,40704,40429,39926,39661,39672,39431,39121,38933,39162,39180,38426,38380,38893,38425,38428,
+49410,37150,37432,37651,37609,37606,37893,38156,38194,38180,38497,38787,38701,38736,38699,38991,39193,39492,39713,39725,39724,39526,41018,42043,42046,42078,42297,42313,41800,42089,42295,42084,41772,42086,41787,42062,42056,42356,42053,41796,42279,42048,42268,42292,42298,42314,42022,42307,41759,21753,5576,5816,10462,17880,19159,20740,20700,19435,17362,16584,16583,17120,17861,16798,14847,12468,11239,10715,10718,11461,13801,16868,17632,15610,12496,11136,11209,11955,13010,13473,14335,15055,15578,15568,16127,15793,16858,20464,24041,15831,8660,32545,42822,41514,41272,41515,41247,41241,41278,40995,41022,40986,41019,41256,40996,40982,41281,41294,41013,41262,41011,40748,40225,40240,39964,39974,39709,39461,39220,39166,38979,38949,38968,38980,38933,38933,38722,37953,
+49405,37125,37410,37890,37591,37586,37873,38391,38165,38145,38455,38743,38649,38683,38668,39234,39179,39482,39699,39968,39969,39766,41002,42019,42025,42052,42267,42283,41768,42055,42007,41797,41743,42054,41757,42290,42279,42068,41761,41759,42245,42017,42241,42271,42278,42292,42001,42028,42250,25327,6345,5815,10464,18652,19411,19451,19408,17882,16320,16311,16316,16606,17864,18084,16640,14265,13542,11733,10970,11202,13804,17644,19685,16620,12478,11119,11701,11931,12985,13451,13290,13494,13765,14274,16114,17575,17611,18661,24034,16068,10168,31223,41761,41221,41241,41227,41217,41212,41251,40965,40993,40962,40989,40968,40966,40950,41247,41262,40727,41234,40470,40462,39939,39952,39420,39688,39424,38918,39192,39135,38698,38923,39203,38703,38399,38140,38443,38446,
+49418,37424,37709,37934,37894,37887,38173,38184,38212,38206,38265,38562,38472,38513,38755,39316,39258,39308,39530,39803,39809,39602,41102,42127,41877,41903,42116,42398,41876,41904,41855,42157,41587,41896,41856,42127,42121,42171,42126,41873,42350,42111,42338,42380,42387,42404,42371,42400,42619,29001,6619,5820,9969,19457,19964,18984,18426,16904,15081,15587,17652,17681,17652,18137,16692,14318,13849,13320,12552,12013,12818,18724,23089,18475,12266,11162,13038,12752,11743,9116,7688,9420,10977,11738,14354,17369,20745,19221,20225,15340,18458,36988,41623,41082,40844,40829,41335,41328,41369,40827,40850,41080,41108,41090,40828,40811,41105,41124,40587,40837,40325,40312,40044,40061,39779,39789,39528,39277,39038,38723,39057,38764,38282,38801,39270,39269,38545,35715,
+49666,37409,37425,37655,37613,37608,37639,37636,37673,37907,37960,38246,38166,38455,38707,39259,39196,39248,39457,39728,39731,39524,41018,42045,41815,41827,42023,42322,41796,41820,41759,42068,41489,41536,41757,42025,41770,42339,42037,41789,42254,42028,42253,42294,42290,42299,42283,42300,42776,34064,8374,6047,9177,19430,20190,19221,17870,15578,15560,17339,16060,14024,14777,13451,11488,11934,12495,12477,11965,12980,11463,13277,20437,17884,10392,11616,13487,11658,9641,6490,6870,9634,10946,10173,13293,17341,21227,18441,8896,13518,29975,41010,41542,41252,41016,41009,40982,40982,41009,40740,40757,40723,40740,40722,40471,40452,40485,40510,39971,39977,39730,39720,39700,39706,39696,39437,39443,39196,38691,39159,39750,39448,37952,35113,29676,22732,15312,10669,
+49917,37115,37381,37348,37317,37310,37348,37343,37632,37611,37944,38226,38416,38685,38688,38981,38922,39228,39695,39715,39975,40030,41253,41776,41531,41802,41721,41770,41777,41522,41730,41785,41724,41764,41998,41760,42004,42067,41759,41761,41967,41971,41958,41993,41989,42243,42251,42267,42993,37372,12760,6583,7898,18134,20437,19449,16354,14304,16352,15805,13506,11981,10433,6797,6393,7610,11231,11491,11986,13792,12007,16118,26857,25333,14536,11901,14285,11945,13011,9365,8694,13262,16091,17381,17916,19408,21472,18941,14800,15563,32255,41756,42058,41508,41256,41261,40706,40716,40457,40710,40735,40436,40456,40186,39929,39922,39955,39981,39938,40207,39962,39681,39676,39672,39924,39673,40442,40198,36613,31454,25364,18147,13558,9693,7357,6081,6098,6862,
+49677,37197,37469,37426,37405,37373,37695,37669,37957,37918,38005,38022,38459,38727,38721,39006,38694,39263,39720,39749,40017,40326,41538,41804,41557,41825,41767,41560,41581,41581,41793,41846,41773,42070,41797,41563,42053,41852,41830,41815,41777,41792,41800,41810,41806,42324,42326,42342,42555,41037,17407,5832,6875,16356,19699,17120,10248,10477,16896,14807,11734,9966,12511,8115,6146,9953,13307,11787,14331,13818,12030,17433,25617,24598,16361,10131,15076,16866,16119,15836,16664,18943,20229,21775,23095,23574,23058,20504,19425,18144,25087,40259,41585,41040,40768,40781,40496,40505,40496,40241,40265,40247,40010,39991,40499,40482,40272,40027,39475,39228,39754,40766,40750,39716,36122,32274,25337,17863,11468,7889,7161,6610,6106,6100,6327,6077,6092,6589,
+49671,36879,37451,37137,37126,37344,37679,37653,37687,37908,38252,38274,38457,38737,38731,39025,38708,39287,39713,39749,39782,40323,41805,42061,41822,41835,41795,41583,41584,41588,41802,41855,41527,41815,41566,41571,42055,41845,41840,41821,41782,41793,41804,42078,42060,42321,42322,42088,41783,42825,23550,5579,9437,17382,14566,16626,17197,16406,17926,15354,14826,15378,16371,14294,13070,15610,15873,16149,16894,13309,9980,17954,25863,26153,18416,11176,17126,18156,16634,17133,17949,18692,19475,21267,23899,25155,25386,21805,20967,18406,21744,41552,41346,40798,40531,40540,40507,40521,40518,40004,40006,39995,39761,39476,39478,39463,40016,40291,40769,39235,35652,29455,22566,16104,10744,7658,6112,5807,6358,6379,6662,6380,6649,6890,7122,7893,8689,8404,
+49926,36618,36941,37137,37128,37346,37411,37384,37417,37904,38481,38249,38191,38464,38697,38990,38929,39493,39412,39446,39746,40287,41777,42033,42037,41800,41505,41545,41534,41529,41516,41550,41485,41505,41524,41785,41758,42069,42060,42050,42002,41757,41776,42050,42031,42289,42294,42074,42273,42838,33311,10697,7360,13522,15310,18140,18687,17906,19159,19672,19906,18923,18637,17847,16608,15301,15554,16617,18892,17886,12485,19460,27871,30984,24532,14714,18884,18881,17096,18632,19173,19930,21224,22772,25141,26920,26359,22533,22750,17881,19145,40985,40513,40242,40237,40241,40461,40223,39960,39702,39708,39949,40223,40718,40721,39932,37661,31227,23508,15807,9908,6504,5588,5517,5811,6059,5790,5747,6556,6567,6853,7080,7092,7069,7835,8074,7355,6535,
+49664,36345,36668,37128,37122,37089,36892,37126,37158,37387,37691,37993,37954,38210,38454,38734,38936,39248,39183,39473,39760,40303,41794,42054,41793,41805,41535,41563,41544,41532,41276,41302,41500,41527,41543,41809,41774,42082,42069,42051,42050,41766,41843,41859,41811,42053,42060,42099,42538,41545,31486,15833,7594,17639,18386,17888,19477,20499,19702,22778,22242,19981,18406,17374,16630,16622,16347,17940,20461,17916,15077,21026,28402,32531,28148,18592,16344,18647,17876,18141,18683,20210,21990,23788,25135,24855,22764,24075,24021,19410,19388,40489,40016,40257,39999,39995,39965,39978,39961,40481,41260,39957,36380,29684,23001,16594,10407,7026,6039,6310,6316,5745,5592,6036,6586,6820,7591,7800,7331,6051,6088,6301,6091,6565,6315,6546,7373,7581,
+49410,36614,36926,37131,37129,37095,37138,37117,37413,37630,37672,37973,38213,38468,38462,38499,38689,39003,39452,39487,39761,40560,42072,42059,41801,41806,42053,41831,41810,41550,41548,41579,41511,41535,41813,41800,41770,42049,42056,41778,41828,41758,41837,41855,41795,41800,41792,41848,42272,38227,20727,17655,11212,18432,19700,18684,19753,22065,20233,19982,23039,22332,20231,18703,17942,16658,15879,18989,17936,16174,17177,22079,28424,33347,30756,22244,18700,18438,17903,16359,15891,16893,18675,20481,22583,23332,24330,24611,24043,19950,22506,40486,40009,39738,40244,40763,40733,40998,37913,31751,23290,15301,9413,6327,5279,5578,5785,6264,6301,6576,6582,7290,7397,7326,8132,8617,8381,7017,6584,6830,7379,7072,7380,8108,8367,9119,9423,8864,
+49416,36901,37172,37134,37158,37110,37158,37134,37434,37659,37705,37760,37984,38503,38494,38527,38717,39025,39495,39512,39786,40576,42100,42081,41829,42102,42339,41870,41842,41837,41836,41871,41547,41580,41871,41830,41809,42086,42106,41834,41884,41832,41889,41932,41848,41868,41851,41906,42069,39533,15353,16386,12520,17963,20763,20263,20543,22098,23333,22073,20229,19524,18961,18468,17449,15388,15904,18479,18482,16730,18230,22880,28182,33362,32037,24821,19742,20754,22279,20736,18714,17934,19725,22049,23639,24148,23602,23622,23814,20994,30993,41300,41858,40554,37713,31797,24622,16631,10221,6644,6133,6369,6127,6125,6100,6141,6603,7590,7633,7386,6633,6300,6420,5813,6149,6103,6130,6814,7666,7916,8460,8670,9489,9957,8677,7381,7418,7121,
+49412,36638,36905,37384,37156,37107,37167,37146,37173,37653,37692,37744,37704,38465,38729,38743,38681,38721,39208,39471,39495,40294,42073,42055,41798,41805,41783,41824,41294,41544,41574,41606,41520,42067,41837,41800,41773,42066,41810,41553,41577,41543,41586,41635,41549,42089,42074,42121,41783,41347,17150,13305,11725,17936,20454,20999,21768,22566,23544,24842,24568,21802,19183,17149,16383,17152,19466,20518,18453,15926,19723,23350,27905,32572,32535,27376,20225,19197,22017,24072,24076,22525,22509,22509,22301,21510,21220,21773,21984,20459,33040,35092,25614,16856,10392,6512,5064,5012,5787,6065,6323,6558,6833,7330,7315,8371,8593,8545,6801,6275,6573,6729,7113,6761,7874,8855,8881,9062,9643,10148,10439,10397,8900,7064,7074,7828,8371,8348,
+49430,36661,37196,37175,36929,37141,37203,37184,37206,37684,37746,37794,37758,38518,38791,38798,38739,38777,39270,39285,39553,40620,42396,42111,41870,41868,41873,41899,41375,41615,41645,41666,41834,42127,41638,41880,41838,42135,41877,41623,41649,41610,41658,41694,41603,41633,41613,41668,41837,41643,18692,13839,12251,18489,19982,21290,22328,23118,23077,22567,22580,21602,20519,19264,18986,20287,20785,19513,17200,16728,20012,23641,27429,32094,33096,29227,22303,18201,19986,23568,24349,23830,23297,22542,20801,20018,19727,20014,20471,20219,14543,7347,5848,5578,5539,5778,6123,6073,7105,7634,7128,6333,6355,6073,6071,6358,6334,6268,7108,7342,7389,8327,9469,9891,8944,8380,7382,7045,7885,8142,7137,7095,7896,8396,8664,8893,8915,8619,
+49407,36613,36640,36869,36876,37088,37147,37126,37405,37362,37679,37709,37915,37912,38189,38457,38654,38954,39435,39199,39206,40777,42546,42514,42026,41761,41768,41786,41779,41508,41789,41816,41472,41506,41530,41514,41468,41246,41245,41252,41529,41495,41545,41585,41751,41785,41507,41560,41980,42045,19951,15630,13531,18713,20716,21505,22290,22311,21238,20731,20221,20010,19957,19993,20480,21006,20469,18688,16906,17456,20992,24352,26337,28943,28910,27619,24571,20737,18413,21477,23553,23543,22760,22513,21534,19975,18917,18956,18898,19422,9917,5307,6634,6365,7097,7337,8192,8660,8154,6884,5867,6609,6889,7116,7880,8160,9167,9620,9677,10423,10737,10653,9731,7334,7156,7364,7905,7572,8406,7897,8688,9930,10219,10207,9960,10189,9956,10686,
+49416,36391,36151,36641,36901,37120,37179,37154,36929,36888,37477,37507,37965,37954,37977,37989,38699,38748,39235,39263,39524,40843,42107,42080,42099,42089,41842,41861,41852,41577,41602,41368,41536,41572,41588,41574,41527,41300,41303,41321,41582,41549,41863,41905,41814,41846,41572,41624,41789,42607,26626,16402,13545,18740,21002,21030,21033,20550,19985,19473,19218,18745,18956,20274,20506,20779,19461,17944,16151,20809,23837,23351,23028,23327,23535,22487,22535,25376,19705,20209,22547,23309,22525,22014,21038,19998,19199,18470,18148,18138,10661,7342,6628,6361,6063,6049,6649,6858,6350,7145,6895,7895,7919,9430,10198,8933,8142,7567,7886,8370,7663,7316,6913,7847,8698,8136,8931,9116,9688,9944,9717,8901,8941,8922,8935,9161,8671,7343,
+49407,36365,36381,36615,36615,37095,37404,37375,37164,37356,37439,37468,37916,37896,37924,38199,38142,38710,39188,39732,39469,41305,42044,41757,41784,41771,41785,42054,42048,41515,41541,41572,41222,41263,41271,41271,41221,41253,41512,41519,41275,41494,41821,42110,42023,41532,41507,41558,41461,42306,35577,17130,13226,18666,20678,20189,19949,19977,19927,19675,19934,19974,20960,21767,21738,21759,20692,18671,17906,23592,21747,18695,18106,18670,18623,16801,16597,20727,20453,20696,23023,23525,22994,22467,20976,20179,19374,17874,17316,17339,10401,6279,6322,6823,6530,7285,7881,8862,9626,9394,9402,9888,10680,8856,7313,7079,7309,7512,7836,8062,7870,9071,9944,9603,9940,9636,9913,10360,9900,7591,6851,7316,8128,8365,7348,7316,7856,8073,
+49414,36395,36408,36646,36647,36860,37169,37134,37204,37392,37229,37507,37953,37938,37962,38250,37926,38493,39222,39771,39506,41344,42087,41794,41824,41808,41823,41835,41819,41551,41567,41601,41249,41294,41296,41295,41250,41285,41030,41293,41300,41265,41592,41883,41809,41067,41034,41091,41771,41580,40482,21280,13791,18972,19962,19223,18980,19264,19727,19993,21013,22089,22812,23615,23836,23599,21760,18968,18729,20306,16151,13618,14572,13857,13789,12991,10227,13846,18185,21248,23063,23571,22798,22265,21539,20480,19175,17678,17122,16610,10190,7377,8200,8179,9431,9933,8983,7658,7391,7930,8195,7661,7164,7394,8670,8685,8914,9117,9699,10689,9981,9130,8718,8115,8971,8921,7913,7329,7134,7386,8185,8646,9201,9174,8672,8641,9441,9916,
+49154,35857,36130,36626,36626,36577,36889,36849,36930,37108,37215,37248,37696,37688,37956,38242,38422,38734,39200,39499,39740,41578,42058,41762,41793,41530,41544,41308,41285,41279,41296,41340,41239,41287,41542,41536,41493,41534,41283,41290,41288,41262,41321,41614,41542,41305,41270,41332,41500,41570,42024,26678,15106,18234,18968,18482,17969,18511,18974,20011,21278,22097,22558,23365,23849,23352,21514,20009,17708,15432,8197,4889,9943,12063,12252,8372,4066,11017,17417,22027,22302,22555,22557,21773,21319,20783,19223,17968,16881,16096,11467,10431,10741,9949,9150,7590,7415,7884,8387,8417,8163,8916,9694,10443,10698,10454,10689,10634,10453,8873,6882,6286,7162,7589,7420,7377,7385,7825,8919,8398,8177,7870,8685,9932,9949,9405,8925,8629,
+48909,35642,35914,36418,36412,36623,36940,36897,36987,37169,37278,37308,37505,37751,38293,38321,38502,38816,39027,39326,40065,41912,41879,41836,41872,41610,41371,41391,41371,41363,41121,41164,41319,41366,41364,41617,41579,41367,41368,41372,41117,41080,41405,41449,41372,41394,41363,41422,41336,41402,42091,32125,19516,19066,18770,18286,17775,18055,18515,19552,20830,21640,22358,22650,22613,22628,21045,21591,16720,13165,7724,3398,8971,11851,12806,9696,8472,12348,18498,23617,22606,22341,22598,21815,20849,20577,19532,18541,17465,16687,9489,7694,8010,7219,7185,9221,9297,8743,9243,10039,10559,10283,9007,8216,8469,8993,8459,8141,7193,7670,8505,8422,9028,9457,9281,9234,9502,9172,9752,9229,9011,9468,10286,10246,7960,6644,6930,7149,
+48897,35861,35878,36377,36373,36577,36635,36852,37186,37108,37209,37219,37423,37681,38214,38239,38426,38742,38952,39510,39739,41585,42067,41763,41801,41792,41553,41573,41553,41288,41038,41075,41234,41282,41021,41014,40973,41268,41271,41276,41029,41254,41325,41354,41272,41299,41261,41320,41231,41041,41501,36667,20973,20517,18169,17681,17433,17717,17926,18959,19985,20808,21784,21827,21542,21557,20746,22320,17701,12091,11017,9755,12246,12310,13534,14026,14589,14861,21768,25096,24089,23057,22544,21512,20525,19981,19444,18705,17362,16071,9642,8093,9183,10452,10677,10665,10738,10443,10687,9942,7641,7108,7633,7870,7864,7367,7084,7791,8895,9124,8928,8847,9199,9888,9970,9669,9172,8064,8648,8895,8678,8110,7646,6575,7891,8379,8410,8633,
+48912,35626,35901,36663,36667,36613,36674,36895,36963,36899,36992,36997,37205,37463,37737,38007,38452,38766,39235,39534,40024,41355,42094,41791,41834,41825,41586,41355,41334,41068,40827,41121,41026,41072,41072,41057,41019,41058,41061,41070,41069,41030,41092,41132,41050,41079,41051,41110,41020,41082,40749,40529,22286,17216,18207,17201,17476,17504,17713,18751,19507,20061,20524,21077,21302,21316,22045,23365,21571,15704,14635,16198,15367,14658,14335,16366,18978,21307,24371,25390,24891,23603,21806,20518,20302,20025,19744,19008,17138,15863,10726,10208,9755,8715,7901,7883,8482,8699,7408,6919,7689,9210,9739,9460,9192,9203,9951,10401,10217,9672,9982,10413,10514,9150,7440,6627,7157,7333,7138,7136,7172,7377,7679,8146,8947,8665,8687,8650,
+48635,35074,35859,36357,36353,36313,36621,36844,36636,36834,36910,36915,37132,37379,37402,37675,38127,38693,39164,39205,39944,41279,41510,41465,41503,41493,41253,41272,41253,40983,41003,41294,40944,40987,41245,40977,40683,40459,40463,40732,40736,40707,40771,40799,40711,40486,40201,40772,40937,41772,42222,40726,29650,18402,18361,17101,16848,17394,17597,18120,18630,19184,19648,20202,20686,21473,23226,24030,23258,21504,19150,19168,18850,17874,17822,20379,22219,23005,24016,24776,24535,22732,21706,20159,19675,19920,19375,18911,17310,15515,10616,7523,7069,7571,8306,8032,7603,7558,8067,9117,9380,9358,8867,9614,10120,9363,8574,8249,8067,8294,8873,8024,7098,6498,8121,8333,8871,8792,9105,9356,9656,9861,9904,9602,9894,9860,10144,10107,
+48644,34844,35890,35869,35868,36094,36403,36635,36671,36874,36947,36967,36921,37419,37442,37715,37913,38477,38952,39231,40009,41323,41298,41524,41549,41025,41295,41310,41306,41290,41309,41338,40994,41041,41302,41293,40983,40244,40510,40771,40789,40758,40816,41108,41526,42068,42563,41330,36638,29771,21240,12029,10971,22833,18935,17176,16399,17193,17919,17915,18175,19235,19703,19986,20732,22029,23546,24076,23052,21808,20991,21283,19939,19475,21734,24024,24323,23828,23294,23544,23561,22800,21502,20234,19767,19735,18936,18204,17118,14563,8411,8936,9491,9223,9184,8912,9750,10472,10226,10750,11006,10981,10748,9951,8155,6893,7385,8094,7898,7868,7416,7325,8197,9142,9196,8897,8662,9359,9932,10180,9973,8644,7140,6861,7129,7612,7895,7349,
+48635,34823,35606,35589,35585,35821,36114,36364,36379,36601,36908,36668,36886,37135,37423,37704,37876,38179,38400,38940,39968,41297,41248,41472,41507,40976,40977,40997,40999,40985,40757,40788,40434,40486,40752,40726,40425,40192,39941,40211,41260,41228,41531,39767,36080,29161,21430,13206,7571,5753,5471,6560,7548,15565,18863,17346,16599,17135,17867,17856,17618,18411,19166,19442,20452,21988,23023,23029,21234,21785,23022,22525,21943,23279,24782,25815,25586,25603,24542,21968,21219,22257,21432,20405,19940,19392,18093,17883,17025,13951,9567,9839,8849,8582,9065,9816,9632,7774,7553,7555,8332,7791,7048,6774,7525,8582,9077,9277,8826,9315,9899,10611,10172,9848,9898,10369,10655,10336,9361,7837,6827,7832,8102,8094,8606,8074,7315,7793,
+48639,34843,35620,35623,35601,35847,36131,36378,36397,36353,36671,36956,37196,37432,37442,37739,37642,38217,38416,38962,40002,41596,41277,41760,41025,41278,41023,40786,40763,40506,40795,40556,40202,40250,40255,39984,41223,42023,42027,39468,34076,25567,17851,11147,8037,6003,6239,7536,7302,6289,6254,6274,6245,8346,16816,17860,16848,16851,17099,17589,17353,17879,18134,18912,20435,21458,21720,20443,20451,22546,21723,18151,18595,22255,22442,18070,16317,19405,24012,23243,20181,20701,20933,20169,19698,19163,17848,17890,16525,13431,9568,10076,9847,9821,8785,6946,7289,7744,8037,7790,7014,7248,8307,9310,10082,9569,9081,9808,9831,10306,9332,6945,6521,6709,7023,7265,7774,7217,6479,7539,8834,9331,9087,8813,8553,8543,8558,9299,
+48383,34343,35365,35368,35349,35855,35880,36118,36142,36108,36447,36474,36949,37447,37440,37739,37904,38221,38415,38711,39998,41350,41035,41002,40773,40516,40265,40275,39997,40253,40536,40312,40212,40780,41028,39224,33014,25342,17888,10973,6852,6103,6068,6330,6570,6346,6832,7090,6867,6630,7126,8424,9172,9188,14582,18168,16645,16893,17162,17125,17403,17669,18174,18954,19965,20219,20214,19977,21782,18745,13021,12285,12992,13812,13985,12450,12522,12263,14044,19433,19192,19455,20197,19182,19480,18937,17398,17175,16342,11972,6860,7074,7362,7081,7082,8336,8918,9381,9153,8902,9154,9888,10183,9891,9712,9392,9447,9650,8900,7576,6861,7846,8670,8628,8662,8397,7867,8347,8900,8658,8666,9163,9179,9424,10188,9671,8912,8376,
+48133,34106,35903,35891,35907,36151,36163,36151,36165,36128,36455,36480,36718,36701,37206,37510,37672,38257,38708,38494,40529,41639,40815,40511,40543,40031,40030,40046,40284,40802,41068,38531,32286,25894,18187,11520,7102,5602,6101,6390,6880,7410,6611,6095,6601,6887,7379,8664,9205,9969,9957,8939,8407,6601,12265,18162,16892,17147,17151,16603,17140,17929,18435,18716,19465,19210,19684,19977,17955,11308,10204,11771,11459,11264,10923,11181,10981,9953,9158,11711,14573,17648,19188,19447,19485,17649,16397,15900,15313,11450,8149,7876,8928,9161,9419,9895,9698,9647,10186,10426,9391,7541,6826,6790,7391,7336,7120,7093,7606,8107,8912,9135,9174,8877,8916,9157,9658,9629,9930,9948,9689,10192,10210,10210,8896,7361,6848,7085,
+48125,34085,35884,35871,35885,35357,35620,35604,35631,35581,35906,36451,36427,36415,36663,37229,37632,37966,38163,38705,40490,40561,39734,39959,41016,41272,41519,40261,35360,27909,19425,12751,8389,7099,6334,6870,7332,6343,6326,6352,6832,6859,7340,8614,9128,8889,8123,8365,9160,9131,8141,8661,9415,9437,13040,17396,16649,17163,17420,16620,16381,17178,17665,18466,18444,18710,17370,15352,11033,8235,8684,9746,9927,9217,9126,10162,12539,12026,9959,9153,11248,15592,17662,19188,19488,16877,15616,15118,14009,10669,8891,9128,9684,8887,7624,8368,8675,8633,7634,7350,6844,7311,8644,8623,9202,9161,8935,8916,9159,9147,9438,9142,9443,9409,10212,9687,8383,7846,7366,7662,8155,8156,8160,7135,6852,8145,9437,8656,
+48127,34106,35376,35353,35379,35360,35367,35614,36148,36126,36184,36473,36430,36697,36696,36721,37398,37980,38175,38467,41030,42643,41302,38179,34620,28440,21218,14293,9644,7595,7315,7327,7083,6305,6314,6575,6544,7859,8872,8888,8360,8363,9908,9891,8093,7834,8891,8611,8899,9907,10711,10966,10931,10953,11456,16341,16360,16364,16860,16608,16125,16379,17120,18431,18161,17917,14269,9936,7907,9222,13532,16396,17860,16623,15798,17608,20993,20469,18418,16576,16599,15293,16349,18658,18692,16074,15325,14589,13229,10400,9389,8315,6809,6260,6544,7541,7593,7296,7322,7315,8076,8817,9111,8839,9147,9885,10176,9894,9876,9612,10170,9878,9397,9374,7593,6824,7304,7546,8082,8121,7847,7587,8105,8362,8888,9144,9391,9132,
+47872,33321,35100,34831,34855,35109,34858,35356,35611,35591,35391,35930,36150,36150,36144,36178,36862,38464,39935,37920,34072,26933,19201,13314,9216,7693,7150,6658,7153,7423,7147,7151,7427,7675,7942,8451,8932,9455,8708,7680,7676,7415,7947,8682,9450,9964,10494,10735,10499,9719,9474,9730,10478,9991,9966,15359,15630,14599,15085,15861,15638,15380,16384,17945,17666,16911,14566,13071,14108,15164,17174,18476,18915,18440,17899,17629,17929,17403,18189,18173,16645,15609,16138,17410,16664,14837,14336,13295,12038,7643,6648,7376,8443,8672,8955,8939,9987,9458,9725,9468,9186,9157,9709,9944,9998,8943,7693,7150,7133,7397,7419,6887,6916,6900,7677,9224,9194,8933,8947,9480,9463,9722,9988,9735,9741,9480,9976,9722,
+47874,33078,34350,34856,35122,35132,34624,34852,35380,35349,35153,35438,35402,35912,36161,36720,37653,34376,24310,13552,8667,7650,7369,7915,8161,7663,6855,6878,8149,8159,8139,9166,9443,8405,7406,7393,7628,7627,8179,9458,9708,10206,10748,10204,10205,10977,10220,8924,8428,9462,9717,9710,10214,9452,7357,12504,15350,13289,13789,15085,15617,15368,15348,16653,17396,16125,15071,15874,16391,14623,15368,15369,15040,16106,16599,15824,14572,13803,14326,13271,13556,14817,15335,15581,14567,14041,12769,12481,10728,8130,8663,8877,9156,8881,8641,9146,9931,9149,8901,8643,9130,9360,9145,8360,6862,6846,8404,8910,8361,8897,8650,8376,8911,9666,9680,9414,9175,8905,9422,9957,9929,9425,8394,7365,7121,7117,7610,8126,
+47869,33332,34597,34591,35103,35100,34859,34830,34852,34807,34870,34386,34604,35374,35878,33871,22251,11509,8398,8164,7898,7658,7367,7654,7898,7917,8399,7905,7373,7384,6597,7623,7646,8643,8950,8924,9415,9675,9967,10214,10462,9689,8164,7360,7368,7892,8411,8924,10472,11245,10694,10221,9220,6138,3287,7411,13846,12818,12028,13570,14610,14615,14595,14871,15868,15612,15606,15882,14101,11299,11281,12574,12249,12554,11993,12760,12024,11496,10227,10707,12798,14330,14076,13811,13312,13037,12289,11750,10240,9422,9446,9433,9209,9194,9468,8685,7674,6629,6896,7409,6867,6594,6369,6872,8447,9470,9737,9219,8924,9191,9464,10477,10231,9184,9206,9450,9982,9467,9207,8964,7660,6638,7658,8683,8188,8699,8168,7404,
+47875,32833,34357,34338,34599,34852,34869,34841,34870,34827,34634,33893,33591,34361,30497,13821,8420,7922,8400,7923,7392,7671,8144,8167,7652,7928,7641,7139,7904,8427,8931,9188,9710,9942,9734,9455,7886,7891,7925,7662,7386,7636,9184,9168,8916,9955,10719,10209,9961,9975,10456,9955,6380,3544,2761,5611,11261,12291,12016,12021,12797,13317,13297,13571,14580,15089,15849,16126,14344,11538,9725,9734,9141,8428,8123,8622,9692,9942,10475,11746,13043,13808,13294,12507,12269,11986,11749,11489,8177,7113,6871,7100,7639,7111,6865,7622,8157,8402,8926,9440,8890,8871,9164,9416,9447,9432,9956,9437,9411,9933,9178,7892,6871,6592,7127,7383,7130,7129,6862,6866,8149,9176,8919,8918,8680,9446,9693,9193,
+47876,32843,34372,34098,34102,34360,34110,34090,34370,34074,33872,33125,33089,32318,16390,8436,8692,8194,7904,8195,8693,8202,7649,7668,7670,8199,8940,9718,9973,9974,8694,7910,7926,7898,7691,7674,8929,9195,9735,9986,9466,9464,10505,10482,9965,9208,9202,9715,8955,8969,10228,8699,4360,4604,7927,9487,9485,11032,12301,11537,11289,12313,12809,12819,12808,13832,14590,14865,14369,13101,11545,10532,9941,9741,9957,10717,11791,12301,12315,12559,12824,12816,12818,12036,11806,11777,11282,9991,7179,8442,8468,7412,7167,7668,8193,9720,9989,8693,8961,8960,9443,9681,9203,8183,7945,8451,9482,9479,8421,7660,6914,8449,9218,9458,9473,9474,8194,8709,8692,8961,8960,8699,8695,9208,8967,9732,9988,9478,
+47877,33109,34120,33848,33593,33604,33864,33592,33608,33572,33367,32627,32595,24099,9428,9174,9686,8166,7362,7138,7900,7914,8657,9696,10222,9459,8665,8671,8153,8413,8163,9179,9190,8387,9210,9957,10189,9936,10219,9443,9185,9442,10225,9434,6593,5578,7632,9694,9447,9454,9687,6851,6858,8664,8395,6868,7886,9435,11475,11490,10474,11248,12014,11762,12010,12522,13549,13561,13311,12822,12541,12292,12728,14057,14796,15040,15080,14552,14313,14306,13030,12762,12515,11734,11495,10958,10192,6854,6345,8131,8912,8889,9160,9148,9162,9662,9678,9409,9163,9424,9135,7063,7092,7092,7371,7623,7370,7104,7333,7865,8661,9424,9679,9154,8909,9165,9432,9431,9410,9419,9688,9680,9419,9672,9690,9171,7371,6606,
+47614,32571,33582,33311,33314,33060,33325,33050,32796,32767,32046,31043,30503,15329,9413,9677,8134,7895,8884,8650,9154,9428,9916,9657,8904,8392,8888,9153,8894,9147,9678,9927,10447,9396,9185,10191,10422,9143,8142,8138,8402,7380,5834,4008,3485,7359,9409,9671,9169,9952,9930,8654,8427,7407,6363,7147,7145,8685,9440,10733,9975,10227,10733,10737,11246,12011,12268,13055,13324,13354,14616,15658,16098,17413,17649,17642,17678,18176,17673,16119,14063,13283,13034,11491,10735,10216,7647,3802,4312,5332,6619,8137,8405,7108,6342,6581,6868,6855,7119,6609,6836,8108,9424,9163,9187,9174,9203,9188,9407,9428,9192,9179,9177,9682,9690,9693,8928,7895,7358,7374,7899,8662,8399,7883,6871,6876,7892,8671,
+47626,32580,33341,33077,33081,32828,32323,32310,32046,31760,31040,30035,26934,11540,8460,8724,9497,10543,10757,9249,8218,7975,8205,8459,9763,10530,10515,10518,9746,9731,10011,9484,8210,7931,7971,7702,7933,8449,9753,10260,9494,5407,3342,5104,8177,9990,8446,7931,8973,10263,9729,7685,6174,7467,9759,9762,8219,8207,8693,8685,9454,9189,9433,9439,10461,11735,11725,11746,13290,15367,17147,18183,17605,17375,18386,17869,17130,18917,18685,17172,15650,14105,12830,10774,9769,8733,4623,3859,3853,3848,3850,4599,5378,6652,8194,8179,8458,7935,8198,7941,8424,8666,9217,8957,8723,9481,9775,9247,8954,9227,9749,9993,9998,9993,8459,7179,6673,6923,7925,8204,7950,7431,7165,7681,7951,8470,8975,8975,
+47607,32285,33047,32778,32774,32526,31759,32006,31999,31465,30483,29485,22001,9925,9667,8636,8898,9430,8876,9421,9658,10191,10419,10676,10699,10436,9142,8112,7604,7332,7097,7600,8379,8614,9683,10435,10151,9897,10685,10426,6575,5559,5552,7849,9386,8884,6055,6050,8375,9145,6818,6828,9413,10195,9913,9672,8393,7392,8160,7401,7939,8194,7922,8438,8960,10234,11772,11790,13332,16447,17966,19002,18418,17416,16377,15338,16661,18443,18457,17679,15846,13502,11447,9388,8380,5029,2456,3766,4525,4262,3750,4509,4520,5795,7347,8870,9404,8881,8635,8888,8862,8589,8627,8882,9158,8886,7372,6322,6304,6311,6846,6576,6842,7093,7351,8639,8647,8646,9135,8899,8649,8641,9153,9409,9169,9169,9169,8654,
+47355,31527,32799,32526,32529,32279,31768,31498,31751,30964,29471,27695,17903,9433,9179,8150,8415,9462,9677,9964,10199,9197,8401,8401,8932,8416,8149,8918,9437,9426,9698,9693,10202,9671,9960,8656,7605,7605,8909,7367,4036,5844,6605,9931,8898,6857,8133,10188,6344,6847,8608,9632,10177,9683,9140,9413,8639,7623,7092,8131,6859,6597,7105,7363,7632,8646,10700,12002,13290,15370,17396,17660,17083,16077,14011,12968,15330,17110,17379,16864,14804,11456,9148,8882,7614,2721,2215,2741,4781,4513,4260,6824,6840,8111,8377,8106,9157,7606,6593,6591,7077,7573,7613,7096,6349,6590,7638,8390,8632,8641,8655,8388,8389,8900,9413,8902,8397,8906,9395,9153,9161,9409,9923,9407,8648,8399,8908,8910,
+47097,31526,32800,32525,32265,32013,31503,31488,31234,30186,28693,25366,14293,8380,8892,9406,9921,9946,9906,9677,8886,8651,8884,9142,10186,10443,9908,9656,9663,9650,7610,7346,7610,7589,7625,7865,9123,10157,8631,4524,4264,3766,8380,9141,6314,7101,8897,10956,6091,6625,9438,10198,9702,9459,10205,10476,9960,9198,7386,7652,8177,6113,6361,6620,6629,6872,8664,10742,12548,14122,15117,15639,15577,15093,14307,13258,14332,15596,15614,14837,13296,9941,8152,8905,8914,2479,2225,2757,4553,4288,4802,5292,5050,6066,7874,8377,7879,7099,6851,7107,6826,6809,6332,6583,7381,8648,8926,8907,8375,9142,8902,8889,9656,9399,9912,9658,9923,9922,9641,9658,10176,8890,6832,6831,7097,6847,7101,6844,
+47094,31517,32797,32516,32254,32000,31500,31229,30459,29667,27914,23044,11455,9149,10171,9403,8626,8649,8088,8379,9646,9927,10152,10150,9399,8370,7834,8346,7837,7825,8612,8867,8609,8850,9402,9123,9613,10650,6036,4241,3978,4766,7325,6536,6285,7318,6798,6020,8093,6313,7073,8891,9675,9425,9399,9400,9664,9663,8102,6569,7869,6326,5799,6314,6323,6058,6321,8138,10462,12029,13281,14058,13472,12980,13482,13222,13002,13489,13764,12721,10165,8351,8113,11451,9924,3225,1947,2992,4006,3738,3237,4232,5008,4739,6034,8336,8867,8858,9124,8357,8072,8317,8607,8344,8635,9133,8646,8376,8622,9130,9411,9409,8627,7085,6827,6830,7607,8117,8094,7345,7083,7078,8103,9387,9652,9399,9143,8629,
+47106,31042,32835,32303,32299,32046,31802,31019,30251,29195,27188,21548,10985,9446,8923,9196,9440,9984,9936,9452,8924,9458,9428,8662,8676,8681,8405,8919,8414,9436,9193,9702,9181,8652,9458,10209,10449,9949,4565,2776,2765,5347,5591,6867,7136,6115,6876,5330,7143,8686,5338,5861,6903,8712,8948,7929,7671,8183,8941,7150,6387,7158,5352,5867,6394,6123,5863,6654,8201,9515,10520,11300,11227,10996,11491,11744,11271,11251,11529,9971,7935,7651,9722,12797,9722,4050,3290,2786,3808,3543,4067,5326,6629,6612,6627,7900,9967,9190,8940,8691,8413,8659,8691,8171,7953,8185,8984,9226,8701,8700,7948,7173,7165,7930,7930,8445,8193,7419,7658,7675,8191,9468,9727,9471,9217,9222,8970,8714,
+47102,31288,32311,32033,31775,31524,31279,30757,29738,28167,25900,18721,9706,9200,9700,10492,10473,9991,8404,7917,7642,7924,8407,9441,9710,9453,9178,10203,10986,12262,12531,11502,9431,8383,7387,7627,8888,9158,4796,2236,5818,8644,9417,10698,9932,6342,5554,5296,3777,7878,8118,4283,3521,4295,5043,6350,7111,7114,7867,7875,6090,7643,6613,4299,5347,5842,6096,5869,6131,6917,7662,8694,8886,9174,9413,9656,9176,8893,8383,7597,7367,8886,10958,13008,8923,2996,4553,3014,3267,4276,4283,4539,5067,6082,7380,7113,9438,9693,9696,9443,8651,7345,5834,6080,6372,6612,6628,6616,6358,6614,7399,8426,9178,8915,8918,9421,9435,8920,9165,9427,9178,9426,9431,9435,8919,8923,9440,9444,
+47096,30749,31514,31753,31497,31242,30996,30472,29462,27891,24848,16882,9916,9410,8629,8647,8631,8400,8615,9677,9654,9689,10167,10426,10697,9152,8622,9908,11204,14031,14820,14056,13520,13517,11747,10188,8114,7871,7105,6856,8882,10440,11463,11202,11214,9422,5542,5542,4281,4295,8132,6614,4824,5354,5336,4839,5340,6120,7126,8161,7147,6118,7397,4560,3553,4809,5589,5618,5887,5912,6141,6663,7112,7916,7644,7888,7661,7389,7146,7644,8952,10473,11503,13555,8946,2487,4552,5064,2495,3003,3258,3737,4268,5542,7350,6822,7609,8885,9659,9412,7585,6277,7080,7843,8127,8626,8646,7860,7864,8372,8900,8644,8375,8628,8893,8885,9411,9407,9400,9402,9161,9920,9667,9413,8645,9167,8401,6856,
+46589,29982,31521,31244,30990,30476,30233,29700,29212,27632,24079,15330,9393,8880,8868,9654,9891,10175,9876,9910,9114,7865,7565,7830,8356,9127,8852,9368,10919,11428,12212,12987,13479,13738,13504,12199,7551,5504,6803,8863,8586,8076,9613,10131,10917,10153,8335,6530,3976,3466,4493,7857,6836,5824,7861,8380,7873,7619,8128,8393,8157,6364,5845,6358,3540,3770,4290,4832,5352,5636,5609,5616,6058,6347,6330,6577,6611,6602,6867,8636,10447,11459,12227,13001,8389,1930,3492,6309,3219,1920,2952,3700,4750,5515,6289,7566,7840,8606,9633,8617,8073,8305,8340,7817,7840,7822,8100,8341,8601,8854,8863,9118,8854,8850,9114,9622,8347,7055,6276,6036,6296,6810,7587,7334,6562,6317,6574,8108,
+46592,30255,31275,31002,30739,30487,30246,29458,28967,27139,22827,12829,8961,9737,9981,9746,8960,8985,9200,7948,8187,9249,8956,8960,11787,12560,10497,10495,11783,12289,11788,11791,12288,12797,13339,13064,11254,6399,3841,4613,6903,9481,7671,9469,10500,9737,7667,5866,4352,3325,3301,5112,7677,5125,4851,7167,7677,7427,7158,7160,8191,7939,5362,6644,5887,4066,3822,4361,4616,5157,5387,5908,6102,6650,6370,6363,6908,7150,8447,9712,10495,11763,12280,13050,8965,2526,2801,6138,4852,1507,2803,4069,5370,5363,5368,6641,8704,9465,9211,8456,8171,8143,8436,8170,8202,8694,8464,7419,6909,6903,6923,7948,8447,7671,7935,6647,6915,7932,8437,7932,8194,8957,8448,8194,8190,8196,8716,8964,
+46585,29986,31009,30735,30214,29703,29724,29195,28445,26096,21000,11503,10200,9944,9157,8149,8134,7901,8368,9157,9651,9934,10159,11957,12735,12740,10675,9384,10161,11698,11969,11458,10931,10927,11472,11969,11693,10937,6320,5554,6310,8895,8111,8104,9653,9659,5013,4753,4775,3229,3213,3247,5057,7640,5047,4041,5320,7134,7898,7900,7907,7654,6616,5327,6884,5318,4043,4585,4328,4350,4843,5369,5820,5858,6350,6340,7388,8653,9955,9937,11232,11726,12234,12745,8650,2198,2223,5320,6325,2457,3506,4525,4800,5561,6339,6325,7110,8382,8638,9676,9136,8345,9146,8108,7366,6836,5828,6069,6587,7348,6858,6859,6594,6586,7366,8388,8909,8906,8895,8639,8659,9673,9169,9168,9167,9167,9434,9436,
+46585,29723,30736,30469,30207,29437,29453,28925,28432,25571,19705,10717,9166,8918,8650,9186,9174,9721,9171,9195,9182,9981,11227,12250,12262,12012,9950,9946,11237,10974,11501,11503,11231,10972,10998,10984,11222,11749,10472,8162,6094,7905,9175,8141,8920,9437,6850,4273,4813,3543,3534,3296,3044,5368,7402,5360,3547,4069,6115,7912,7407,7409,7658,5597,6129,6620,5602,5879,5107,4877,5113,5373,5823,6366,6872,7384,8945,9704,10741,10982,11510,11499,12530,12531,8435,1984,2772,4841,7389,4042,3536,4520,5811,5800,6064,7340,6587,7605,7864,9417,10159,9632,9916,8880,6851,6575,7368,7613,8389,8383,8150,8154,8141,8907,9167,9162,9171,8399,8385,8648,8923,9422,9431,9432,8656,8663,8668,8409,
+46334,29221,30491,30230,29966,29195,28953,28425,28185,24816,17924,9452,9190,9450,9437,9967,9432,8173,7626,8418,7893,9461,12250,12242,11486,10981,10203,9945,11492,11226,10979,11241,11227,11219,10736,10465,10449,10210,9959,8164,5581,5860,9432,7369,7885,8662,7876,4273,4049,4821,3772,4037,3018,2761,4035,6626,5853,3041,2776,5339,6637,8176,8169,6369,4850,6364,6633,5885,6136,6159,6399,6662,6603,7399,8423,9192,9976,10472,11257,11500,11508,11750,12514,11738,7886,2215,2736,4294,7627,5321,3276,4017,5314,5559,6081,7354,7112,7618,8134,8659,9397,9893,10441,9923,9188,8912,8945,8420,8158,8408,8679,8423,8671,8925,8926,8922,9445,9189,8915,8927,9197,9187,8167,6627,6368,7149,7918,7141,
+46331,28960,30489,30221,29698,29182,28685,28415,27662,24299,16373,9954,9176,8932,8668,8177,8415,8952,9427,9196,10210,12039,12770,11989,11497,11242,9952,9435,10984,10973,10727,10731,10462,10712,10741,10209,10192,10207,9699,8420,7633,7648,10717,7631,9945,9694,7369,4021,3028,4045,4277,4039,3786,3022,2748,3276,4810,5074,3773,2745,4293,6601,7363,7875,5833,5555,6591,6099,6103,6641,7398,7666,7607,8402,9429,10195,10468,11220,11483,11468,11737,11983,12238,11471,7892,2218,2998,4304,6857,6345,3028,4052,5095,5340,6375,7134,7153,7659,8175,8447,9182,9421,9711,9700,9211,8425,8447,8177,8431,8424,8448,8707,8179,6376,6635,6377,6898,6891,6878,6629,6902,6636,6898,7923,8691,9214,9216,8697,
+46330,28704,30231,29967,29439,29181,28936,28151,27141,23528,14819,9171,8390,8910,9148,9424,9661,9685,9392,8903,10432,11747,11966,11704,11207,10950,10434,10174,10442,10429,10437,10440,10174,10679,10963,10433,10417,10435,9930,8906,7860,9413,9399,7602,10428,9660,7597,5281,3512,2726,3976,4735,3704,3465,2433,2457,2451,4005,5272,3475,2203,3996,5787,6808,6565,4236,5791,7348,6581,6610,7367,7886,8351,9400,10166,10674,10945,11187,11197,11694,11963,11953,11436,10925,8614,3187,3436,4474,6268,7043,2977,3746,4792,4524,5045,6578,7367,7361,7621,8149,8886,9382,9417,9409,9176,8648,8672,8140,8911,8902,7899,6613,6607,7111,8141,8653,8917,8400,7360,7364,7389,8145,8918,8917,8918,8926,8928,8922,
+46332,28712,29721,29455,29189,29193,28945,27649,26377,22250,12252,8660,9683,9705,9699,8691,8417,8181,8148,8682,9184,8957,10205,10712,10727,10471,9188,10208,9960,9949,9956,10469,10461,10196,10484,10211,10192,10205,9960,8937,8417,9453,7640,8928,9952,8927,7887,5825,4054,2768,2770,3543,4309,3800,3262,2518,3017,3284,3528,4809,4308,3024,4300,6098,6627,5067,4822,7147,7405,7165,7407,8432,8887,9679,10466,10713,10983,11227,11242,11228,11242,11749,11238,10476,8166,3260,2746,4559,5840,7882,3264,3510,4555,5065,6102,6608,7137,7128,7133,7402,8140,8639,8926,8660,8681,8156,8181,8416,8929,5326,5098,7925,8678,8670,8670,8921,8678,8676,8920,8661,8943,9444,9195,8940,9198,8948,9217,8955,
+46066,28702,29710,29438,28917,28923,28163,27125,25595,20958,11989,9685,9158,9168,8900,9179,8649,8156,8638,10198,8392,5847,6840,8894,9417,9416,9414,9923,10185,9917,9930,10444,10434,9652,9427,9669,9651,9920,9671,9160,8895,8391,7350,9926,9928,9160,8379,6832,5573,4283,2726,2474,2740,2497,3242,3018,2748,3013,2996,4019,5835,4808,3516,4795,6605,6333,4034,5079,7139,7666,7908,8421,8878,9661,10191,10437,10972,10705,10975,10959,10975,11740,11227,10457,8409,3761,2472,4769,5782,7829,4262,3499,3769,4022,5053,6072,6345,6595,7367,8148,8887,9388,8908,8382,7120,6599,7652,8144,6345,4031,7652,8935,8403,8398,8653,8648,8147,8921,9162,9157,9440,9684,9181,9183,9434,9179,9441,9182,
+45816,28448,29715,29451,28933,28423,27923,27137,25351,19425,10194,8658,8654,9176,9680,9442,8908,9957,9923,9945,10190,7130,4786,7360,8649,8392,9161,9672,10449,9924,9937,10196,10185,9404,8919,9163,9401,9671,9427,9168,9417,6855,9156,9413,9414,9159,7867,7607,6594,5847,4841,2806,2551,2818,3312,3071,3573,3060,3314,4328,5108,6387,5353,3553,3819,5854,5080,4075,6387,8207,8200,8457,8913,9440,9970,10727,10743,10476,10745,11237,11250,11759,11247,10481,8178,4307,3033,5628,6149,7675,5090,1963,2749,3511,5056,6079,6347,6860,7111,7639,9148,8618,7625,5559,6091,6853,6874,7373,5572,7110,8670,8670,8400,8395,8394,8388,9171,8403,6848,6337,6356,6348,6612,7641,7891,7641,6609,6606,
+45820,28456,29464,29202,28942,28431,27934,26891,24850,17656,9200,8946,9446,10222,10211,9975,10209,10490,10202,9968,10471,8955,4563,5342,7650,7394,8162,9445,10221,10212,9964,9960,9954,9944,8950,8936,9177,9187,9198,8941,7905,7659,9702,8678,9188,8676,8412,8155,7138,5855,5577,4567,2779,3048,3283,3564,3553,3555,3289,4060,4075,6900,6639,5871,3061,3559,5359,4613,5386,7455,8463,8720,8931,9713,9986,10482,10492,9964,10745,11246,11256,11249,11251,10484,7929,4826,3285,6353,6855,7857,5315,1470,3029,4045,5334,5587,5345,6369,6878,6637,7632,6588,5594,5325,7399,6361,7153,7398,7649,7134,6380,6889,7649,7893,7638,6863,6617,6867,7121,7632,8420,8670,7909,7138,7133,7650,7906,8414,
+45815,28193,29462,28941,28676,28419,28175,26879,24062,16082,9147,9157,8645,8916,8645,9694,10441,10981,10942,10451,10440,9436,5290,4013,6328,6838,7352,8895,9928,10178,9930,9926,9665,9655,8910,8378,8878,8635,9672,7876,6586,8389,8377,7606,8378,8632,8873,8610,7342,6613,5335,4860,4607,3068,1759,3317,4074,4596,4074,3294,4327,5869,7408,6380,5870,3030,4828,5103,4333,6663,8453,8706,8651,9180,9715,10221,9977,10215,10745,10729,10996,11248,11252,10480,7918,5332,4304,6629,3828,4582,6121,2449,3494,3742,4776,5799,6582,6585,6834,6337,7075,7568,7345,6304,5817,6058,6854,6845,7361,7365,7384,7386,6862,6337,6084,6852,7890,8399,8649,8389,8409,8659,8671,8670,8664,8925,9182,9179,
+46063,28170,29696,28664,28144,28143,27897,26341,23263,14497,8842,8841,8832,9360,10635,10656,10115,10658,10875,10897,10623,9100,6242,3934,5482,6762,6765,8050,9087,9854,10117,10118,9597,9077,9099,8052,8039,8822,9601,7030,7281,7549,7544,7543,8576,9094,8825,8566,7812,7350,6327,5341,5347,4322,3263,1987,2482,2740,3253,3248,3778,4813,7119,5829,7629,5305,4021,5589,4055,5350,8426,8941,8373,8638,9170,9415,9434,9926,10452,10430,10953,11465,11729,10193,7628,5808,5559,6586,1464,1432,4012,2403,3443,3950,4983,5497,6024,6542,7046,6550,6517,7014,6530,5231,6029,6525,6558,7057,7570,8083,8098,7844,8089,8083,8342,8080,8348,8602,8337,8073,8612,8858,8868,9382,9376,9377,8862,8346,
+46071,28183,29188,28670,28153,27894,27393,25841,22255,12223,8894,9159,9927,10966,10697,10459,10180,10464,10425,10705,10174,9424,7605,4015,4535,6069,5559,6581,7872,9665,10181,10182,9405,9141,9678,8635,7858,9160,7888,7382,7377,7382,7369,8398,8656,8914,9163,9163,8146,7906,7117,5847,6356,5855,4540,3016,2486,3001,3511,3764,3007,3779,5823,6082,6342,7361,4275,4299,3786,4061,7130,8675,8630,8900,8917,9154,8915,9150,10196,10433,10959,11473,11729,10190,7885,6073,6339,5565,1733,1973,3783,2214,3778,3767,5053,5566,5832,5835,6338,7128,6581,6307,5304,5037,6090,6585,6874,6342,6854,7625,7900,7899,8143,7107,7883,8906,8917,8397,8392,8643,8925,8913,8666,8408,6606,6347,6863,7118,
+45813,27660,28681,28673,28416,28155,27396,25329,20720,11720,9157,9929,10940,10429,10425,10446,9909,10197,10421,10953,10425,9680,8370,5288,4273,5791,5286,5801,7092,9152,9923,9923,9912,9392,9680,9146,8877,8380,7351,8121,15827,26855,20978,10435,8118,7855,7592,7333,6833,7613,7071,6075,6338,6881,5564,4293,3261,3792,4799,4802,3270,3530,5320,7387,5341,7385,6354,3555,5099,3829,5097,8187,8396,8671,8695,9176,8933,9171,10204,10443,10964,11468,11463,9932,7890,5556,6085,3512,1722,3243,3479,2701,4011,4002,5028,5536,5808,5799,6049,6577,6289,6023,6043,5521,5783,6558,6816,6303,7075,7336,7862,8115,8618,8083,7841,7583,6566,5780,6555,6811,7086,7336,6570,6828,7599,8620,9145,9142,
+45818,27417,28433,28169,27917,27660,26640,24577,19203,9946,8925,11001,10733,9966,10471,10486,10209,10496,10464,10999,10464,9716,8660,6858,4052,5060,5069,5576,6614,7899,9698,9958,10196,9162,9445,9426,8646,7120,9178,7382,29215,22001,20486,18441,8140,8139,6853,5820,5842,6879,7383,6384,6390,7178,6637,5114,4587,3055,3545,4062,3562,3561,4320,7669,4854,5869,7663,4601,4608,3600,3065,6662,8414,8422,8446,8672,8699,8941,10237,10736,10999,11509,11503,10217,7158,5343,5868,2527,1767,3294,2513,2500,2781,3548,4575,5858,5874,5871,6122,6389,5595,5326,6119,5855,6347,6888,6892,6618,6638,7412,7676,8185,8176,8420,8686,8167,7668,7398,7400,7398,7927,7410,8185,8701,8959,9210,9477,9474,
+45556,27644,28665,28403,27375,27117,27123,24025,17097,9894,11178,11185,10657,10143,10654,10930,10910,10681,9882,10157,9876,9901,9361,8067,4485,4460,4983,4725,5508,6799,8856,9894,9876,9098,9901,9362,7036,7566,8853,7822,17338,32741,32021,15544,8834,10111,9598,8309,6779,6035,6806,6326,6338,7126,7093,5576,4791,3525,2202,1964,3012,3781,2995,6350,6360,4805,7119,7134,3026,3559,3802,5085,7875,8654,8432,8393,8417,8663,9958,10960,11730,11464,11712,9916,6066,6040,5284,1700,1974,2735,1667,2156,2684,2940,4736,6531,6548,5257,5257,5530,5759,5491,6282,5762,6532,7056,6780,6269,6025,6800,6809,8091,8086,8063,8342,8082,7838,7821,7828,8343,8877,9134,9137,9136,9143,8875,9146,8886,
+45568,27694,28703,27928,27669,26900,21526,16389,13325,11270,11534,12060,11527,11282,11017,11032,11011,10781,10236,9744,9471,9750,9218,8439,6152,4092,5122,4605,5123,5889,7174,8208,8451,8445,8213,8456,8182,7683,7430,7941,8196,11012,11786,8451,9218,9473,9730,9221,7939,6918,6641,6137,6399,7182,6906,6409,4858,4860,5106,2799,2558,2812,2538,3569,6912,4595,5105,6920,5131,4637,4104,3846,7157,9203,8450,8425,9217,9709,9990,10490,11519,11774,9981,7420,6403,6640,3830,1527,3069,2797,1520,2285,3069,3574,4600,6134,7171,5892,5116,5642,6895,6626,6140,6392,6915,7159,6924,6401,6652,6910,6412,6915,7939,7930,8448,8448,7687,7678,7934,7424,8458,8965,9226,9220,9230,9224,8975,9235,
+45811,27144,28674,28660,26094,17105,11976,11453,11723,11714,11457,11463,11191,11200,10943,10696,10932,10445,10416,9931,9403,9420,8385,8370,6594,3762,4785,5301,8129,6583,7868,8645,8374,6570,4023,4781,6301,7088,7338,7342,8104,8098,8358,8101,7844,7847,8103,8100,8361,7356,6581,5571,6604,7383,7381,6368,4817,5077,5332,2242,1746,2518,3012,3016,5593,5065,4029,4825,5597,5346,4822,3025,5062,9420,9180,8904,9697,9940,10214,10977,11233,8401,5575,5317,5831,6576,2727,1716,2752,2226,1447,2187,3225,3478,4502,6041,7075,6821,5533,5799,6544,6283,6051,6551,6566,6816,6580,5798,6558,7332,7342,7343,7853,8619,8365,8368,8380,8122,8119,7863,7869,7350,6587,6839,7357,7099,7096,6847,
+45307,26908,28694,25347,15610,11506,11765,11243,12025,12019,11511,11529,11261,11520,11005,10758,10987,10488,10452,10476,10202,9444,7642,7628,7905,4308,3538,5078,6876,6873,7640,8937,9435,7381,4322,2516,2507,3027,3539,4316,4823,5844,7383,7896,7378,7904,8678,8422,8172,7659,6620,7926,8449,6906,7409,6140,5107,6641,3809,1498,1517,2031,2780,3047,4081,6895,4834,5112,6910,8446,6901,3566,2784,7905,10746,10217,10744,10985,11255,10222,7661,6117,6121,6121,5356,4576,2022,2029,2540,2528,2282,2025,3066,3835,4598,5619,6654,7416,7420,6654,6125,5341,6905,6122,5891,6139,6416,6146,6135,7166,7688,7690,8194,8962,8442,8190,8452,8199,8192,7938,7945,8189,7945,8196,7688,7177,7684,7693,
+45555,26387,21496,14049,10970,11739,11746,11485,11489,11481,11217,11222,10962,10974,10709,10720,10952,10463,10173,10193,9919,9166,8132,6836,7364,5305,4018,4274,5300,6584,7351,8131,8117,7087,6326,4265,2729,2477,2212,1963,2214,2732,3246,4023,4278,4290,5316,6081,7114,7901,8659,11499,10486,6387,6893,5873,6387,6125,2271,984,1509,1758,3288,3809,3045,5346,6874,4315,5083,6891,7658,4834,3031,6356,11507,11768,12036,10992,7919,5336,4586,4829,5350,5342,3535,1974,1222,2002,3287,3795,4058,2741,2492,3003,2744,4792,6089,5827,7631,6862,6344,6322,6862,5056,5336,5837,6121,6107,6098,6871,7396,8175,8416,8674,8665,8162,8418,7912,7902,8157,7903,7895,8418,8927,8677,8423,8930,9450,
+44026,15106,11497,11504,11756,11759,11250,11255,11249,11245,10993,10995,10998,11006,10242,10516,10750,10254,9978,9473,9464,9226,9216,8693,7683,6139,4596,4852,4339,5372,5882,7175,7930,7678,7671,6642,5106,3573,2792,2028,2285,2292,2025,2277,2273,1771,2028,2018,3318,6130,8934,8936,7918,7669,7149,6376,7141,4572,1487,974,2025,3314,2802,2289,3066,3061,6141,6658,4608,5129,6919,6920,4345,4589,10741,13563,11764,7129,6112,5842,5600,5331,5601,3802,1752,1734,1494,2537,3297,4313,5856,4583,2797,2530,3307,3805,5876,5357,5619,6647,6385,7141,5619,4586,5116,5869,6148,6135,6127,6642,7421,8195,8182,8446,8441,7675,8188,8192,8186,8187,8443,8442,8446,8700,7937,9216,9220,8970,
+42212,10164,11437,11442,11179,10673,10675,10677,10412,10662,10404,10403,11180,10934,10421,10430,10667,10166,9890,9378,8853,8874,8603,8589,9120,6801,4482,4483,3197,4233,5769,5007,4231,5779,7059,7568,7058,4750,3200,2689,2177,2441,2430,2682,1913,1919,1917,1904,1926,2971,5816,8642,8649,7363,6848,7873,5814,3242,1435,1693,1712,3776,3517,1726,2255,3016,3276,6112,6623,4575,6366,7904,6610,3785,8155,12013,6865,6073,6586,6837,6592,6844,5572,2225,1447,1676,1684,2217,2713,4258,6304,5754,4214,3947,5244,4462,4224,5505,5513,5773,6541,6276,4487,4743,4755,5513,6046,6034,6287,6026,7064,8092,7825,8089,8342,8086,8088,8347,8084,7574,7831,8344,8350,8343,8607,8347,6549,6557,
+42480,9940,10955,10951,10946,10955,10181,10184,10434,10183,10198,10463,10708,10195,10187,9941,10434,9934,9919,9671,8887,8645,8122,8112,8898,7611,4779,4271,5040,6336,8905,8405,6850,5057,4539,5307,6849,6591,4534,3511,2740,2489,2734,2729,1962,1967,1966,1961,1722,1975,2758,5325,8165,8166,8162,7130,3529,1991,2242,1725,2510,3798,4566,4309,3042,2518,2264,3043,5870,6656,4592,5871,8687,5862,4833,6111,5613,5857,5846,6090,6356,6346,4301,1470,1729,1717,1733,2253,3015,5329,6365,6354,7127,7374,7385,6860,5073,5579,6616,6357,6870,5062,4044,4811,4567,5330,6114,6110,6612,6097,6622,7656,8413,8160,8157,8676,8160,8419,8162,7901,8165,8163,8168,7911,8678,8417,7650,8428,
+42238,8967,10501,10758,10754,11019,10758,11012,10753,10491,10244,10255,9731,9479,9726,9990,9966,9718,9708,9716,8938,8438,7916,7910,8174,7669,6903,4605,5373,6152,7686,9235,9218,9482,8709,7167,5888,5368,6141,5374,4100,2317,2299,2806,2555,2561,2043,2037,1547,1549,2284,2780,4849,7181,6150,4353,2545,2028,3056,3311,3066,4090,4852,5094,4084,2793,3068,2822,2825,5909,6660,4101,6671,7945,4585,5606,6654,6644,7164,7407,6646,6138,3329,1530,2049,2037,2047,3318,5861,6888,6928,7185,7697,7430,7698,7689,6668,6404,6668,7438,6155,4091,4358,4608,4617,5377,6158,6412,6401,5887,6146,7176,8450,8201,7942,8205,8458,8458,8195,8176,7926,8677,8161,7896,8142,8397,8143,8406,
+41715,7385,8408,8152,8400,8917,9947,10718,10719,10458,10207,10222,9951,9705,9694,9707,9686,9694,9426,9176,8661,8160,7376,7377,7630,7382,7631,5328,4552,5333,7377,9184,9170,9434,9684,9169,8145,7376,5576,5577,5323,3276,2747,2486,2491,2497,2493,2234,2246,1470,2030,2559,3076,3572,3304,3057,2289,3313,3553,4068,3309,3565,4072,4331,4861,4331,3565,2798,2537,2540,5109,5866,4556,7389,6369,4836,6642,5345,4835,4056,2797,2529,2279,2018,2534,2521,2274,5362,7160,7159,7410,7377,7379,7114,7382,7631,6864,7369,7895,7123,4301,4036,4813,4808,5070,5578,6103,6359,6350,5837,6354,6354,7632,8155,7896,7909,8431,8188,7946,7941,8472,8979,8216,8473,7441,7700,8472,7959,
+42228,9703,10982,10724,8921,7124,6610,8666,9435,8912,9422,10213,10198,9947,9935,9688,9665,9420,8383,8135,8136,7890,7363,7107,7355,7107,6846,6085,3511,4293,6592,8909,9157,9422,9419,8643,8647,8909,8394,4804,4290,4806,4028,2994,3254,2745,2486,2485,3010,1994,2265,2536,3317,3055,2788,2274,3029,3798,3540,3541,3809,3041,3555,4330,4596,4328,3571,3056,3315,3054,2798,5358,4835,4843,7151,5861,6388,5349,5351,3800,2533,2264,2789,2790,3048,2781,3814,6378,7396,7649,7393,7359,6844,6581,6590,6841,8379,8886,6328,4020,3763,4529,4534,4781,5048,5556,6078,6077,6324,6066,6073,6078,7106,8140,8390,8139,8139,7886,7891,7621,8149,8141,8148,8148,7634,7633,7890,7895,
+42485,8928,9687,10202,9944,9944,7887,6095,6612,8412,9447,9972,9951,9698,9947,9700,9939,9697,8919,7895,8151,7905,7896,7389,6863,7133,7124,6876,4810,3793,6352,8410,9173,9697,9436,8659,8405,8666,9433,6863,5582,4299,4293,3778,3013,3278,2509,2770,3039,3040,2540,2556,2820,3067,3057,3065,3567,4084,4079,4076,4338,3562,3047,4071,4596,5359,4340,3308,3829,3574,2803,3823,5359,4584,6375,6894,5107,3555,4078,4582,3057,2017,3047,3303,3306,4579,7157,8437,8448,7930,7941,8178,7923,7656,8183,8179,7405,4573,3557,3302,3299,4325,4841,4840,5098,5608,5873,6135,6389,6136,6145,6659,6916,7689,7687,7942,7943,7944,7950,7941,8465,8459,8206,8203,7944,7946,7434,7952,
+42223,7374,8647,8134,8382,8896,9665,9413,6329,5559,7875,9168,9419,9677,9418,9182,9678,9172,8138,7883,8653,8146,7622,7108,6852,7114,7368,6600,6081,4031,5828,7632,8904,9430,9167,8904,8648,8912,9681,7627,8147,7125,4804,4032,4030,3005,2746,3000,2756,2777,2275,2550,3576,3311,2785,3310,4060,4835,4324,4322,5102,4840,3038,3295,3566,4586,5622,5361,4076,3563,3307,3304,4847,5372,5111,7170,6149,4595,4338,4829,3554,2522,3810,3554,5102,8188,7674,5361,4071,3548,4069,4812,4801,4542,4292,3509,2498,1714,2752,3524,3781,4552,5322,5319,5058,5569,5837,5830,5829,5569,5582,6610,6607,7119,7120,7632,7888,7633,7897,7887,8666,8404,8148,8148,7633,7629,7120,7124,
+42483,7638,7886,7894,8399,8399,8401,8918,8923,6615,5076,6880,8937,9970,9716,9210,8931,7394,6369,6879,7141,7402,7651,7135,6887,7142,7142,6625,6365,4827,4569,6630,8160,8940,9193,8933,8672,8680,9706,7909,8423,8934,6619,4049,4047,3795,3793,4053,3285,3543,2773,2519,3029,3546,2784,3331,4345,4867,4865,4865,5129,5382,3835,2802,3072,3318,4870,5379,4617,4357,4098,3581,4083,5630,4342,6398,7175,5372,5118,5109,4097,4603,5117,5115,8446,5856,2510,2265,2004,2260,2515,2254,2253,1995,1741,2760,1746,1477,2770,3801,3543,4828,5349,5088,5086,5596,5603,6373,5857,5337,5091,5858,6629,7137,7143,7398,7396,7402,8431,8679,8432,8171,7915,8429,7918,7399,7405,6894,
+43507,10977,8400,7635,7884,8394,7882,7627,8141,8135,4798,4029,6597,9170,9429,8925,8646,6601,5832,6602,5832,5579,6084,6332,6339,6596,6594,6847,6080,5570,4025,6084,7102,8131,8642,8636,8372,8378,9147,8634,8640,9159,7354,4524,3752,4272,4269,3755,3248,4043,3795,2526,2779,3040,2781,3568,4073,4344,4598,4852,4854,4850,4587,2529,3312,2789,3056,3821,4345,4601,4603,3829,3560,4851,4854,4853,7172,5883,4856,4331,4347,4845,5352,7919,5613,2512,2247,2513,2256,2255,3006,2214,2208,2467,2727,2976,1703,1175,1950,3755,3751,4773,5039,5034,5031,5545,5807,5807,5293,4774,5296,10180,7601,6311,6833,7341,7086,7351,8634,8885,8635,8116,7605,8121,7864,7604,7866,6584,
+45047,16111,11726,8900,7864,8120,7616,7618,7625,8653,7629,4795,4275,5832,8405,9184,9163,8148,6340,5831,4795,4798,5046,5808,6344,6343,6340,6595,5829,5575,4028,5843,6865,7643,8932,9182,8923,8417,8153,9176,8658,9174,9433,6869,6358,6102,4294,3769,4044,4049,3292,3054,2796,2801,2788,3811,4072,4079,4583,4840,4073,4325,5090,3543,2524,2249,2000,1998,2266,2788,4072,4054,3279,3531,5335,4306,6376,6631,5349,5079,5103,5352,6896,6890,2020,2519,1756,1506,2267,2265,2252,2764,2511,3294,3552,3032,1766,1242,1757,3568,4849,4840,4594,4584,4838,5095,5871,5614,4072,3810,15646,23641,14082,8176,6898,7142,7414,7675,8186,7928,8187,8436,7927,8194,8190,7932,7411,6388,
+33992,16340,13751,9633,7571,7826,7576,7319,6808,7320,7324,6297,4487,4758,6304,7853,8354,7850,6554,6048,5270,5013,5259,5253,5521,6032,5774,5773,5775,5775,4741,5523,6547,6806,7841,8091,8090,7837,7316,7311,7316,7572,8604,7324,6801,6801,4995,4214,4487,5273,4292,4055,3812,3817,3808,4839,4844,4843,4836,4830,4574,4571,4826,5093,3808,4050,4049,3792,4055,4063,3806,4299,4291,4539,5071,5327,5852,6368,5850,6105,6124,6632,7915,5075,3539,3526,3025,3536,4047,3787,3769,4008,4006,4524,4518,4258,3499,2979,3236,4017,5554,5290,5815,5548,5291,5032,5549,4773,5034,5547,17133,18951,9656,10686,6578,6571,7095,7356,7096,7102,7624,7882,7892,7642,7639,7127,7120,6870};\r
+\r
+//¼ÆËãͼƬ\r
+int ftt_test(void)\r
+{\r
+       int i,j;\r
+       int mLen = 128, nLen = 128;\r
+       bool flag = 0;\r
+       RK_complex_INT *A_In;\r
+\r
+       //ÅжÏÊÇ·ñÂú×㳤¶ÈÒªÇó\r
+       int M = nexttopow2(mLen);   //Èç¹ûinit_mLen²»ÊÇ2µÄÃݴη½£¬Ôò½«³¤¶ÈÉèΪ²»Ð¡ÓÚinit_mLenµÄ×îС2µÄÃݴη½ÕûÊý£¬M¼´Îª´ÎÊý\r
+       int N = nexttopow2(nLen);\r
+\r
+       mLen = Pow2_table[M];         //еĸß\r
+       nLen = Pow2_table[N];         \r
+\r
+       A_In = (RK_complex_INT *)malloc(sizeof(RK_complex_INT)*nLen*mLen);\r
+\r
+       for(i=0; i<mLen; i++)\r
+       {\r
+               for(j=0; j<nLen; j++)\r
+               {\r
+                       uchar tmp = image[i*128+j];\r
+                       \r
+                       A_In[i*nLen+j].real = (int)(tmp << Q_INPUT);\r
+                       A_In[i*nLen+j].image = 0;\r
+               }\r
+       }\r
+\r
+       //¶ÔÀ©Õ¹ºóµÄÈËÁ³Í¼Ïñ×öFFT\r
+       fft_2D_fixed_point(mLen,nLen,M,N,A_In,0);\r
+       fft_2D_fixed_point(mLen,nLen,M,N,A_In,1);\r
+\r
+       //ÅжϽá¹ûÊÇ·ñÕýÈ·\r
+       for(i=0; i<mLen; i++)\r
+       {\r
+               for(j=0; j<nLen; j++)\r
+               {\r
+                       int tmp = A_In[i*nLen+j].real;\r
+                       if (tmp != result[i*128+j])\r
+                       {\r
+                               flag = 1;\r
+                       }\r
+               }\r
+       }\r
+       \r
+       free(A_In);A_In = NULL;\r
+       if (!flag)\r
+       {\r
+               //printf("correct!\n");\r
+               return 0;\r
+       }\r
+       else\r
+       {\r
+               //printf("wrong!\n");\r
+               \r
+               return -1;\r
+       }\r
+       ;\r
+}\r
diff --git a/arch/arm/mach-rockchip/rk_pm_tests/ft/ft_cpu_tst.S b/arch/arm/mach-rockchip/rk_pm_tests/ft/ft_cpu_tst.S
new file mode 100755 (executable)
index 0000000..75fc8ff
--- /dev/null
@@ -0,0 +1,156 @@
+
+#include <linux/linkage.h>
+#include <asm/assembler.h>
+#include "cpu_test.h"
+
+.text
+
+//r0,array0
+ENTRY(test_cpus_l1_0)
+
+       stmfd   sp!, { r1 - r12, lr }
+1:     mov r0,r0
+       //b 1b
+       mov r4,r0
+
+       test_cpus_l1_loop_200_k
+       test_cpus_l1_loop_200_k
+       test_cpus_l1_loop_200_k
+       test_cpus_l1_loop_200_k
+       test_cpus_l1_loop_200_k
+
+       test_cpus_l1_loop_200_k
+       test_cpus_l1_loop_200_k
+       test_cpus_l1_loop_200_k
+       test_cpus_l1_loop_200_k
+       test_cpus_l1_loop_200_k
+
+       //test_cpus_l1_loop_200_k
+       //test_cpus_l1_loop_200_k
+       //test_cpus_l1_loop_200_k
+
+
+1: mov r1,r1
+       //b 1b
+       ldmfd   sp!, { r1 - r12, pc }
+l1_test_error:
+       mov r1,r1
+       mov r1,r1
+       mov r1,r1
+       1: mov r1,r1
+       b 1b
+ENDPROC(test_cpus_l1_0)
+
+
+//r0 adr base,r1 end adr,r2 tst date
+ENTRY(test_cpus_l2)
+
+       stmfd   sp!, { r3 - r12, lr }
+1:     mov r0,r0
+       //b 1b
+       
+       mov r4,r0
+       sub r5,r1,#4
+       
+l2_write:      
+       str r2,[r4],#4
+               
+       cmp r4,r5
+       bls     l2_write
+
+       mov r4,r0
+l2_check:              
+       ldr     r6,[r4],#4
+       cmp r6,r2
+       bne l2_test_error
+       mov r6,#0
+       cmp r4,r5
+       bhi l2_tst_end
+       
+       ldr r7,[r4],#4
+       cmp r7,r2
+       bne l2_test_error
+       mov r7,#0
+       
+       cmp r4,r5
+       bls l2_check
+       
+l2_tst_end:    
+3: mov r0,#0
+       //b 3b
+       ldmfd   sp!, { r3 - r12, pc }
+l2_test_error:
+1: mov r0,#1
+       //b 1b
+       ldmfd   sp!, { r3 - r12, pc }
+ENDPROC(test_cpus_l2)
+
+//r0 adr base,r1 end adr,r2 tst date
+ENTRY(test_cpus_mem_set)
+
+       stmfd   sp!, { r3 - r12, lr }
+1:     mov r0,r0
+       //b 1b
+       
+       mov r4,r0
+       sub r5,r1,#4
+       
+mem_set:       
+       str r2,[r4],#4
+               
+       cmp r4,r5
+       bls     mem_set
+       
+1: mov r0,#1
+       //b 1b
+       ldmfd   sp!, { r3 - r12, pc }
+ENDPROC(test_cpus_mem_set)
+
+
+
+//r0 adr base,r1 end adr,r2 tst date
+ENTRY(test_cpus_mem_check)
+
+       stmfd   sp!, { r3 - r12, lr }
+1:     mov r0,r0
+       //b 1b
+       mov r4,r0
+       sub r5,r1,#4
+       
+mem_check:             
+       ldr     r6,[r4],#4
+       cmp r6,r2
+       bne mem_check_error
+       mov r6,#0
+       cmp r4,r5
+       bhi mem_check_end
+       
+       ldr r7,[r4],#4
+       cmp r7,r2
+       bne mem_check_error
+       mov r7,#0
+       
+       cmp r4,r5
+       bls mem_check
+       
+mem_check_end: 
+3: mov r0,#0
+       //b 3b
+       ldmfd   sp!, { r3 - r12, pc }
+mem_check_error:
+1: mov r0,#1
+       //b 1b
+       ldmfd   sp!, { r3 - r12, pc }
+
+ENDPROC(test_cpus_mem_check)
+/***************while for dbg***************/
+ENTRY(rk30_cpu_while_tst)
+stmfd  sp!, { r3 - r12, lr }
+
+1: mov r3,r3
+   b 1b
+
+ldmfd  sp!, { r3 - r12, pc }
+
+ENDPROC(rk30_cpu_while_tst)
+
diff --git a/arch/arm/mach-rockchip/rk_pm_tests/ft/ft_cpu_tst1.S b/arch/arm/mach-rockchip/rk_pm_tests/ft/ft_cpu_tst1.S
new file mode 100755 (executable)
index 0000000..6f96fce
--- /dev/null
@@ -0,0 +1,44 @@
+
+#include <linux/linkage.h>
+#include <asm/assembler.h>
+//#include <asm/memory.h>
+//#include <mach/io.h>
+#include "cpu_test.h"
+
+.text
+
+//r0,array0
+ENTRY(test_cpus_l1_1)
+       stmfd   sp!, { r1 - r12, lr }
+1:     mov r0,r0
+       //b 1b
+       mov r4,r0
+
+       test_cpus_l1_loop_200_k
+       test_cpus_l1_loop_200_k
+       test_cpus_l1_loop_200_k
+       test_cpus_l1_loop_200_k
+       test_cpus_l1_loop_200_k
+
+
+       test_cpus_l1_loop_200_k
+       test_cpus_l1_loop_200_k
+       test_cpus_l1_loop_200_k
+       test_cpus_l1_loop_200_k
+       test_cpus_l1_loop_200_k
+
+
+1: mov r1,r1
+       //b 1b
+       ldmfd   sp!, { r1- r12, pc }
+l1_test_error:
+1: mov r1,r1
+       b 1b
+ENDPROC(test_cpus_l1_1)
+
+
+
+
+
+
+
diff --git a/arch/arm/mach-rockchip/rk_pm_tests/ft/ft_test.c b/arch/arm/mach-rockchip/rk_pm_tests/ft/ft_test.c
new file mode 100644 (file)
index 0000000..9b30a44
--- /dev/null
@@ -0,0 +1,1073 @@
+/* arch/arm/mach-rk30/rk_pm_tests.c
+ *
+ * Copyright (C) 2010 ROCKCHIP, Inc.
+ *
+ * This software is licensed under the terms of the GNU General Public
+ * License version 2, as published by the Free Software Foundation, and
+ * may be copied, distributed, and modified under those terms.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ */
+
+
+/*************************************************************************/
+/*       COPYRIGHT (C)  ROCK-CHIPS FUZHOU . ALL RIGHTS RESERVED.*/
+/*************************************************************************
+FILE           :        ft_test.c
+DESC   :        ft test
+AUTHOR :         xie xiu xin
+DATE           :         2013-7-2
+NOTES          :
+$LOG: GPIO.C,V $
+REVISION 0.01
+#include <linux/clk.h>
+#include <linux/kobject.h>
+ ***************************************************************************/
+#include <linux/string.h>
+#include <linux/resume-trace.h>
+#include <linux/workqueue.h>
+#include <linux/mutex.h>
+#include <linux/module.h>
+#include <linux/syscalls.h>
+#include <linux/init.h>
+#include <linux/interrupt.h>
+#include <linux/delay.h>
+#include <linux/cpu.h>
+#include <linux/slab.h>
+#include <linux/freezer.h>
+
+//#include <linux/dma-mapping.h>
+//#include <linux/regulator/machine.h>
+//#include <plat/dma-pl330.h>
+//#include <linux/mfd/wm831x/core.h>
+#include <linux/sysfs.h>
+#include <linux/err.h>
+#include <linux/kthread.h>
+#include <linux/clk.h>
+#include <linux/suspend.h>
+
+#include <linux/io.h>
+#include <linux/gpio.h>
+#include <linux/lzo.h>
+#include <linux/vmalloc.h>
+
+
+#if 1
+#define ft_printk(fmt, arg...) \
+       printk(KERN_EMERG fmt, ##arg)
+ //KERN_DEBUG
+#define ft_printk_dbg(fmt, arg...) \
+       printk(KERN_EMERG fmt, ##arg) 
+       
+ //KERN_WARNING
+#define ft_printk_info(fmt, arg...) \
+       printk(KERN_EMERG fmt, ##arg)
+#else
+
+#define ft_printk(fmt, arg...) \
+       printk(fmt, ##arg)
+       //printk(KERN_EMERG fmt, ##arg)
+ //KERN_DEBUG
+#define ft_printk_dbg(fmt, arg...) {while(0);}
+
+ //KERN_WARNING
+#define ft_printk_info(fmt, arg...) {while(0);}
+#endif
+
+
+#define MHZ (1000*1000)
+#define TST_SETUPS (4)
+
+//#define ENABLE_FT_SUSPEND_TST   // for ft seting 1.6G volt
+
+#define l1_DCACHE_SIZE (32*1024)
+#define l2_DCACHE_SIZE (512*1024)
+#define TST_TASK_NUM (10*10*3)
+
+
+static unsigned long arm_setups_rate[TST_SETUPS]={816*MHZ,1656*MHZ*0,1608*MHZ*0,312*MHZ*0};
+const static unsigned long ddr_setups_rate[TST_SETUPS]={600*0,800*1,800*1,600*1};
+
+#define L1_JUST (0-10)
+const static unsigned int l1_tst_cnt[TST_SETUPS]={5*10*2+L1_JUST,5*10*2+L1_JUST,5*10*2+L1_JUST,80+L1_JUST};
+const static unsigned int l2_cpy_cnt[TST_SETUPS]={5,5,4,4};
+const static unsigned int ftt_tst_cnt [TST_SETUPS]={1,1,1,0};
+const static unsigned int pi_tst_cnt[TST_SETUPS]={12,12,10,8};
+const static unsigned int stp_task_num[TST_SETUPS]={300,250,250,80};
+
+static u32 ft_end_cnt=0x24;
+
+//
+#define get_setup_tasks(a) ft_get_min(a,TST_TASK_NUM)
+
+//0-7 :test setup1
+#define CPU_TST_L1_STP0 (1<<0)
+#define CPU_TST_L2_STP0 (1<<1)
+
+static DEFINE_PER_CPU(int [TST_SETUPS], cpu_tst_flags)={0};
+static DEFINE_PER_CPU(wait_queue_head_t [TST_SETUPS][TST_TASK_NUM], wait_setups);
+static DEFINE_PER_CPU(struct mutex[TST_SETUPS], lzo_lock);
+static DEFINE_PER_CPU(struct mutex[TST_SETUPS], l2_lock);
+static DEFINE_PER_CPU(u32, sem_setups_cnt);
+static DEFINE_PER_CPU(unsigned char *, lzo_wrk);
+
+struct tst_task_struct {
+       struct task_struct *task;
+       u32 thread_id;
+       int cpu;
+       int tst_step;
+};
+
+static struct tst_task_struct tst_task_date[TST_TASK_NUM][NR_CPUS];
+static struct semaphore sem_setups[TST_SETUPS];
+static int setups_flag[TST_SETUPS]={0,0,0,0};
+
+static struct clk *arm_clk;
+
+#ifdef FT_DDR_FREQ
+static struct clk *ddr_clk;
+#endif
+
+void ft_test_flag_seting(void);
+
+int ftt_test(void);
+int test_cpus_l2(char *data_s,char *data_e,u32 data);
+int Test_mem(void *aligned, unsigned long b_size) ;
+
+int test_cpus_mem_set(char *data_s,char *data_e,u32 data);
+int test_cpus_mem_check(char *data_s,char *data_e,u32 data);
+
+
+#define BUF_SIZE (l2_DCACHE_SIZE*8)// tst buf size
+
+#define BUF_SIZE_M (BUF_SIZE)// tst buf size
+
+static char memtest_buf0[BUF_SIZE] __attribute__((aligned(4096)));
+#if (NR_CPUS>=2)
+static char memtest_buf1[BUF_SIZE] __attribute__((aligned(4096)));
+#if (NR_CPUS>=4)
+static char memtest_buf2[BUF_SIZE] __attribute__((aligned(4096)));
+static char memtest_buf3[BUF_SIZE] __attribute__((aligned(4096)));
+#if (NR_CPUS>=8)
+static char memtest_buf4[BUF_SIZE] __attribute__((aligned(4096)));
+static char memtest_buf5[BUF_SIZE] __attribute__((aligned(4096)));
+static char memtest_buf6[BUF_SIZE] __attribute__((aligned(4096)));
+static char memtest_buf7[BUF_SIZE] __attribute__((aligned(4096)));
+#endif
+#endif
+#endif
+
+static char *l2_test_buf[NR_CPUS]=
+{
+       memtest_buf0,
+#if (NR_CPUS>=2)
+       memtest_buf1,
+#if (NR_CPUS>=4)
+       memtest_buf2,
+       memtest_buf3,
+#if (NR_CPUS>=8)
+       memtest_buf4,
+       memtest_buf5,
+       memtest_buf6,
+       memtest_buf7,
+#endif
+#endif
+#endif
+
+};
+
+// for board init arm rate
+unsigned long __init ft_test_init_arm_rate(void)
+{
+       return arm_setups_rate[0];
+}
+static int ft_get_min(int a, int b)
+{
+       if(a<b)
+       return a;
+       else
+       return b;
+
+}
+void printk_dbg_data(char *buf,int len,char *info)
+{
+       int i;
+       char *buf_s=buf;
+       
+       printk("%s\n",info);
+       for(i=0;i<len;i++)
+       {       
+               printk("%x ",*(buf_s));
+               buf_s++;
+               if(i%40==0&&i!=0)
+               printk("\n");   
+       }
+       printk("\n");   
+
+}      
+
+/*******************************************************************************************/
+static const char pi_result[]="3141592653589793238462643383279528841971693993751058209749445923078164062862089986280348253421170679821480865132823664709384469555822317253594081284811174502841270193852115559644622948954930381964428810975665933446128475648233786783165271201991456485669234634861045432664821339360726024914127372458706606315588174881520920962829254917153643678925903611330530548820466521384146951941511609433057273657595919530921861173819326117931051185480744623799627495673518857527248912279381830119491298336733624406566438602139494639522473719070217986943702770539217176293176752384674818467669451320005681271452635608277857713427577896091736371787214684409012249534301465495853710579227968925892354201995611212902196864344181598136297747713099605187072113499999983729780499510597317328160963185";
+
+int calc_pi(void)
+{  
+       int bit = 0, i=0;
+       long a=10000,b=0,c=2800,d=0,e=0,g=0;
+       int *result;
+       long *f;
+       int len=0;
+       
+       char *pi_calc,*pi_tmp;
+       char *pi_just=(char *)&pi_result[0];
+       size_t pi_just_size=sizeof(pi_result);
+
+        result = vmalloc(10000*sizeof(int));
+        if (result == NULL)
+               return -ENOMEM;
+        
+        f = vmalloc(2801*sizeof(long));
+         if (f == NULL)
+                return -ENOMEM;
+         
+        pi_calc= vmalloc(1000*sizeof(char));
+         if (pi_calc == NULL)
+                return -ENOMEM;
+
+        for(;b-c;)
+                f[b++]=a/5;
+        for(;d=0,g=c*2;c-=14,result[bit++] = e+d/a,e=d%a)
+                for(b=c;d+=f[b]*a,f[b]=d%--g,d/=g--,--b;d*=b);
+
+       //ft_printk("lzo1 calc_pi end\n");
+
+       pi_tmp = pi_calc;
+        for(i = 0; i < bit; i++) {
+               len += sprintf(pi_tmp+len, "%d", result[i]);
+               //ft_printk("%c", pi_calc[i]);
+               //ft_printk("lzo1 calc_pi end\n");
+        }
+
+       for(i = 0; i < bit; i++) {
+        //  ft_printk("%c,%d", pi_calc[i],result[i]);
+          }
+       
+       //ft_printk("lzo1 calc_pi end,i=%d,len=%d\n",i,len);
+       
+       if (strncmp(pi_just, pi_calc, pi_just_size) == 0){      
+               vfree(result);
+               vfree(f );
+               vfree(pi_calc);
+               //ft_printk("calc_pi ok\n");
+               return 0;
+       } else {
+                       vfree(result);
+                       vfree(f );
+                       vfree(pi_calc);
+                       ft_printk("calc_pi error\n");
+                       return -EPERM;
+       }
+
+}
+
+/************************************l1 tst***************************************/
+void test_cpus_l1_0(u32 *data);
+void test_cpus_l1_1(u32 *data);
+
+void ft_cpu_l1_test(u32 cnt0,u32 cnt1)
+{
+       //u32 cpu = smp_processor_id();
+       int test_array[100];
+       int i;
+       
+       for(i=0;i<cnt0;i++)
+       {
+               test_cpus_l1_1(&test_array[0]);
+               barrier();
+               
+       }
+       //if(cpu==0&&cnt0>40&&cnt1>40)
+       //ft_printk(".");
+       for(i=0;i<cnt1;i++)
+       {
+               test_cpus_l1_0(&test_array[0]);
+               barrier();
+               
+       }
+}
+
+/********************lzo***************************************/
+
+#define SIZE_M (1024*1024)
+#define lzo_dcmp_size (l2_DCACHE_SIZE*2)
+
+int  lzo_cmp_decmp_tst(unsigned char *cmp_in,size_t in_size,unsigned char *cmp_out,size_t out_size,void *wrkmem)
+{
+       
+       //u32 cpu = smp_processor_id();
+       int ret;                
+       size_t cmp_len,decmp_len;
+       //unsigned char *cmp_sr;
+       
+       cmp_len=out_size;
+       ret=lzo1x_1_compress(cmp_in,in_size,cmp_out,&cmp_len,wrkmem);
+       
+       if(ret != LZO_E_OK)
+       {       
+               ft_printk("lzo1 comp error\n");
+               return ret; 
+       }
+       else
+               ft_printk_dbg("cmplen=%d\n",cmp_len);
+
+
+       decmp_len=in_size;
+       
+       ret=lzo1x_decompress_safe(cmp_out,cmp_len,cmp_in,&decmp_len);
+
+       if(ret != LZO_E_OK)
+               ft_printk("lzo1 decomp error=%d\n",ret);
+       
+       //ft_printk("lzo cmp (%x,%x)\n",decmp_len,cmp_len);
+       return ret;     
+}
+
+/************************************l2 tst***************************************/
+
+static char *test_mal_buf[NR_CPUS]=
+{
+       NULL,
+};
+int ft_test_cpus_l2_memcpy(char *data_d,char *data_s,u32 buf_size,u32 cnt)
+{
+       
+       //u32 cpu = smp_processor_id();
+       int j;
+       int ret=0;
+       int test_size;
+       char *mem_start;
+       char *mem_end;
+       
+       char *cpy_start;
+       char *cpy_end;
+       
+       test_size=(buf_size/(2*l2_DCACHE_SIZE))*(2*l2_DCACHE_SIZE);
+       
+       for(j=0;j<cnt;j++)      
+       {       
+               mem_start=data_s;
+               mem_end=data_s+test_size;
+
+               if(data_d)
+               {       
+                       cpy_start=data_d;
+                       cpy_end=data_d+test_size;
+               }
+               else
+                       cpy_start=NULL;
+
+               for(;mem_start<=(mem_end-2*l2_DCACHE_SIZE);)
+               {
+                       
+
+                       test_cpus_mem_set(mem_start,mem_start+l2_DCACHE_SIZE,0xaaaaaaaa);
+                       test_cpus_mem_set(mem_start+l2_DCACHE_SIZE,mem_start+l2_DCACHE_SIZE*2,0x55555555);
+
+                       
+                       //ret=test_cpus_mem_check(mem_start,mem_start+l2_DCACHE_SIZE,0xaaaaaaaa);
+                       //ft_printk("mem_start end=%x,%x\n",mem_start+l2_DCACHE_SIZE,ret);
+
+
+                       if(test_cpus_mem_check(mem_start,mem_start+l2_DCACHE_SIZE,0xaaaaaaaa))
+                       {       
+                               ret=0xaa;
+                               return ret;
+                       }
+                                               
+                       if(test_cpus_mem_check(mem_start+l2_DCACHE_SIZE,mem_start+l2_DCACHE_SIZE*2,0x55555555))
+                       {       
+                               ret=0x55;
+                               return ret;
+                       }
+#if 1
+                       if(cpy_start)
+                       {       
+                               memcpy(cpy_start,mem_start,l2_DCACHE_SIZE);
+                               memcpy(cpy_start+l2_DCACHE_SIZE,mem_start+l2_DCACHE_SIZE,l2_DCACHE_SIZE);
+
+                               if(test_cpus_mem_check(cpy_start,cpy_start+l2_DCACHE_SIZE,0xaaaaaaaa))
+                               {       
+                                       ret=0x1aa;
+                                       return ret;
+                               }
+                               
+                               if(test_cpus_mem_check(cpy_start+l2_DCACHE_SIZE,cpy_start+l2_DCACHE_SIZE*2,0x55555555))
+                               {       
+                                       ret=0x155;
+                                       return ret;
+                               }
+                       }
+
+                       
+                       test_cpus_mem_set(mem_start+l2_DCACHE_SIZE,mem_start+l2_DCACHE_SIZE*2,0xaaaaaaaa);
+                       test_cpus_mem_set(mem_start,mem_start+l2_DCACHE_SIZE,0x55555555);
+
+
+                       if(test_cpus_mem_check(mem_start+l2_DCACHE_SIZE,mem_start+l2_DCACHE_SIZE*2,0xaaaaaaaa))
+                       {       
+                               ret=0x2aa;
+                               return ret;
+                       }
+                                               
+                       if(test_cpus_mem_check(mem_start,mem_start+l2_DCACHE_SIZE,0x55555555))
+                       {       
+                               ret=0x255;
+                               return ret;
+                       }
+
+                       //cpy_start is aa  mem_start is  55
+                       
+                       if(cpy_start)
+                       {                               
+                               memcpy(cpy_start+l2_DCACHE_SIZE,mem_start+l2_DCACHE_SIZE,l2_DCACHE_SIZE);//aa
+                               memcpy(cpy_start,mem_start,l2_DCACHE_SIZE);//55
+
+                               if(test_cpus_mem_check(cpy_start+l2_DCACHE_SIZE,cpy_start+l2_DCACHE_SIZE*2,0xaaaaaaaa))
+                               {       
+                                       ret=0x3aa;
+                                       return ret;
+                               }
+                               
+                               if(test_cpus_mem_check(cpy_start,cpy_start+l2_DCACHE_SIZE,0x55555555))
+                               {       
+                                       ret=0x355;
+                                       return ret;
+                               }
+                       }
+#endif
+                       mem_start+=2*l2_DCACHE_SIZE;
+                       if(cpy_start)
+                       {
+                               cpy_start+=2*l2_DCACHE_SIZE;
+                       }
+
+                       //if((j%20==0)&&(cpu==0))
+                       //ft_printk(".");
+               }
+               
+       }
+
+       return 0;
+}
+
+
+static int ft_test_mem(void *aligned, unsigned long b_size,int cnt) 
+{
+       int ret=0,i;
+       
+       for(i=0;i<cnt;i++)
+       {
+               ret=Test_mem(aligned,b_size);
+               if(ret)
+               return -1;      
+       }
+
+       return 0;       
+}
+
+/*************************************type0 tst case**************************************/
+void ft_cpu_test_type0(int steps,struct tst_task_struct *task_data)
+{
+       u32 temp=-1;
+       u32 cpu = smp_processor_id();
+       u32 id_start;
+       
+       if(task_data->thread_id==0)
+       {       
+
+               if(steps<2)
+               {
+                       mutex_lock(&per_cpu(lzo_lock, cpu)[steps]);
+
+                       temp=lzo_cmp_decmp_tst((unsigned char *) test_cpus_l1_0,lzo_dcmp_size,
+                       test_mal_buf[cpu],sizeof(char)*BUF_SIZE,per_cpu(lzo_wrk, cpu));
+                       if(temp)
+                       per_cpu(cpu_tst_flags, cpu)[steps]|=CPU_TST_L1_STP0;
+
+                       temp=lzo_cmp_decmp_tst((unsigned char *) test_cpus_l1_1,lzo_dcmp_size,
+                       test_mal_buf[cpu],sizeof(char)*BUF_SIZE,per_cpu(lzo_wrk, cpu));
+                       if(temp)
+                       per_cpu(cpu_tst_flags, cpu)[steps]|=CPU_TST_L1_STP0;
+
+                       mutex_unlock(&per_cpu(lzo_lock, cpu)[steps]);   
+
+                       
+                       if(ftt_test())
+                       {       
+                               ft_printk("ftt_test  err(%d,%d)\n",task_data->thread_id,cpu);
+                               while(1);
+                       }
+                       if(calc_pi()==(-EPERM))
+                       {       
+                               ft_printk("calc_pi      err(%d,%d)\n",task_data->thread_id,cpu);
+                               while(1);
+                       }
+                       
+               }
+               msleep(50);
+
+               ft_cpu_l1_test(l1_tst_cnt[steps]/2,l1_tst_cnt[steps]/2);
+               
+               ft_printk(".");
+
+               mutex_lock(&per_cpu(l2_lock, cpu)[steps]);
+               if(steps==0)
+               {       
+                       temp=ft_test_cpus_l2_memcpy(NULL,l2_test_buf[cpu],sizeof(char)*BUF_SIZE,l2_cpy_cnt[steps]);
+               }
+               else
+                       temp=ft_test_cpus_l2_memcpy(test_mal_buf[cpu],l2_test_buf[cpu],sizeof(char)*BUF_SIZE_M,l2_cpy_cnt[steps]);
+                       
+               if(temp)
+               per_cpu(cpu_tst_flags, cpu)[steps]|=CPU_TST_L2_STP0;
+               
+               msleep(50);
+               #if 1
+               // max rate test //1658 
+               if(steps==1)
+               {
+               
+                       //ft_printk("ft_test_mem l2 end (%d,%d)",task_data->thread_id,cpu);
+                       temp=ft_test_mem((void *)l2_test_buf[cpu], l2_DCACHE_SIZE+l2_DCACHE_SIZE/20,1) ;//l2_DCACHE_SIZE
+                       //if(cpu==0)
+                       //ft_printk(".");
+
+                       if(temp)
+                       per_cpu(cpu_tst_flags, cpu)[steps]|=CPU_TST_L2_STP0;
+               }
+               #endif
+
+               //ft_printk("test_type0 l2 end (%d,%d)",task_data->thread_id,cpu);
+               mutex_unlock(&per_cpu(l2_lock, cpu)[steps]);
+
+
+       }       
+
+       if((task_data->thread_id<l1_tst_cnt[steps]))
+       {       
+               if(task_data->thread_id%2)
+                       ft_cpu_l1_test(1,0);
+               else
+                       ft_cpu_l1_test(0,1);
+               
+       }
+       
+       if(steps<2)
+       {       
+               if(get_setup_tasks(stp_task_num[steps])>pi_tst_cnt[steps])
+                       id_start=get_setup_tasks(stp_task_num[steps])-pi_tst_cnt[steps];
+               else
+                       id_start=1;
+
+               if(task_data->thread_id>=(id_start))
+               {
+                       //ft_printk("calc_pi st=%d(%d,%d)\n",id_start,task_data->thread_id,cpu);
+                       if(calc_pi()==(-EPERM))
+                       {       
+                               ft_printk("calc_pi  err(%d,%d)\n",task_data->thread_id,cpu);
+                               while(1);
+                       }
+
+               }
+       }
+
+
+}
+
+
+int ft_cpu_test_type0_check(int steps,const char *str)
+{
+       int cpu, ret = 0,i;
+
+
+       for (cpu = 0; cpu < NR_CPUS; cpu++)
+               for(i=0;i<get_setup_tasks(stp_task_num[0]);i++) 
+                       down(&sem_setups[0]);
+               
+       ret=0;
+       for (cpu = 0; cpu < NR_CPUS; cpu++)
+       {
+               ret|=per_cpu(cpu_tst_flags, cpu)[steps];
+       
+       }
+       ft_printk("setup%d end:ret=%x,arm=%lu\n",
+                       steps,ret,clk_get_rate(arm_clk)/MHZ);
+       if(ret)
+       {
+               ft_printk("#R01%s*\n",str);
+               while(1);
+       }
+       else
+               ft_printk("#R00%s*\n",str);
+       
+       return ret;
+}      
+/*************************************type1 rate tst case**************************************/
+#define ENABLE_FT_TEST_GPIO   // for ft seting 1.6G volt
+
+#ifdef ENABLE_FT_TEST_GPIO
+
+#define FT_CLIENT_READY_PIN    (0x0c20)
+#define FT_CLIENT_IDLE_PIN     (0x0a00)
+#include "rk_ft_io.c"
+
+#endif
+
+//uint32_t ddr_change_freq(uint32_t nMHz);
+
+int ft_cpu_test_type1_check(int steps,const char *str)
+{
+       int cpu, ret = 0,i;
+       int rate;
+        #ifdef ENABLE_FT_TEST_GPIO
+        u8 idle_gpio_level;
+        #endif
+
+       if(arm_setups_rate[steps])
+       {       
+               
+               ft_printk("#S%s*\n",str);               
+               ft_printk("start test arm=%lu\n",arm_setups_rate[steps]/MHZ);
+
+               rate=clk_get_rate(arm_clk);
+
+               if(arm_setups_rate[steps]<rate)
+               {       
+                       clk_set_rate(arm_clk,arm_setups_rate[steps]);
+                        #ifdef FT_DDR_FREQ
+                       if(ddr_setups_rate[steps])
+                       {       
+                               ddr_change_freq(ddr_setups_rate[steps]);
+                               clk_set_rate(ddr_clk,0);
+                       }       
+                        #endif    
+               }
+               
+       
+
+               #ifdef ENABLE_FT_TEST_GPIO
+               
+               //ft_printk("S%s,will set gpio\n",str);
+
+        
+               idle_gpio_level=gpio_get_input_level(FT_CLIENT_IDLE_PIN);
+               // send msg to ctr board to up the volt
+                gpio_set_output_level(FT_CLIENT_READY_PIN,RKPM_GPIO_OUT_H);  
+               
+               // waiting for volt upping ok 
+               while( idle_gpio_level== gpio_get_input_level(FT_CLIENT_IDLE_PIN));
+                gpio_set_output_level(FT_CLIENT_READY_PIN,RKPM_GPIO_OUT_L);  
+
+               //ft_printk("S%s,set gpio end\n",str);
+               #endif
+
+               if(arm_setups_rate[steps]>=rate)
+               {       
+                       clk_set_rate(arm_clk,arm_setups_rate[steps]);
+            #ifdef FT_DDR_FREQ
+                       if(ddr_setups_rate[steps])
+                       {       
+                               ddr_change_freq(ddr_setups_rate[steps]);
+                               clk_set_rate(ddr_clk,0);
+                       }       
+            #endif
+                               
+               }
+               
+               setups_flag[steps-1]=1;
+               for (cpu = 0; cpu < NR_CPUS; cpu++)
+               {       
+                       wake_up(&per_cpu(wait_setups, cpu)[steps-1][0]);
+               }
+               
+               
+               for (cpu = 0; cpu < NR_CPUS; cpu++)
+               for(i=0;i<get_setup_tasks(stp_task_num[steps]);i++)
+                       down(&sem_setups[steps]);
+                       
+               ret=0;
+               for (cpu = 0; cpu < NR_CPUS; cpu++)
+               {
+                       ret|=per_cpu(cpu_tst_flags, cpu)[steps];
+                       //ft_printk("setup%d,cpu%d flags=%x\n",steps,cpu,
+                       //per_cpu(cpu_tst_flags, cpu)[steps]);
+               }       
+               ft_printk("setup%d end:ret=%x,arm=%lu\n",
+                               steps,ret,clk_get_rate(arm_clk)/MHZ);
+               if(ret)
+               {
+                       ft_printk("#R01%s*\n",str);
+                       while(1);
+               }
+               else
+                       ft_printk("#R00%s*\n",str);
+               
+       }
+       return ret;
+
+}
+
+int rk_ft_tests_cpus_init(int cpu);
+#define   FT_TIME_PRT (80)
+// tst thread callback for per cpu
+static int ft_cpu_test(void *data)
+{      
+       u32 cpu = smp_processor_id();
+       struct tst_task_struct *task_data=(struct tst_task_struct *)data;
+       int i;  
+
+       if(task_data->thread_id==0)
+       {
+               per_cpu(sem_setups_cnt, cpu)=0;
+               rk_ft_tests_cpus_init(cpu);
+               
+               for(i=1;i<get_setup_tasks(stp_task_num[0]);i++)
+               {       
+                       wake_up_process(tst_task_date[i][cpu].task);
+               }
+               //ft_printk("cpu %d wake up\n",cpu);
+       }
+       
+       //steps 0
+       if(task_data->thread_id<stp_task_num[0])
+       {
+               ft_cpu_test_type0(0,task_data);
+
+               per_cpu(sem_setups_cnt, cpu)+=1;
+
+               if((per_cpu(sem_setups_cnt, cpu)%FT_TIME_PRT==0||per_cpu(sem_setups_cnt, cpu)==get_setup_tasks(stp_task_num[0]))
+                       &&cpu==0
+                       )
+               ft_printk(".\n");
+
+               //if(per_cpu(sem_setups_cnt, cpu)==get_setup_tasks(stp_task_num[0]))
+               //      ft_printk("setup=%d cpu=%d,totall=%d,id=%d\n",0,cpu,per_cpu(sem_setups_cnt, cpu),task_data->thread_id);
+               up(&sem_setups[0]);
+       }
+
+       // steps 1
+       if(task_data->thread_id<stp_task_num[1])
+       {               
+               wait_event_freezable(per_cpu(wait_setups, cpu)[0][task_data->thread_id],(setups_flag[0]==1)||kthread_should_stop());
+
+
+               if(task_data->thread_id==0)
+               {               
+               
+                       per_cpu(sem_setups_cnt, cpu)=0;
+                       for(i=1;i<get_setup_tasks(stp_task_num[1]);i++)
+                       {       
+                               wake_up_process(tst_task_date[i][cpu].task);
+                       }
+                       //ft_printk("setup 1 cpu %d wake up\n",cpu);
+               }
+
+               ft_cpu_test_type0(1,task_data);
+
+
+               per_cpu(sem_setups_cnt, cpu)+=1;
+               
+               if((per_cpu(sem_setups_cnt, cpu)%FT_TIME_PRT==0||per_cpu(sem_setups_cnt, cpu)==get_setup_tasks(stp_task_num[1]))
+                       &&cpu==0
+                       )
+               ft_printk(".\n");
+
+               //if(per_cpu(sem_setups_cnt, cpu)==get_setup_tasks(stp_task_num[1]))
+               //      ft_printk("setup=%d cpu=%d,totall=%d,id=%d\n",1,cpu,per_cpu(sem_setups_cnt, cpu),task_data->thread_id);
+               up(&sem_setups[1]);
+       }
+
+       
+       if(task_data->thread_id<stp_task_num[2])
+       {       
+               wait_event_freezable(per_cpu(wait_setups, cpu)[1][task_data->thread_id],(setups_flag[1]==1)||kthread_should_stop());
+
+               if(task_data->thread_id==0)
+               {       
+                       per_cpu(sem_setups_cnt, cpu)=0;
+                       for(i=1;i<get_setup_tasks(stp_task_num[2]);i++)
+                       {       
+                               wake_up_process(tst_task_date[i][cpu].task);
+                       }
+                       //ft_printk("cpu %d wake up\n",cpu);
+               }
+               
+               ft_cpu_test_type0(2,task_data);
+               per_cpu(sem_setups_cnt, cpu)+=1;
+
+               if((per_cpu(sem_setups_cnt, cpu)%FT_TIME_PRT==0||per_cpu(sem_setups_cnt, cpu)==get_setup_tasks(stp_task_num[2]))
+                       &&cpu==0
+                       )
+               ft_printk(".\n");
+               
+               //if(per_cpu(sem_setups_cnt, cpu)==get_setup_tasks(stp_task_num[2]))
+               //      ft_printk("setup=%d cpu=%d,totall=%d,id=%d\n",2,cpu,per_cpu(sem_setups_cnt, cpu),task_data->thread_id);
+               up(&sem_setups[2]);
+       }
+
+       
+       if(task_data->thread_id<stp_task_num[3])
+       {
+               wait_event_freezable(per_cpu(wait_setups, cpu)[2][task_data->thread_id],(setups_flag[2]==1)||kthread_should_stop());
+
+
+               if(task_data->thread_id==0)
+               {
+               
+                       per_cpu(sem_setups_cnt, cpu)=0;
+                       
+                       for(i=1;i<get_setup_tasks(stp_task_num[3]);i++)
+                       {       
+                               wake_up_process(tst_task_date[i][cpu].task);
+                       }
+                       //ft_printk("cpu %d wake up\n",cpu);
+               }
+               
+               ft_cpu_test_type0(3,task_data);
+               per_cpu(sem_setups_cnt, cpu)+=1;
+
+               
+               if((per_cpu(sem_setups_cnt, cpu)%FT_TIME_PRT==0||per_cpu(sem_setups_cnt, cpu)==get_setup_tasks(stp_task_num[3]))
+                       &&cpu==0
+                       )
+               ft_printk(".\n");
+               
+               //if(per_cpu(sem_setups_cnt, cpu)==get_setup_tasks(stp_task_num[3]))
+               //      ft_printk("setup=%d cpu=%d,totall=%d,id=%d\n",3,cpu,per_cpu(sem_setups_cnt, cpu),task_data->thread_id);
+               up(&sem_setups[3]);
+       }
+       return 0;
+}
+
+int rk_ft_tests_cpus_init(int cpu)
+{
+       int i,ret = 0,j;
+       struct sched_param param = { .sched_priority = 0 }; 
+       char *buf;
+       unsigned char *wrk;
+       struct task_struct *task;
+
+       //arm_clk=clk_get(NULL, "cpu");
+       
+       // malloc buf
+       test_mal_buf[cpu]=NULL;
+       buf = kmalloc(BUF_SIZE_M, GFP_KERNEL);
+       if (buf)
+       {
+               test_mal_buf[cpu]=buf;
+               //printk("xdbg but=%x\n",(void*)buf);
+       }
+       
+       for(i=0;i<TST_SETUPS;i++)
+       {       
+               for(j=0;j<TST_TASK_NUM;j++)
+               init_waitqueue_head(&per_cpu(wait_setups, cpu)[i][j]);
+               
+       }       
+
+       for(i=0;i<TST_SETUPS;i++)
+       {       
+               mutex_init(&per_cpu(lzo_lock, cpu)[i]);
+               mutex_init(&per_cpu(l2_lock, cpu)[i]);
+       }
+               
+               
+       wrk = vmalloc(LZO1X_MEM_COMPRESS);
+       if (wrk==NULL) {
+               printk("xxx  Failed to allocate LZO workspace\n");
+               //return -1;
+       }
+       else
+               per_cpu(lzo_wrk, cpu)=wrk;              
+       
+       for(i=1;i<TST_TASK_NUM;i++)
+       {       
+               task= kthread_create(ft_cpu_test,(void *)&tst_task_date[i][cpu],"ft_test_task%d_cpu%d",i,cpu);
+               tst_task_date[i][cpu].task=task;
+               tst_task_date[i][cpu].thread_id=i;
+               tst_task_date[i][cpu].cpu=cpu;
+               
+               tst_task_date[i][cpu].tst_step=0;
+               if((i%3)==0)
+                       sched_setscheduler_nocheck(task, /*task_policy[i%4]*/SCHED_RR, &param);
+               else if((i%3)==1)
+                       sched_setscheduler_nocheck(task, /*task_policy[i%4]*/SCHED_FIFO, &param); 
+               else
+                       sched_setscheduler_nocheck(task, /*task_policy[i%4]*/SCHED_NORMAL, &param); 
+               
+               get_task_struct(task);
+               kthread_bind(task, cpu);
+       }
+       
+       return ret;
+}
+
+static int __init rk_ft_tests_init(void)
+{
+       int cpu, i,ret = 0;
+       struct sched_param param = { .sched_priority =0 };      
+    
+        struct device_node *parent;
+
+        ft_printk_info("%s\n",__FUNCTION__);
+
+        
+        arm_clk=clk_get(NULL, "clk_core");       
+
+        parent = of_find_node_by_name(NULL, "rockchip_ft_test");    
+        
+        if (IS_ERR_OR_NULL(parent)) {
+             printk("%s dev node err\n", __func__);
+             return -1;
+         }
+              
+         if(of_property_read_u32_array(parent,"rockchip,arm_rate",(u32 *)&arm_setups_rate[0],TST_SETUPS))
+         {
+                 printk("%s:arm_setups_rate\n",__FUNCTION__);
+                 return -1;
+         }
+         ft_printk_info("arm_setups_rate=%lu(init clk),%lu,%lu,%lu\n",
+                        clk_get_rate(arm_clk)/MHZ,arm_setups_rate[1]/MHZ,arm_setups_rate[2]/MHZ,arm_setups_rate[3]/MHZ);
+
+
+        if(of_property_read_u32_array(parent,"rockchip,l1_tst_cnt",(u32 *)&l1_tst_cnt[0],TST_SETUPS))
+        {
+               printk("%s:l1_tst_cnt error\n",__FUNCTION__);
+               return -1;
+        }
+
+        if(of_property_read_u32_array(parent,"rockchip,l2_cpy_cnt",(u32 *)&l2_cpy_cnt[0],TST_SETUPS))
+         {
+                 printk("%s:l1_tst_cnt error\n",__FUNCTION__);
+                 return -1;
+         }
+        if(of_property_read_u32_array(parent,"rockchip,ftt_tst_cnt",(u32 *)&ftt_tst_cnt[0],TST_SETUPS))
+        {
+               printk("%s:ftt_tst_cnt error\n",__FUNCTION__);
+               return -1;
+        }
+        if(of_property_read_u32_array(parent,"rockchip,pi_tst_cnt",(u32 *)&pi_tst_cnt[0],TST_SETUPS))
+        {
+               printk("%s:pi_tst_cnt error\n",__FUNCTION__);
+               return -1;
+        }
+          if(of_property_read_u32_array(parent,"rockchip,stp_task_num",(u32 *)&stp_task_num[0],TST_SETUPS))
+        {
+               printk("%s:stp_task_num error\n",__FUNCTION__);
+               return -1;
+        }
+
+        for(i=0;i<TST_SETUPS;i++)
+        {   
+            ft_printk_info("index=%d,l1_tst_cnt=%u,l2_cpy_cnt=%u,ftt_tst_cnt=%u,pi_tst_cnt=%u,stp_task_num=%u\n"
+                                    ,i,l1_tst_cnt[i],l2_cpy_cnt[i],ftt_tst_cnt[i],pi_tst_cnt[i],stp_task_num[i]);
+       
+        }
+
+        if(of_property_read_u32_array(parent,"rockchip,ft_end_cnt",&ft_end_cnt,1))
+        {
+             printk("%s:ft_end_cnt error\n",__FUNCTION__);
+             return -1;
+        }
+
+        
+
+        
+
+       //ddr_clk = clk_get(NULL, "clk_ddr");
+       //clk_set_rate(ddr_clk,0);
+
+
+       for(i=0;i<TST_SETUPS;i++)
+       {       
+               sema_init(&sem_setups[i], 0);   
+       }
+
+       for(i=0;i<1;i++)
+       {       
+               struct task_struct *task;
+               
+               for (cpu = 0; cpu < NR_CPUS; cpu++) 
+               {       
+                       task= kthread_create(ft_cpu_test,(void *)&tst_task_date[i][cpu],"ft_test_task%d_cpu%d",i,cpu);
+                       tst_task_date[i][cpu].task=task;
+                       tst_task_date[i][cpu].thread_id=i;
+                       tst_task_date[i][cpu].cpu=cpu;
+                       
+                       tst_task_date[i][cpu].tst_step=0;
+                       
+                       sched_setscheduler_nocheck(task, /*task_policy[i%4]*/SCHED_RR, &param);
+       
+                       get_task_struct(task);
+                       kthread_bind(task, cpu);
+               }
+       }
+
+       
+       for (cpu = 1; cpu < NR_CPUS; cpu++) 
+       {       
+               wake_up_process(tst_task_date[0][cpu].task);    
+       }
+       
+       wake_up_process(tst_task_date[0][0].task);
+
+       return ret;     
+}
+
+//core_initcall(rk_ft_tests_init);
+
+static int rk_ft_tests_over(void)
+{
+       int ret = 0;
+       //int gpio_ret,gpio_ret1;
+        rk_ft_tests_init();
+
+    
+        #ifdef ENABLE_FT_TEST_GPIO
+            //GPIO0_A0
+            gpio_set_in_output(FT_CLIENT_READY_PIN,RKPM_GPIO_OUTPUT);    
+            gpio_set_output_level(FT_CLIENT_READY_PIN,RKPM_GPIO_OUT_L);  
+            
+            gpio_set_in_output(FT_CLIENT_IDLE_PIN,RKPM_GPIO_INPUT);
+            
+            pin_set_fun(FT_CLIENT_READY_PIN);    //ready
+            pin_set_fun(FT_CLIENT_IDLE_PIN);    // idle
+           
+       #endif  
+
+       ft_cpu_test_type0_check(0,"KERNEL");
+       
+       ft_cpu_test_type1_check(1,"HSPEED");
+       
+       ft_cpu_test_type1_check(2,"HSPEED2");
+       
+       ft_cpu_test_type1_check(3,"KERNEL2");
+
+       ft_printk("#END%x*\n",ft_end_cnt);
+       
+       #ifdef ENABLE_FT_SUSPEND_TST
+       {       
+               ft_test_flag_seting();
+               pm_suspend(PM_SUSPEND_MEM);
+       }
+       #endif
+       while(1);
+
+       return ret;
+}
+
+late_initcall_sync(rk_ft_tests_over);
+
diff --git a/arch/arm/mach-rockchip/rk_pm_tests/ft/rk_ft_io.c b/arch/arm/mach-rockchip/rk_pm_tests/ft/rk_ft_io.c
new file mode 100755 (executable)
index 0000000..0b0d4f2
--- /dev/null
@@ -0,0 +1,176 @@
+
+#include <linux/rockchip/cpu.h>
+#include <linux/rockchip/grf.h>
+#include <linux/rockchip/iomap.h>
+#include <linux/irqchip/arm-gic.h>
+#include <linux/rockchip/pmu.h>
+
+#include "../../pm.h"
+
+/* GPIO control registers */
+#define GPIO_SWPORT_DR         0x00
+#define GPIO_SWPORT_DDR                0x04
+#define GPIO_INTEN                     0x30
+#define GPIO_INTMASK           0x34
+#define GPIO_INTTYPE_LEVEL     0x38
+#define GPIO_INT_POLARITY      0x3c
+#define GPIO_INT_STATUS                0x40
+#define GPIO_INT_RAWSTATUS     0x44
+#define GPIO_DEBOUNCE          0x48
+#define GPIO_PORTS_EOI         0x4c
+#define GPIO_EXT_PORT          0x50
+#define GPIO_LS_SYNC           0x60
+
+
+static void rk3288_pin_set_fun(u8 port,u8 bank,u8 b_gpio,u8 fun)
+{ 
+        u8 off_set;
+        bank-=0xa;
+    
+        if(port==0)
+        { 
+            if(bank>2)
+                return;
+            off_set=RK3288_PMU_GPIO0_A_IOMUX+bank*4;
+            pmu_writel(RKPM_VAL_SETBITS(pmu_readl(off_set),fun,b_gpio*2,0x3),off_set);
+        }
+        else if(port==1||port==2)
+        {
+            off_set=port*(4*4)+bank*4;
+            reg_writel(RKPM_W_MSK_SETBITS(fun,b_gpio*2,0x3),RK_GRF_VIRT+0+off_set);
+        }
+        else if(port==3)
+        {
+            if(bank<=2)
+            {
+                off_set=0x20+bank*4;
+                reg_writel(RKPM_W_MSK_SETBITS(fun,b_gpio*2,0x3),RK_GRF_VIRT+0+off_set);
+
+            }
+            else
+            {
+                off_set=0x2c+(b_gpio/4)*4;
+                reg_writel(RKPM_W_MSK_SETBITS(fun,(b_gpio%4)*4,0x3),RK_GRF_VIRT+0+off_set);
+            }
+
+        }
+        else if(port==4)
+        {
+            if(bank<=1)
+            {
+                off_set=0x34+bank*8+(b_gpio/4)*4;
+                reg_writel(RKPM_W_MSK_SETBITS(fun,(b_gpio%4)*4,0x3),RK_GRF_VIRT+0+off_set);
+            }
+            else
+            {
+                off_set=0x44+(bank-2)*4;
+                reg_writel(RKPM_W_MSK_SETBITS(fun,b_gpio*2,0x3),RK_GRF_VIRT+0+off_set);
+            }
+
+        }
+        else if(port==5||port==6)
+        {
+                off_set=0x4c+(port-5)*4*4+bank*4;
+                reg_writel(RKPM_W_MSK_SETBITS(fun,b_gpio*2,0x3),RK_GRF_VIRT+0+off_set);
+        }
+        else if(port==7)
+        {
+            if(bank<=1)
+            {
+                off_set=0x6c+bank*4;
+                reg_writel(RKPM_W_MSK_SETBITS(fun,b_gpio*2,0x3),RK_GRF_VIRT+0+off_set);
+            }
+            else
+            {
+                off_set=0x74+(bank-2)*8+(b_gpio/4)*4;
+                //rkpm_ddr_printascii("gpio");
+                //rkpm_ddr_printhex(off_set);                   
+                //rkpm_ddr_printascii("-");
+                //rkpm_ddr_printhex((b_gpio%4)*4);
+
+                reg_writel(RKPM_W_MSK_SETBITS(fun,(b_gpio%4)*4,0x3),RK_GRF_VIRT+0+off_set);
+
+                //rkpm_ddr_printhex(reg_readl(RK_GRF_VIRT+0+off_set));    
+                //rkpm_ddr_printascii("\n");        
+            }
+
+        }
+        else if(port==8)
+        {
+            if(bank<=1)
+            {
+                off_set=0x80+bank*4;
+                reg_writel(RKPM_W_MSK_SETBITS(fun,b_gpio*2,0x3),RK_GRF_VIRT+0+off_set);
+            }
+        }
+               
+}
+static void pin_set_fun(u32 pins)
+{
+    rk3288_pin_set_fun(RKPM_PINBITS_PORT(pins),RKPM_PINBITS_BANK(pins),
+                                                        RKPM_PINBITS_BGPIO(pins),RKPM_PINBITS_FUN(pins));
+}
+
+static void rk3288_gpio_set_in_output(u8 port,u8 bank,u8 b_gpio,u8 type)
+{
+    u32 val;    
+    
+    bank-=0xa;
+    b_gpio=bank*8+b_gpio;//
+
+    val=reg_readl(RK_GPIO_VIRT(port)+GPIO_SWPORT_DDR);
+
+    if(type==RKPM_GPIO_OUTPUT)
+        val|=(0x1<<b_gpio);
+    else
+        val&=~(0x1<<b_gpio);
+    
+    reg_writel(val,RK_GPIO_VIRT(port)+GPIO_SWPORT_DDR);
+}
+static void gpio_set_in_output(u32 pins,u8 type)
+{
+    rk3288_gpio_set_in_output(RKPM_PINBITS_PORT(pins),RKPM_PINBITS_BANK(pins),
+                                                                        RKPM_PINBITS_BGPIO(pins),type);
+}
+
+static  void rk3288_gpio_set_output_level(u8 port,u8 bank,u8 b_gpio,u8 level)
+{
+    u32 val;    
+
+    bank-=0xa;
+    b_gpio=bank*8+b_gpio;
+        
+    val=reg_readl(RK_GPIO_VIRT(port)+GPIO_SWPORT_DR);
+
+    if(level==RKPM_GPIO_OUT_H)
+        val|=(0x1<<b_gpio);
+    else //
+        val&=~(0x1<<b_gpio);
+
+     reg_writel(val,RK_GPIO_VIRT(port)+GPIO_SWPORT_DR);
+}
+
+static void gpio_set_output_level(u32 pins,u8 level)
+{
+
+        rk3288_gpio_set_output_level(RKPM_PINBITS_PORT(pins),RKPM_PINBITS_BANK(pins),
+                                                                RKPM_PINBITS_BGPIO(pins),level);
+
+}
+
+static u8 rk3288_gpio_get_input_level(u8 port,u8 bank,u8 b_gpio)
+{
+
+    bank-=0xa;
+    b_gpio=bank*8+b_gpio;
+
+    return (reg_readl(RK_GPIO_VIRT(port)+GPIO_EXT_PORT)>>b_gpio)&0x1;
+}
+static u8 gpio_get_input_level(u32 pins)
+{
+
+        return rk3288_gpio_get_input_level(RKPM_PINBITS_PORT(pins),RKPM_PINBITS_BANK(pins),
+                                                                RKPM_PINBITS_BGPIO(pins));
+}
+
+
diff --git a/arch/arm/mach-rockchip/rk_pm_tests/ft/test_mem.c b/arch/arm/mach-rockchip/rk_pm_tests/ft/test_mem.c
new file mode 100755 (executable)
index 0000000..f323486
--- /dev/null
@@ -0,0 +1,447 @@
+
+#include <linux/string.h>
+#include <linux/resume-trace.h>
+#include <linux/workqueue.h>
+#include <linux/mutex.h>
+#include <linux/module.h>
+#include <linux/syscalls.h>
+#include <linux/init.h>
+#include <linux/interrupt.h>
+#include <linux/delay.h>
+#include <linux/cpu.h>
+#include <linux/slab.h>
+#include <linux/freezer.h>
+#include <linux/vmalloc.h>
+
+
+//#include     "..\common\config.h"
+
+//#define TEST_VALUE 0xAAAAAAAA//0x55555555
+
+
+
+
+
+#define TEST_VALUE 0x55555555
+#define UL_ONEBITS 0xffffffff
+//#define TEST_DEBUG_EN   0
+#define UL_BYTE(x) ((x | x << 8 | x << 16 | x << 24))
+#define BUF_SIZE     (256 * 1024)  //(256 * 1024)
+
+//static char memtest_buf[BUF_SIZE]  __attribute__((aligned(4096)));
+
+#define printf(fmt, arg...) \
+               printk(KERN_EMERG fmt, ##arg)
+
+
+//char* pTestmem=(char *)CACHE_FIFO_STAR_ADDR0;
+
+typedef unsigned long ul;
+typedef unsigned long volatile ulv;
+
+static int compare_regions(ulv *bufa, ulv *bufb, ul count) 
+{
+    int r = 0;
+    ul i;
+    ulv *p1 = bufa;
+    ulv *p2 = bufb;
+  //  int n=0;
+
+    for (i = 0; i < count; i++, p1++, p2++) 
+    {
+        if (*p1 != *p2) 
+        {
+            r = -1;
+          //  n++;
+         //   if(n>10)
+         //   {
+            break;
+         //   }
+        }
+    }
+    return r;
+}
+
+static int compare_regions_reverse(ulv *bufa, ulv *bufb, ul count) 
+{
+    int r = 0;
+    ul i;
+    ulv *p1 = bufa;
+    ulv *p2 = bufb;
+  //  int n=0;
+
+    for (i = 0; i < count; i++, p1++, p2++) 
+    {
+        if (*p1 != ~(*p2)) 
+        {
+            r = -1;
+           // n++;
+           // if(n>10)
+           // {
+                break;
+           // }
+        }
+    }
+    return r;
+}
+
+static int test_stuck_address(ulv *bufa, ul count) 
+{
+    ulv *p1 = bufa;
+    ul j;
+    ul i;
+
+    for (j = 0; j < 16; j++)
+    {
+        p1 = (ulv *) bufa;
+        for (i = 0; i < count; i++) 
+        {
+            *p1 = (((j + i) % 2) == 0 ? (ul) p1 : ~((ul) p1));
+            *p1++;
+        }
+        p1 = (ulv *) bufa;
+        for (i = 0; i < count; i++, p1++) 
+        {
+            if (*p1 != (((j + i) % 2) == 0 ? (ul) p1 : ~((ul) p1))) 
+            {
+                return -1;
+            }
+        }
+    }
+    return 0;
+}
+
+static int test_random_value(ulv *bufa, ulv *bufb, ul count) 
+{
+    ulv *p1 = bufa;
+    ulv *p2 = bufb;
+    ul i;
+    
+    for (i = 0; i < count; i++) 
+    {
+        *p1++ = *p2++ = TEST_VALUE;
+    }
+    return compare_regions(bufa, bufb, count);
+}
+
+static int test_xor_comparison(ulv *bufa, ulv *bufb, ul count) 
+{
+    ulv *p1 = bufa;
+    ulv *p2 = bufb;
+    ul i;
+    ul q = TEST_VALUE;
+
+    for (i = 0; i < count; i++) 
+    {
+        *p1++ ^= q;
+        *p2++ ^= q;
+    }
+    return compare_regions(bufa, bufb, count);
+}
+
+static int test_sub_comparison(ulv *bufa, ulv *bufb, ul count) 
+{
+    ulv *p1 = bufa;
+    ulv *p2 = bufb;
+    ul i;
+    ul q = TEST_VALUE;
+
+    for (i = 0; i < count; i++) 
+    {
+        *p1++ -= q;
+        *p2++ -= q;
+    }
+    return compare_regions(bufa, bufb, count);
+}
+
+static int test_mul_comparison(ulv *bufa, ulv *bufb, ul count) 
+{
+    ulv *p1 = bufa;
+    ulv *p2 = bufb;
+    ul i;
+    ul q = TEST_VALUE;
+
+    for (i = 0; i < count; i++) 
+    {
+        *p1++ *= q;
+        *p2++ *= q;
+    }
+    return compare_regions(bufa, bufb, count);
+}
+
+static int test_div_comparison(ulv *bufa, ulv *bufb, ul count) 
+{
+    ulv *p1 = bufa;
+    ulv *p2 = bufb;
+    ul i;
+    ul q = TEST_VALUE;
+
+    for (i = 0; i < count; i++) 
+    {
+        if (!q) 
+        {
+            q++;
+        }
+        *p1++ /= q;
+        *p2++ /= q;
+    }
+    return compare_regions(bufa, bufb, count);
+}
+
+static int test_or_comparison(ulv *bufa, ulv *bufb, ul count) 
+{
+    ulv *p1 = bufa;
+    ulv *p2 = bufb;
+    ul i;
+    ul q = TEST_VALUE;
+
+    for (i = 0; i < count; i++) 
+    {
+        *p1++ |= q;
+        *p2++ |= q;
+    }
+    return compare_regions(bufa, bufb, count);
+}
+
+static int test_and_comparison(ulv *bufa, ulv *bufb, ul count)
+{
+    ulv *p1 = bufa;
+    ulv *p2 = bufb;
+    ul i;
+    ul q = TEST_VALUE;
+
+    for (i = 0; i < count; i++) 
+    {
+        *p1++ &= q;
+        *p2++ &= q;
+    }
+    return compare_regions(bufa, bufb, count);
+}
+
+static int test_seqinc_comparison(ulv *bufa, ulv *bufb, ul count) 
+{
+    ulv *p1 = bufa;
+    ulv *p2 = bufb;
+    ul i;
+    ul q = TEST_VALUE;
+    ul value;
+
+    for (i = 0; i < count; i++) 
+    {
+        value = (i+q);
+        *p1++ = value;
+        *p2++ = ~value;
+    }
+    return compare_regions_reverse(bufa, bufb, count);
+}
+
+static int test_solidbits_comparison(ulv *bufa, ulv *bufb, ul count) 
+{
+    ulv *p1 = bufa;
+    ulv *p2 = bufb;
+    ul j;
+    ul q;
+    ul i;
+    ul value;
+
+    for (j = 0; j < 64; j++) 
+    {
+        q = ((j % 2) == 0 ? UL_ONEBITS : 0);
+        p1 = (ulv *) bufa;
+        p2 = (ulv *) bufb;
+        for (i = 0; i < count; i++) 
+        {
+            value = ((i % 2) == 0 ? q : ~q);
+            *p1++ = value;
+            *p2++ = ~value;
+        }
+        if (compare_regions_reverse(bufa, bufb, count))
+        {
+            return -1;
+        }
+    }
+    return 0;
+}
+
+static int test_blockseq_comparison(ulv *bufa, ulv *bufb, ul count) 
+{
+    ulv *p1 = bufa;
+    ulv *p2 = bufb;
+    ul j;
+    ul i;
+    ul value;
+
+    for (j = 0; j < 256; j++) 
+    {
+        p1 = (ulv *) bufa;
+        p2 = (ulv *) bufb;
+        for (i = 0; i < count; i++) 
+        {
+            value = (ul) UL_BYTE(j); 
+            *p1++ = value;
+            *p2++ = ~value;
+        }
+        if (compare_regions_reverse(bufa, bufb, count))
+        {
+            return -1;
+        }
+    }
+    return 0;
+}
+
+#define FT_FAILED (-1)
+int Test_mem(void *aligned, unsigned long b_size) 
+{
+    ulv *bufa, *bufb;
+    ul halflen, count;
+   // int i;
+
+    halflen = b_size / 2;        
+    count = halflen / sizeof(ul);
+    //aligned = memtest_buf;
+    bufa = (ulv *)aligned;
+    bufb = (ulv *)((ul)aligned + halflen);
+
+    if (test_stuck_address(aligned, (BUF_SIZE) / sizeof(ul)))
+    {   
+         printf("test_stuck_address_ error\n");
+          return FT_FAILED;//-1;
+    }   
+    if(test_random_value(bufa, bufb, count))
+    {   
+         printf("test_random_value_ error\n");
+          return FT_FAILED;//-1;
+    } 
+    if(test_xor_comparison(bufa, bufb, count))
+    {   
+         printf("test_xor_comparison_ error\n");
+          return FT_FAILED;//-1;
+    }
+    if(test_sub_comparison(bufa, bufb, count))
+    {    
+         printf("test_sub_comparison_ error\n");
+         return FT_FAILED;//-1;
+    }
+    //     printf("test_sub_comparison_ OK\n");
+    if(test_mul_comparison(bufa, bufb, count))
+    {    
+         printf("test_mul_comparison_ error\n");
+         return FT_FAILED;//-1;
+    }
+    if(test_div_comparison(bufa, bufb, count))
+    {    
+         printf("test_div_comparison_ error\n");
+         return FT_FAILED;//-1;
+    }
+    if(test_or_comparison(bufa, bufb, count))
+    {   
+         printf("test_or_comparison_ error\n");
+          return FT_FAILED;//-1;
+    }
+    if(test_and_comparison(bufa, bufb, count))
+    {    
+         printf("test_and_comparison_ error\n");
+         return FT_FAILED;//-1;
+    }
+    //     printf("test_and_comparison_ OK\n");
+    if(test_seqinc_comparison(bufa, bufb, count))
+    {    
+         printf("test_seqinc_comparison_ error\n");
+         return FT_FAILED;//-1;
+    }
+    //printf("test_seqinc_comparison_ OK\n");
+    
+    if(test_solidbits_comparison(bufa, bufb, count))
+    {              
+         printf("test_solidbits_comparison_ error\n");
+         return FT_FAILED;//-1;  
+    }
+    //printf("test_solidbits_comparison_ OK\n");
+    if(test_blockseq_comparison(bufa, bufb, count))
+    {    
+         printf("test_blockseq_comparison_ error\n");
+         return FT_FAILED;//-1;
+    }
+
+    return 0;
+}
+
+#if 0//def TEST_CPU123
+uint8 Test_mem123(char offset)
+{
+    void volatile *aligned;
+    ulv *bufa, *bufb;
+    ul halflen, count;
+    int i;
+
+    halflen = BUF_SIZE / 2;
+    count = halflen / sizeof(ul);
+    aligned = pTestmem + (offset << 20);
+    bufa = (ulv *)aligned;
+    bufb = (ulv *)((ul)aligned + halflen);
+
+    if (test_stuck_address(aligned, (BUF_SIZE) / sizeof(ul)))
+    {   
+         printf("cpu%d test_stuck_address_ error\n",(offset+1));
+          return FT_FAILED;//-1;
+    }   
+    if(test_random_value(bufa, bufb, count))
+    {   
+         printf("cpu%d test_random_value_ error\n",(offset+1));
+          return FT_FAILED;//-1;
+    } 
+    if(test_xor_comparison(bufa, bufb, count))
+    {   
+         printf("cpu%d test_xor_comparison_ error\n",(offset+1));
+          return FT_FAILED;//-1;
+    }
+    if(test_sub_comparison(bufa, bufb, count))
+    {    
+         printf("cpu%d test_sub_comparison_ error\n",(offset+1));
+         return FT_FAILED;//-1;
+    }
+    //     printf("test_sub_comparison_ OK\n");
+    if(test_mul_comparison(bufa, bufb, count))
+    {    
+         printf("cpu%d test_mul_comparison_ error\n",(offset+1));
+         return FT_FAILED;//-1;
+    }
+    if(test_div_comparison(bufa, bufb, count))
+    {    
+         printf("cpu%d test_div_comparison_ error\n",(offset+1));
+         return FT_FAILED;//-1;
+    }
+    if(test_or_comparison(bufa, bufb, count))
+    {   
+         printf("cpu%d test_or_comparison_ error\n",(offset+1));
+          return FT_FAILED;//-1;
+    }
+    if(test_and_comparison(bufa, bufb, count))
+    {    
+         printf("cpu%d test_and_comparison_ error\n",(offset+1));
+         return FT_FAILED;//-1;
+    }
+    //     printf("test_and_comparison_ OK\n");
+    if(test_seqinc_comparison(bufa, bufb, count))
+    {    
+         printf("cpu%d test_seqinc_comparison_ error\n",(offset+1));
+         return FT_FAILED;//-1;
+    }
+    //printf("test_seqinc_comparison_ OK\n");
+    
+    if(test_solidbits_comparison(bufa, bufb, count))
+    {              
+         printf("cpu%d test_solidbits_comparison_ error\n",(offset+1));
+         return FT_FAILED;//-1;  
+    }
+    //printf("test_solidbits_comparison_ OK\n");
+    if(test_blockseq_comparison(bufa, bufb, count))
+    {    
+         printf("cpu%d test_blockseq_comparison_ error\n",(offset+1));
+         return FT_FAILED;//-1;
+    }
+       printf("cpu%d memory test ok\n",(offset+1));
+    return FT_SUCCESS;
+}
+#endif
+