src/2017/knot-hash.lisp @ 5f6c2d777533
2019/08 and fix some test failures
author |
Steve Losh <steve@stevelosh.com> |
date |
Sun, 08 Dec 2019 13:47:53 -0500 |
parents |
b18a8ad31735 |
children |
182bdd87fd9e |
(defpackage :advent/knot-hash
#.cl-user::*advent-use*
(:export :simple-knot-hash :full-knot-hash))
(in-package :advent/knot-hash)
(defun reverse-chunk (vector start length)
(iterate
(repeat (truncate length 2))
(for i :modulo (length vector) :from start)
(for j :modulo (length vector) :downfrom (+ start (1- length)))
(rotatef (aref vector i) (aref vector j))))
(defun simple-knot-hash (lengths)
(iterate
(with numbers = (coerce (alexandria:iota 256) 'vector))
(with current = 0)
(with skip = 0)
(for length :in lengths)
(reverse-chunk numbers current length)
(zapf current (mod (+ % length skip) 256))
(incf skip)
(finally (return (* (aref numbers 0) (aref numbers 1))))))
(defun sparse->dense (numbers)
(iterate
(for i :from 0 :by 16 :below (length numbers))
(collect (reduce #'logxor numbers :start i :end (+ i 16)))))
(defun initial-lengths (string)
(append (map 'list #'char-code string)
(list 17 31 73 47 23)))
(defun full-knot-hash (string &key (result-type 'string))
(iterate
(with lengths = (initial-lengths string))
(with numbers = (coerce (alexandria:iota 256) 'vector))
(with current = 0)
(with skip = 0)
(repeat 64)
(iterate
(for length :in lengths)
(reverse-chunk numbers current length)
(zapf current (mod (+ % length skip) 256))
(incf skip))
(returning
(let ((hash (sparse->dense numbers)))
(ecase result-type
(string (bytes->hex hash))
(list hash)
(vector (coerce hash 'vector))
(integer (bytes->integer hash)))))))