2011-07-03 00:28:51 +10:00
|
|
|
// -ck modified kernel taken from Phoenix taken from phatk
|
2011-06-28 21:11:04 +10:00
|
|
|
// This file is taken and modified from the public-domain poclbm project, and
|
|
|
|
// we have therefore decided to keep it public-domain in Phoenix.
|
2011-07-03 00:28:51 +10:00
|
|
|
// Modified version copyright 2011 Con Kolivas
|
2011-06-28 21:11:04 +10:00
|
|
|
|
|
|
|
// The X is a placeholder for patching to suit hardware
|
|
|
|
#define VECTORSX
|
|
|
|
|
|
|
|
#ifdef VECTORS4
|
|
|
|
typedef uint4 u;
|
|
|
|
#elif defined VECTORS2
|
|
|
|
typedef uint2 u;
|
|
|
|
#else
|
|
|
|
typedef uint u;
|
|
|
|
#endif
|
|
|
|
|
|
|
|
__constant uint K[64] = {
|
|
|
|
0x428a2f98, 0x71374491, 0xb5c0fbcf, 0xe9b5dba5, 0x3956c25b, 0x59f111f1, 0x923f82a4, 0xab1c5ed5,
|
|
|
|
0xd807aa98, 0x12835b01, 0x243185be, 0x550c7dc3, 0x72be5d74, 0x80deb1fe, 0x9bdc06a7, 0xc19bf174,
|
|
|
|
0xe49b69c1, 0xefbe4786, 0x0fc19dc6, 0x240ca1cc, 0x2de92c6f, 0x4a7484aa, 0x5cb0a9dc, 0x76f988da,
|
|
|
|
0x983e5152, 0xa831c66d, 0xb00327c8, 0xbf597fc7, 0xc6e00bf3, 0xd5a79147, 0x06ca6351, 0x14292967,
|
|
|
|
0x27b70a85, 0x2e1b2138, 0x4d2c6dfc, 0x53380d13, 0x650a7354, 0x766a0abb, 0x81c2c92e, 0x92722c85,
|
|
|
|
0xa2bfe8a1, 0xa81a664b, 0xc24b8b70, 0xc76c51a3, 0xd192e819, 0xd6990624, 0xf40e3585, 0x106aa070,
|
|
|
|
0x19a4c116, 0x1e376c08, 0x2748774c, 0x34b0bcb5, 0x391c0cb3, 0x4ed8aa4a, 0x5b9cca4f, 0x682e6ff3,
|
|
|
|
0x748f82ee, 0x78a5636f, 0x84c87814, 0x8cc70208, 0x90befffa, 0xa4506ceb, 0xbef9a3f7, 0xc67178f2
|
|
|
|
};
|
|
|
|
|
|
|
|
__constant uint H[8] = {
|
|
|
|
0x6a09e667, 0xbb67ae85, 0x3c6ef372, 0xa54ff53a, 0x510e527f, 0x9b05688c, 0x1f83d9ab, 0x5be0cd19
|
|
|
|
};
|
|
|
|
|
|
|
|
#define BFI_INTX
|
|
|
|
#define BITALIGNX
|
|
|
|
|
|
|
|
#ifdef BITALIGN
|
|
|
|
#pragma OPENCL EXTENSION cl_amd_media_ops : enable
|
|
|
|
#define rot(x, y) amd_bitalign(x, x, (u)(32-y))
|
|
|
|
#else
|
|
|
|
#define rotr(x, y) rotate((u)x, (u)(32-y))
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// This part is not from the stock poclbm kernel. It's part of an optimization
|
|
|
|
// added in the Phoenix Miner.
|
|
|
|
|
|
|
|
// Some AMD devices have the BFI_INT opcode, which behaves exactly like the
|
|
|
|
// SHA-256 Ch function, but provides it in exactly one instruction. If
|
|
|
|
// detected, use it for Ch. Otherwise, construct Ch out of simpler logical
|
|
|
|
// primitives.
|
|
|
|
|
|
|
|
#ifdef BFI_INT
|
|
|
|
// Well, slight problem... It turns out BFI_INT isn't actually exposed to
|
|
|
|
// OpenCL (or CAL IL for that matter) in any way. However, there is
|
|
|
|
// a similar instruction, BYTE_ALIGN_INT, which is exposed to OpenCL via
|
|
|
|
// amd_bytealign, takes the same inputs, and provides the same output.
|
|
|
|
// We can use that as a placeholder for BFI_INT and have the application
|
|
|
|
// patch it after compilation.
|
|
|
|
|
|
|
|
// This is the BFI_INT function
|
|
|
|
#define Ch(x, y, z) amd_bytealign(x, y, z)
|
|
|
|
|
|
|
|
// Ma can also be implemented in terms of BFI_INT...
|
|
|
|
#define Ma(a, b, c) amd_bytealign((c ^ a), (b), (a))
|
|
|
|
#else
|
|
|
|
#define Ch(x, y, z) (z ^ (x & (y ^ z)))
|
|
|
|
#define Ma(x, y, z) ((x & z) | (y & (x | z)))
|
|
|
|
#endif
|
|
|
|
|
|
|
|
//Various intermediate calculations for each SHA round
|
|
|
|
#define s0(n) (rot(Vals[(0 + 128 - (n)) % 8], 30)^rot(Vals[(0 + 128 - (n)) % 8], 19)^rot(Vals[(0 + 128 - (n)) % 8], 10))
|
|
|
|
#define s1(n) (rot(Vals[(4 + 128 - (n)) % 8], 26)^rot(Vals[(4 + 128 - (n)) % 8], 21)^rot(Vals[(4 + 128 - (n)) % 8], 7))
|
|
|
|
#define ch(n) (Ch(Vals[(4 + 128 - (n)) % 8],Vals[(5 + 128 - (n)) % 8],Vals[(6 + 128 - (n)) % 8]))
|
|
|
|
#define maj(n) (Ma(Vals[(1 + 128 - (n)) % 8],Vals[(2 + 128 - (n)) % 8],Vals[(0 + 128 - (n)) % 8]))
|
|
|
|
#define t1(n) (Vals[(7 + 128 - (n)) % 8] + K[(n) % 64]+ W[(n)] + ch(n) + s1(n))
|
|
|
|
#define t1W(n) (Vals[(7 + 128 - (n)) % 8] + K[(n) % 64]+ w(n) + ch(n) + s1(n))
|
|
|
|
#define t2(n) (s0(n) + maj(n))
|
|
|
|
|
|
|
|
//W calculation used for SHA round
|
|
|
|
#define w(n) (W[n] = P1(n) + P2(n) + P3(n) + P4(n))
|
|
|
|
|
|
|
|
//Full W calculation
|
|
|
|
#define R(x) (W[x] = (rot(W[x-2],15)^rot(W[x-2],13)^((W[x-2])>>10U)) + W[x-7] + (rot(W[x-15],25)^rot(W[x-15],14)^((W[x-15])>>3U)) + W[x-16])
|
|
|
|
|
|
|
|
//Partial W calculations (used for the begining where only some values are nonzero)
|
|
|
|
#define r0(x) ((rot(x,25)^rot(x,14)^((x)>>3U)))
|
|
|
|
#define r1(x) ((rot(x],15)^rot(x,13)^((x)>>10U)))
|
|
|
|
#define R0(n) ((rot(W[(n)],25)^rot(W[(n)],14)^((W[(n)])>>3U)))
|
|
|
|
#define R1(n) ((rot(W[(n)],15)^rot(W[(n)],13)^((W[(n)])>>10U)))
|
|
|
|
#define P1(x) R1(x-2)
|
|
|
|
#define P2(x) R0(x-15)
|
|
|
|
#define P3(x) W[x-7]
|
|
|
|
#define P4(x) W[x-16]
|
|
|
|
|
|
|
|
//SHA round with built in W calc
|
|
|
|
#define sharound2(n) { Vals[(3 + 128 - (n)) % 8] += t1W(n); Vals[(7 + 128 - (n)) % 8] = t1W(n) + t2(n); }
|
|
|
|
//SHA round without W calc
|
|
|
|
#define sharound(n) {t1 = t1(n); Vals[(3 + 128 - (n)) % 8] += t1(n); Vals[(7 + 128 - (n)) % 8] = t1(n) + t2(n); }
|
|
|
|
|
|
|
|
//Partial SHA calculations (used for begining and end)
|
|
|
|
#define partround(n) {Vals[(7 + 128 - n) % 8]=(Vals[(7 + 128 - n) % 8]+W[n]); Vals[(3 + 128 - n) % 8]+=Vals[(7 + 128 - n) % 8]; Vals[(7 + 128 - n) % 8]+=t1;}
|
|
|
|
|
|
|
|
__kernel
|
|
|
|
|
|
|
|
void search( const uint state0, const uint state1, const uint state2, const uint state3,
|
|
|
|
const uint state4, const uint state5, const uint state6, const uint state7,
|
|
|
|
const uint B1, const uint C1, const uint D1,
|
|
|
|
const uint F1, const uint G1, const uint H1,
|
|
|
|
const uint base,
|
|
|
|
const uint W2,
|
|
|
|
const uint W16, const uint W17,
|
|
|
|
const uint PreVal4, const uint T1,
|
|
|
|
__global uint * output)
|
|
|
|
{
|
|
|
|
|
|
|
|
u W[128];
|
|
|
|
u Vals[8];
|
|
|
|
u t1 = T1;
|
|
|
|
|
|
|
|
Vals[0]=state0;
|
|
|
|
Vals[1]=B1;
|
|
|
|
Vals[2]=C1;
|
|
|
|
Vals[3]=D1;
|
|
|
|
Vals[4]=PreVal4;
|
|
|
|
Vals[5]=F1;
|
|
|
|
Vals[6]=G1;
|
|
|
|
Vals[7]=H1;
|
|
|
|
|
|
|
|
W[2] = W2;
|
|
|
|
W[4]=0x80000000U;
|
|
|
|
W[5]=0x00000000U;
|
|
|
|
W[6]=0x00000000U;
|
|
|
|
W[7]=0x00000000U;
|
|
|
|
W[8]=0x00000000U;
|
|
|
|
W[9]=0x00000000U;
|
|
|
|
W[10]=0x00000000U;
|
|
|
|
W[11]=0x00000000U;
|
|
|
|
W[12]=0x00000000U;
|
|
|
|
W[13]=0x00000000U;
|
|
|
|
W[14]=0x00000000U;
|
|
|
|
W[15]=0x00000280U;
|
|
|
|
W[16] = W16;
|
|
|
|
W[17] = W17;
|
|
|
|
|
|
|
|
W[19] = P1(19) + P2(19) + P3(19);
|
|
|
|
W[18] = P1(18) + P3(18) + P4(18);
|
|
|
|
W[20] = P2(20) + P3(20) + P4(20);
|
2011-06-29 23:38:16 +10:00
|
|
|
uint it = get_local_id(0);
|
2011-06-28 21:11:04 +10:00
|
|
|
|
|
|
|
#ifdef VECTORS4
|
|
|
|
W[3] = base + (get_global_id(0)<<2) + (uint4)(0, 1, 2, 3);
|
|
|
|
#elif defined VECTORS2
|
|
|
|
W[3] = base + (get_global_id(0)<<1) + (uint2)(0, 1);
|
|
|
|
#else
|
|
|
|
W[3] = base + get_global_id(0);
|
|
|
|
#endif
|
|
|
|
|
|
|
|
//the order of the W calcs and Rounds is like this because the compiler needs help finding how to order the instructions
|
|
|
|
W[31] = P2(31) + P4(31);
|
|
|
|
W[18] += P2(18);
|
|
|
|
partround(3);
|
|
|
|
W[19] += P4(19);
|
|
|
|
sharound(4);
|
|
|
|
W[20] += P1(20);
|
|
|
|
sharound(5);
|
|
|
|
W[32] = P2(32) + P4(32);
|
|
|
|
W[21] = P1(21);
|
|
|
|
sharound(6);
|
|
|
|
W[22] = P3(22) + P1(22);
|
|
|
|
W[23] = P3(23) + P1(23);
|
|
|
|
sharound(7);
|
|
|
|
W[24] = P1(24) + P3(24);
|
|
|
|
sharound(8);
|
|
|
|
W[25] = P1(25) + P3(25);
|
|
|
|
sharound(9);
|
|
|
|
W[26] = P1(26) + P3(26);
|
|
|
|
W[27] = P1(27) + P3(27);
|
|
|
|
sharound(10);
|
|
|
|
sharound(11);
|
|
|
|
W[28] = P1(28) + P3(28);
|
|
|
|
sharound(12);
|
|
|
|
W[29] = P1(29) + P3(29);
|
|
|
|
W[30] = P1(30) + P2(30) + P3(30);
|
|
|
|
sharound(13);
|
|
|
|
sharound(14);
|
|
|
|
W[31] += (P1(31) + P3(31));
|
|
|
|
sharound(15);
|
|
|
|
sharound(16);
|
|
|
|
W[32] += (P1(32) + P3(32));
|
|
|
|
sharound(17);
|
|
|
|
sharound(18);
|
|
|
|
sharound(19);
|
|
|
|
sharound(20);
|
|
|
|
sharound(21);
|
|
|
|
sharound(22);
|
|
|
|
sharound(23);
|
|
|
|
sharound(24);
|
|
|
|
sharound(25);
|
|
|
|
sharound(26);
|
|
|
|
sharound(27);
|
|
|
|
sharound(28);
|
|
|
|
sharound(29);
|
|
|
|
sharound(30);
|
|
|
|
sharound(31);
|
|
|
|
sharound(32);
|
|
|
|
sharound2(33);
|
|
|
|
sharound2(34);
|
|
|
|
sharound2(35);
|
|
|
|
sharound2(36);
|
|
|
|
sharound2(37);
|
|
|
|
sharound2(38);
|
|
|
|
sharound2(39);
|
|
|
|
sharound2(40);
|
|
|
|
sharound2(41);
|
|
|
|
sharound2(42);
|
|
|
|
sharound2(43);
|
|
|
|
sharound2(44);
|
|
|
|
sharound2(45);
|
|
|
|
sharound2(46);
|
|
|
|
//for some reason, this is faster than using all sharound2...
|
|
|
|
R(47);
|
|
|
|
sharound(47);
|
|
|
|
R(48);
|
|
|
|
sharound(48);
|
|
|
|
R(49);
|
|
|
|
sharound(49);
|
|
|
|
R(50);
|
|
|
|
sharound(50);
|
|
|
|
R(51);
|
|
|
|
sharound(51);
|
|
|
|
R(52);
|
|
|
|
sharound(52);
|
|
|
|
R(53);
|
|
|
|
sharound(53);
|
|
|
|
R(54);
|
|
|
|
sharound(54);
|
|
|
|
R(55);
|
|
|
|
sharound(55);
|
|
|
|
R(56);
|
|
|
|
sharound(56);
|
|
|
|
R(57);
|
|
|
|
sharound(57);
|
|
|
|
R(58);
|
|
|
|
sharound(58);
|
|
|
|
R(59);
|
|
|
|
sharound(59);
|
|
|
|
R(60);
|
|
|
|
sharound(60);
|
|
|
|
R(61);
|
|
|
|
sharound(61);
|
|
|
|
sharound2(62);
|
|
|
|
sharound2(63);
|
|
|
|
|
|
|
|
W[64]=state0+Vals[0];
|
|
|
|
W[65]=state1+Vals[1];
|
|
|
|
W[66]=state2+Vals[2];
|
|
|
|
W[67]=state3+Vals[3];
|
|
|
|
W[68]=state4+Vals[4];
|
|
|
|
W[69]=state5+Vals[5];
|
|
|
|
W[70]=state6+Vals[6];
|
|
|
|
W[71]=state7+Vals[7];
|
|
|
|
|
|
|
|
W[64 + 8]=0x80000000U;
|
|
|
|
W[64 + 9]=0x00000000U;
|
|
|
|
W[64 + 10]=0x00000000U;
|
|
|
|
W[64 + 11]=0x00000000U;
|
|
|
|
W[64 + 12]=0x00000000U;
|
|
|
|
W[64 + 13]=0x00000000U;
|
|
|
|
W[64 + 14]=0x00000000U;
|
|
|
|
W[64 + 15]=0x00000100U;
|
|
|
|
|
|
|
|
Vals[0]=H[0];
|
|
|
|
Vals[1]=H[1];
|
|
|
|
Vals[2]=H[2];
|
|
|
|
Vals[3]=H[3];
|
|
|
|
Vals[4]=H[4];
|
|
|
|
Vals[5]=H[5];
|
|
|
|
Vals[6]=H[6];
|
|
|
|
Vals[7]=H[7];
|
|
|
|
|
|
|
|
Vals[7] = 0xb0edbdd0 + K[0] + W[64] + 0x08909ae5U;
|
|
|
|
Vals[3] = 0xa54ff53a + 0xb0edbdd0 + K[0] + W[64];
|
|
|
|
|
|
|
|
R(64 + 16);
|
|
|
|
|
|
|
|
sharound(64 + 1);
|
|
|
|
sharound(64 + 2);
|
|
|
|
W[64 + 17] = P1(64 + 17) + P2(64 + 17) + P4(64 + 17);
|
|
|
|
W[64 + 18] = P1(64 + 18) + P2(64 + 18) + P4(64 + 18);
|
|
|
|
sharound(64 + 3);
|
|
|
|
W[64 + 19] = P1(64 + 19) + P2(64 + 19) + P4(64 + 19);
|
|
|
|
sharound(64 + 4);
|
|
|
|
W[64 + 20] = P1(64 + 20) + P2(64 + 20) + P4(64 + 20);
|
|
|
|
sharound(64 + 5);
|
|
|
|
W[64 + 21] = P1(64 + 21) + P2(64 + 21) + P4(64 + 21);
|
|
|
|
sharound(64 + 6);
|
|
|
|
R(64 + 22);
|
|
|
|
sharound(64 + 7);
|
|
|
|
sharound(64 + 8);
|
|
|
|
R(64 + 23);
|
|
|
|
W[64 + 24] = P1(64 + 24) + P3(64 + 24) + P4(64 + 24);
|
|
|
|
sharound(64 + 9);
|
|
|
|
sharound(64 + 10);
|
|
|
|
W[64 + 25] = P1(64 + 25) + P3(64 + 25);
|
|
|
|
W[64 + 26] = P1(64 + 26) + P3(64 + 26);
|
|
|
|
sharound(64 + 11);
|
|
|
|
sharound(64 + 12);
|
|
|
|
W[64 + 27] = P1(64 + 27) + P3(64 + 27);
|
|
|
|
W[64 + 28] = P1(64 + 28) + P3(64 + 28);
|
|
|
|
sharound(64 + 13);
|
|
|
|
sharound(64 + 14);
|
|
|
|
sharound(64 + 15);
|
|
|
|
sharound(64 + 16);
|
|
|
|
sharound(64 + 17);
|
|
|
|
sharound(64 + 18);
|
|
|
|
sharound(64 + 19);
|
|
|
|
sharound(64 + 20);
|
|
|
|
sharound(64 + 21);
|
|
|
|
sharound(64 + 22);
|
|
|
|
sharound(64 + 23);
|
|
|
|
sharound(64 + 24);
|
|
|
|
sharound(64 + 25);
|
|
|
|
sharound(64 + 26);
|
|
|
|
sharound(64 + 27);
|
|
|
|
sharound(64 + 28);
|
|
|
|
sharound2(64 + 29);
|
|
|
|
sharound2(64 + 30);
|
|
|
|
sharound2(64 + 31);
|
|
|
|
sharound2(64 + 32);
|
|
|
|
sharound2(64 + 33);
|
|
|
|
sharound2(64 + 34);
|
|
|
|
sharound2(64 + 35);
|
|
|
|
sharound2(64 + 36);
|
|
|
|
sharound2(64 + 37);
|
|
|
|
sharound2(64 + 38);
|
|
|
|
sharound2(64 + 39);
|
|
|
|
sharound2(64 + 40);
|
|
|
|
sharound2(64 + 41);
|
|
|
|
sharound2(64 + 42);
|
|
|
|
sharound2(64 + 43);
|
|
|
|
sharound2(64 + 44);
|
|
|
|
sharound2(64 + 45);
|
|
|
|
sharound2(64 + 46);
|
|
|
|
sharound2(64 + 47);
|
|
|
|
sharound2(64 + 48);
|
|
|
|
sharound2(64 + 49);
|
|
|
|
R(64 + 50);
|
|
|
|
sharound(64 + 50);
|
|
|
|
R(64 + 51);
|
|
|
|
sharound(64 + 51);
|
|
|
|
R(64 + 52);
|
|
|
|
sharound(64 + 52);
|
|
|
|
R(64 + 53);
|
|
|
|
sharound(64 + 53);
|
|
|
|
R(64 + 54);
|
|
|
|
sharound(64 + 54);
|
|
|
|
R(64 + 55);
|
|
|
|
sharound(64 + 55);
|
|
|
|
sharound2(64 + 56);
|
|
|
|
sharound2(64 + 57);
|
|
|
|
sharound2(64 + 58);
|
|
|
|
sharound2(64 + 59);
|
|
|
|
|
|
|
|
//Faster to write it this way...
|
|
|
|
Vals[3] += K[60] +s1(124) + ch(124);
|
|
|
|
R(64+60);
|
|
|
|
partround(64 + 60);
|
|
|
|
Vals[7] += H[7];
|
|
|
|
|
2011-06-29 23:38:16 +10:00
|
|
|
#define MAXBUFFERS (4 * 512)
|
|
|
|
|
2011-06-28 21:11:04 +10:00
|
|
|
#if defined(VECTORS4) || defined(VECTORS2)
|
|
|
|
if (Vals[7].x == 0)
|
|
|
|
{
|
2011-06-29 23:38:16 +10:00
|
|
|
// Unlikely event there is something here already !
|
|
|
|
if (output[it]) {
|
|
|
|
for (it = 0; it < MAXBUFFERS; it++) {
|
|
|
|
if (!output[it])
|
|
|
|
break;
|
2011-06-28 21:11:04 +10:00
|
|
|
}
|
|
|
|
}
|
2011-06-29 23:38:16 +10:00
|
|
|
output[it] = W[3].x;
|
|
|
|
output[MAXBUFFERS] = 1;
|
2011-06-28 21:11:04 +10:00
|
|
|
}
|
|
|
|
if (Vals[7].y == 0)
|
|
|
|
{
|
2011-06-29 23:38:16 +10:00
|
|
|
it += 512;
|
|
|
|
if (output[it]) {
|
|
|
|
for (it = 0; it < MAXBUFFERS; it++) {
|
|
|
|
if (!output[it])
|
|
|
|
break;
|
2011-06-28 21:11:04 +10:00
|
|
|
}
|
|
|
|
}
|
2011-06-29 23:38:16 +10:00
|
|
|
output[it] = W[3].y;
|
|
|
|
output[MAXBUFFERS] = 1;
|
2011-06-28 21:11:04 +10:00
|
|
|
}
|
|
|
|
#ifdef VECTORS4
|
|
|
|
if (Vals[7].z == 0)
|
|
|
|
{
|
2011-06-29 23:38:16 +10:00
|
|
|
it += 1024;
|
|
|
|
if (output[it]) {
|
|
|
|
for (it = 0; it < MAXBUFFERS; it++) {
|
|
|
|
if (!output[it])
|
|
|
|
break;
|
2011-06-28 21:11:04 +10:00
|
|
|
}
|
|
|
|
}
|
2011-06-29 23:38:16 +10:00
|
|
|
output[it] = W[3].z;
|
|
|
|
output[MAXBUFFERS] = 1;
|
2011-06-28 21:11:04 +10:00
|
|
|
}
|
|
|
|
if (Vals[7].w == 0)
|
|
|
|
{
|
2011-06-29 23:38:16 +10:00
|
|
|
it += 1536;
|
|
|
|
if (output[it]) {
|
|
|
|
for (it = 0; it < MAXBUFFERS; it++) {
|
|
|
|
if (!output[it])
|
|
|
|
break;
|
2011-06-28 21:11:04 +10:00
|
|
|
}
|
|
|
|
}
|
2011-06-29 23:38:16 +10:00
|
|
|
output[it] = W[3].w;
|
|
|
|
output[MAXBUFFERS] = 1;
|
2011-06-28 21:11:04 +10:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#else
|
|
|
|
if (Vals[7] == 0)
|
|
|
|
{
|
2011-06-29 23:38:16 +10:00
|
|
|
if (output[it]) {
|
|
|
|
for (it = 0; it < MAXBUFFERS; it++) {
|
|
|
|
if (!output[it])
|
|
|
|
break;
|
2011-06-28 21:11:04 +10:00
|
|
|
}
|
|
|
|
}
|
2011-06-29 23:38:16 +10:00
|
|
|
output[it] = W[3];
|
|
|
|
output[MAXBUFFERS] = 1;
|
2011-06-28 21:11:04 +10:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
}
|
|
|
|
|