cpu-miner.c 9.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489
  1. /*
  2. Copyright 2010 Jeff Garzik
  3. Distributed under the MIT/X11 software license, see
  4. http://www.opensource.org/licenses/mit-license.php
  5. */
  6. #define _GNU_SOURCE
  7. #include <stdio.h>
  8. #include <stdlib.h>
  9. #include <string.h>
  10. #include <stdbool.h>
  11. #include <unistd.h>
  12. #include <jansson.h>
  13. #include <curl/curl.h>
  14. #include <openssl/bn.h>
  15. #include "sha256_generic.c"
  16. enum {
  17. POW_SLEEP_INTERVAL = 5,
  18. };
  19. static const bool opt_verbose = false;
  20. static const bool opt_debug = false;
  21. struct data_buffer {
  22. void *buf;
  23. size_t len;
  24. };
  25. struct upload_buffer {
  26. const void *buf;
  27. size_t len;
  28. };
  29. struct work {
  30. unsigned char midstate[32];
  31. unsigned char data[128];
  32. unsigned char hash[32];
  33. unsigned char hash1[64];
  34. BIGNUM *target;
  35. };
  36. static void databuf_free(struct data_buffer *db)
  37. {
  38. if (!db)
  39. return;
  40. free(db->buf);
  41. memset(db, 0, sizeof(*db));
  42. }
  43. static size_t all_data_cb(const void *ptr, size_t size, size_t nmemb,
  44. void *user_data)
  45. {
  46. struct data_buffer *db = user_data;
  47. size_t len = size * nmemb;
  48. size_t oldlen, newlen;
  49. void *newmem;
  50. static const unsigned char zero;
  51. if (opt_debug)
  52. fprintf(stderr, "DBG(%s): %p, %lu, %lu, %p\n",
  53. __func__, ptr, (unsigned long) size,
  54. (unsigned long) nmemb, user_data);
  55. oldlen = db->len;
  56. newlen = oldlen + len;
  57. newmem = realloc(db->buf, newlen + 1);
  58. if (!newmem)
  59. return 0;
  60. db->buf = newmem;
  61. db->len = newlen;
  62. memcpy(db->buf + oldlen, ptr, len);
  63. memcpy(db->buf + newlen, &zero, 1); /* null terminate */
  64. return len;
  65. }
  66. static size_t upload_data_cb(void *ptr, size_t size, size_t nmemb,
  67. void *user_data)
  68. {
  69. struct upload_buffer *ub = user_data;
  70. int len = size * nmemb;
  71. if (opt_debug)
  72. fprintf(stderr, "DBG(%s): %p, %lu, %lu, %p\n",
  73. __func__, ptr, (unsigned long) size,
  74. (unsigned long) nmemb, user_data);
  75. if (len > ub->len)
  76. len = ub->len;
  77. if (len) {
  78. memcpy(ptr, ub->buf, len);
  79. ub->buf += len;
  80. ub->len -= len;
  81. }
  82. return len;
  83. }
  84. static json_t *json_rpc_call(const char *url, const char *userpass,
  85. const char *rpc_req)
  86. {
  87. CURL *curl;
  88. json_t *val;
  89. int rc;
  90. struct data_buffer all_data = { };
  91. struct upload_buffer upload_data;
  92. json_error_t err = { };
  93. struct curl_slist *headers = NULL;
  94. char len_hdr[64];
  95. curl = curl_easy_init();
  96. if (!curl)
  97. return NULL;
  98. if (opt_verbose)
  99. curl_easy_setopt(curl, CURLOPT_VERBOSE, 1);
  100. curl_easy_setopt(curl, CURLOPT_URL, url);
  101. curl_easy_setopt(curl, CURLOPT_ENCODING, "");
  102. curl_easy_setopt(curl, CURLOPT_FAILONERROR, 1);
  103. curl_easy_setopt(curl, CURLOPT_TCP_NODELAY, 1);
  104. curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, all_data_cb);
  105. curl_easy_setopt(curl, CURLOPT_WRITEDATA, &all_data);
  106. curl_easy_setopt(curl, CURLOPT_READFUNCTION, upload_data_cb);
  107. curl_easy_setopt(curl, CURLOPT_READDATA, &upload_data);
  108. if (userpass) {
  109. curl_easy_setopt(curl, CURLOPT_USERPWD, userpass);
  110. curl_easy_setopt(curl, CURLOPT_HTTPAUTH, CURLAUTH_BASIC);
  111. }
  112. curl_easy_setopt(curl, CURLOPT_POST, 1);
  113. upload_data.buf = rpc_req;
  114. upload_data.len = strlen(rpc_req);
  115. sprintf(len_hdr, "Content-Length: %lu",
  116. (unsigned long) upload_data.len);
  117. headers = curl_slist_append(headers,
  118. "Content-type: application/json");
  119. headers = curl_slist_append(headers, len_hdr);
  120. headers = curl_slist_append(headers, "Expect:"); /* disable Expect hdr*/
  121. curl_easy_setopt(curl, CURLOPT_HTTPHEADER, headers);
  122. rc = curl_easy_perform(curl);
  123. if (rc)
  124. goto err_out;
  125. if (opt_debug)
  126. printf("====\nSERVER RETURNS:\n%s\n====\n",
  127. (char *) all_data.buf);
  128. val = json_loads(all_data.buf, &err);
  129. if (!val) {
  130. fprintf(stderr, "JSON failed(%d): %s\n", err.line, err.text);
  131. goto err_out;
  132. }
  133. databuf_free(&all_data);
  134. curl_slist_free_all(headers);
  135. curl_easy_cleanup(curl);
  136. return val;
  137. err_out:
  138. databuf_free(&all_data);
  139. curl_slist_free_all(headers);
  140. curl_easy_cleanup(curl);
  141. return NULL;
  142. }
  143. static char *bin2hex(unsigned char *p, size_t len)
  144. {
  145. int i;
  146. char *s = malloc((len * 2) + 1);
  147. if (!s)
  148. return NULL;
  149. for (i = 0; i < len; i++)
  150. sprintf(s + (i * 2), "%02x", p[i]);
  151. return s;
  152. }
  153. static bool hex2bin(unsigned char *p, const char *hexstr, size_t len)
  154. {
  155. while (*hexstr && len) {
  156. char hex_byte[3];
  157. unsigned int v;
  158. if (!hexstr[1]) {
  159. fprintf(stderr, "hex2bin str truncated\n");
  160. return false;
  161. }
  162. hex_byte[0] = hexstr[0];
  163. hex_byte[1] = hexstr[1];
  164. hex_byte[2] = 0;
  165. if (sscanf(hex_byte, "%x", &v) != 1) {
  166. fprintf(stderr, "hex2bin sscanf '%s' failed\n",
  167. hex_byte);
  168. return false;
  169. }
  170. *p = (unsigned char) v;
  171. p++;
  172. hexstr += 2;
  173. len--;
  174. }
  175. return (len == 0 && *hexstr == 0) ? true : false;
  176. }
  177. static bool jobj_binary(const json_t *obj, const char *key,
  178. void *buf, size_t buflen)
  179. {
  180. const char *hexstr;
  181. json_t *tmp;
  182. tmp = json_object_get(obj, key);
  183. if (!tmp) {
  184. fprintf(stderr, "JSON key '%s' not found\n", key);
  185. return false;
  186. }
  187. hexstr = json_string_value(tmp);
  188. if (!hexstr) {
  189. fprintf(stderr, "JSON key '%s' is not a string\n", key);
  190. return false;
  191. }
  192. if (!hex2bin(buf, hexstr, buflen))
  193. return false;
  194. return true;
  195. }
  196. static void work_free(struct work *work)
  197. {
  198. if (!work)
  199. return;
  200. if (work->target)
  201. BN_free(work->target);
  202. free(work);
  203. }
  204. static struct work *work_decode(const json_t *val)
  205. {
  206. struct work *work;
  207. work = calloc(1, sizeof(*work));
  208. if (!work)
  209. return NULL;
  210. if (!jobj_binary(val, "midstate",
  211. work->midstate, sizeof(work->midstate))) {
  212. fprintf(stderr, "JSON inval midstate\n");
  213. goto err_out;
  214. }
  215. if (!jobj_binary(val, "data", work->data, sizeof(work->data))) {
  216. fprintf(stderr, "JSON inval data\n");
  217. goto err_out;
  218. }
  219. if (!jobj_binary(val, "hash1", work->hash1, sizeof(work->hash1))) {
  220. fprintf(stderr, "JSON inval hash1\n");
  221. goto err_out;
  222. }
  223. if (!BN_hex2bn(&work->target,
  224. json_string_value(json_object_get(val, "target")))) {
  225. fprintf(stderr, "JSON inval target\n");
  226. goto err_out;
  227. }
  228. return work;
  229. err_out:
  230. work_free(work);
  231. return NULL;
  232. }
  233. #define ___constant_swab32(x) ((u32)( \
  234. (((u32)(x) & (u32)0x000000ffUL) << 24) | \
  235. (((u32)(x) & (u32)0x0000ff00UL) << 8) | \
  236. (((u32)(x) & (u32)0x00ff0000UL) >> 8) | \
  237. (((u32)(x) & (u32)0xff000000UL) >> 24)))
  238. static inline uint32_t swab32(uint32_t v)
  239. {
  240. return ___constant_swab32(v);
  241. }
  242. static void runhash(void *state, void *input, const void *init)
  243. {
  244. memcpy(state, init, 32);
  245. sha256_transform(state, input);
  246. }
  247. static const uint32_t init_state[8] = {
  248. 0x6a09e667, 0xbb67ae85, 0x3c6ef372, 0xa54ff53a,
  249. 0x510e527f, 0x9b05688c, 0x1f83d9ab, 0x5be0cd19
  250. };
  251. /* suspiciously similar to ScanHash* from bitcoin */
  252. static uint32_t scanhash(unsigned char *midstate, unsigned char *data,
  253. unsigned char *hash1, unsigned char *hash)
  254. {
  255. uint32_t *hash32 = (uint32_t *) hash;
  256. uint32_t *nonce = (uint32_t *)(data + 12);
  257. uint32_t n;
  258. while (1) {
  259. n = *nonce;
  260. n++;
  261. *nonce = n;
  262. runhash(hash1, data, midstate);
  263. runhash(hash, hash1, init_state);
  264. if (hash32[7] == 0) {
  265. if (1) {
  266. char *hexstr;
  267. hexstr = bin2hex(hash, 32);
  268. fprintf(stderr,
  269. "DBG: found zeroes in hash:\n%s\n",
  270. hexstr);
  271. free(hexstr);
  272. }
  273. return n;
  274. }
  275. if ((n & 0xffffff) == 0) {
  276. if (1)
  277. fprintf(stderr, "DBG: end of nonce range\n");
  278. return 0;
  279. }
  280. }
  281. }
  282. static const char *url = "http://127.0.0.1:8332/";
  283. static const char *userpass = "pretzel:smooth";
  284. static void submit_work(struct work *work)
  285. {
  286. char *hexstr = NULL, *s = NULL;
  287. json_t *val, *res;
  288. int i;
  289. unsigned char hash_rev[32];
  290. BIGNUM *hashnum;
  291. char *s_hash, *s_target;
  292. printf("PROOF OF WORK FOUND? submitting...\n");
  293. for (i = 0; i < 32/4; i++)
  294. ((uint32_t *)hash_rev)[i] =
  295. swab32(((uint32_t *)work->hash)[i]);
  296. hashnum = BN_bin2bn(hash_rev, sizeof(hash_rev), NULL);
  297. if (!hashnum) {
  298. fprintf(stderr, "BN_bin2bn failed\n");
  299. return;
  300. }
  301. s_hash = BN_bn2hex(hashnum);
  302. s_target = BN_bn2hex(work->target);
  303. fprintf(stderr, " hash:%s\n hashTarget:%s\n",
  304. s_hash, s_target);
  305. free(s_hash);
  306. free(s_target);
  307. #if 0
  308. i = BN_cmp(hashnum, work->target);
  309. #endif
  310. BN_free(hashnum);
  311. #if 0
  312. if (i >= 0) {
  313. fprintf(stderr, "---INVALID--- proof of work found.\n");
  314. return;
  315. }
  316. #endif
  317. #if 0
  318. /* byte reverse data */
  319. for (i = 0; i < 128/4; i ++)
  320. ((uint32_t *)work->data)[i] =
  321. swab32(((uint32_t *)work->data)[i]);
  322. #endif
  323. /* build hex string */
  324. hexstr = bin2hex(work->data, sizeof(work->data));
  325. if (!hexstr)
  326. goto out;
  327. /* build JSON-RPC request */
  328. if (asprintf(&s,
  329. "{\"method\": \"getwork\", \"params\": [ \"%s\" ], \"id\":1}\r\n",
  330. hexstr) < 0) {
  331. fprintf(stderr, "asprintf failed\n");
  332. goto out;
  333. }
  334. if (opt_debug)
  335. fprintf(stderr, "DBG: sending RPC call:\n%s", s);
  336. /* issue JSON-RPC request */
  337. val = json_rpc_call(url, userpass, s);
  338. if (!val) {
  339. fprintf(stderr, "submit_work json_rpc_call failed\n");
  340. goto out;
  341. }
  342. res = json_object_get(val, "result");
  343. printf("PROOF OF WORK RESULT: %s\n",
  344. json_is_true(res) ? "true (yay!!!)" : "false (booooo)");
  345. json_decref(val);
  346. out:
  347. free(s);
  348. free(hexstr);
  349. }
  350. static int main_loop(void)
  351. {
  352. static const char *rpc_req =
  353. "{\"method\": \"getwork\", \"params\": [], \"id\":0}\r\n";
  354. while (1) {
  355. json_t *val;
  356. struct work *work;
  357. uint32_t nonce;
  358. /* obtain new work from bitcoin */
  359. val = json_rpc_call(url, userpass, rpc_req);
  360. if (!val) {
  361. fprintf(stderr, "json_rpc_call failed\n");
  362. return 1;
  363. }
  364. if (opt_verbose) {
  365. char *s = json_dumps(val, JSON_INDENT(2));
  366. printf("JSON output:\n%s\n", s);
  367. free(s);
  368. }
  369. /* decode result into work state struct */
  370. work = work_decode(json_object_get(val, "result"));
  371. if (!work) {
  372. fprintf(stderr, "work decode failed\n");
  373. return 1;
  374. }
  375. json_decref(val);
  376. /* scan nonces for a proof-of-work hash */
  377. nonce = scanhash(work->midstate, work->data + 64,
  378. work->hash1, work->hash);
  379. /* if nonce found, submit work */
  380. if (nonce) {
  381. submit_work(work);
  382. fprintf(stderr, "sleeping, after proof-of-work...\n");
  383. sleep(POW_SLEEP_INTERVAL);
  384. }
  385. work_free(work);
  386. }
  387. return 0;
  388. }
  389. int main (int argc, char *argv[])
  390. {
  391. return main_loop();
  392. }