mirror of
https://github.com/monero-project/research-lab.git
synced 2025-01-07 03:19:24 +00:00
1003 lines
94 KiB
TeX
1003 lines
94 KiB
TeX
\documentclass[12pt,english]{mrl}
|
|
\usepackage{graphicx}
|
|
\usepackage{listings}
|
|
\usepackage{cite}
|
|
\usepackage{amsthm}
|
|
\newtheorem*{example}{Example}
|
|
|
|
\usepackage[toc,page]{appendix}
|
|
|
|
\renewcommand{\familydefault}{\rmdefault}
|
|
\usepackage[T1]{fontenc}
|
|
\usepackage[latin9]{inputenc}
|
|
\usepackage{color}
|
|
\usepackage{babel}
|
|
\usepackage{verbatim}
|
|
\usepackage{float}
|
|
\usepackage{url}
|
|
\usepackage{amsthm}
|
|
\usepackage{amsmath}
|
|
\usepackage{amssymb}
|
|
\usepackage[unicode=true,pdfusetitle, bookmarks=true,bookmarksnumbered=false,bookmarksopen=false, breaklinks=false,pdfborder={0 0 1},backref=false,colorlinks=true]{hyperref}
|
|
\usepackage{breakurl}
|
|
|
|
|
|
\usepackage{amsmath}
|
|
\usepackage{amsfonts}
|
|
\usepackage{amssymb,enumerate}
|
|
\usepackage{amsthm}
|
|
\usepackage{cite}
|
|
\usepackage{comment}
|
|
\usepackage[all]{xy}
|
|
%\usepackage[notref,notcite]{showkeys}
|
|
\usepackage{hyperref}
|
|
\usepackage{todonotes}
|
|
|
|
% THEOREM ENVIRONMENTS
|
|
|
|
\theoremstyle{definition}
|
|
\newtheorem{lem}{Lemma}[section]
|
|
\newtheorem{cor}[lem]{Corollary}
|
|
\newtheorem{prop}[lem]{Proposition}
|
|
\newtheorem{thm}[lem]{Theorem}
|
|
\newtheorem{soln}[]{Solution}
|
|
\newtheorem{conj}[lem]{Conjecture}
|
|
\newtheorem{Defn}[lem]{Definition}
|
|
\newtheorem{Ex}[lem]{Example}
|
|
\newtheorem{Question}[lem]{Question}
|
|
\newtheorem{Property}[lem]{Property}
|
|
\newtheorem{Properties}[lem]{Properties}
|
|
\newtheorem{Discussion}[lem]{Remark}
|
|
\newtheorem{Construction}[lem]{Construction}
|
|
\newtheorem{Notation}[lem]{Notation}
|
|
\newtheorem{Fact}[lem]{Fact}
|
|
\newtheorem{Notationdefinition}[lem]{Definition/Notation}
|
|
\newtheorem{Remarkdefinition}[lem]{Remark/Definition}
|
|
\newtheorem{rem}[lem]{Remark}
|
|
\newtheorem{Subprops}{}[lem]
|
|
\newtheorem{Para}[lem]{}
|
|
\newtheorem{Exer}[lem]{Exercise}
|
|
\newtheorem{Exerc}{Exercise}
|
|
|
|
\newenvironment{defn}{\begin{Defn}\rm}{\end{Defn}}
|
|
\newenvironment{ex}{\begin{Ex}\rm}{\end{Ex}}
|
|
\newenvironment{question}{\begin{Question}\rm}{\end{Question}}
|
|
\newenvironment{property}{\begin{Property}\rm}{\end{Property}}
|
|
\newenvironment{properties}{\begin{Properties}\rm}{\end{Properties}}
|
|
\newenvironment{notation}{\begin{Notation}\rm}{\end{Notation}}
|
|
\newenvironment{fact}{\begin{Fact}\rm}{\end{Fact}}
|
|
\newenvironment{notationdefinition}{\begin{Notationdefinition}\rm}{\end{Notationdefinition}}
|
|
\newenvironment{remarkdefinition}{\begin{Remarkdefinition}\rm}{\end{Remarkdefinition}}
|
|
\newenvironment{subprops}{\begin{Subprops}\rm}{\end{Subprops}}
|
|
\newenvironment{para}{\begin{Para}\rm}{\end{Para}}
|
|
\newenvironment{disc}{\begin{Discussion}\rm}{\end{Discussion}}
|
|
\newenvironment{construction}{\begin{Construction}\rm}{\end{Construction}}
|
|
\newenvironment{exer}{\begin{Exer}\rm}{\end{Exer}}
|
|
\newenvironment{exerc}{\begin{Exerc}\rm}{\end{Exerc}}
|
|
|
|
\newtheorem{intthm}{Theorem}
|
|
\renewcommand{\theintthm}{\Alph{intthm}}
|
|
|
|
% COMENTS
|
|
|
|
%\newcommand{\ssw}[1]{\footnote{#1}}
|
|
\newcommand{\nt}[2][$^\spadesuit$]{\hspace{0pt}#1\marginpar{\tt\raggedleft
|
|
#1 #2}}
|
|
\newcommand{\dw}[2][$^\spadesuit$]{\nt[#1]{DW:#2}}
|
|
\newcommand{\ssw}[2][$^\spadesuit$]{\nt[#1]{SSW:#2}}
|
|
\newcommand{\ts}[2][$^\spadesuit$]{\nt[#1]{TS:#2}}
|
|
|
|
\newcommand{\ds}{\displaystyle}
|
|
|
|
% CATEGORIES
|
|
|
|
\newcommand{\A}{\mathcal{A}}
|
|
\newcommand{\D}{\mathcal{D}}
|
|
\newcommand{\R}{\mathcal{R}}
|
|
\newcommand{\cat}[1]{\mathcal{#1}}
|
|
\newcommand{\catx}{\cat{X}}
|
|
\newcommand{\caty}{\cat{Y}}
|
|
\newcommand{\catm}{\cat{M}}
|
|
\newcommand{\catv}{\cat{V}}
|
|
\newcommand{\catw}{\cat{W}}
|
|
\newcommand{\catg}{\cat{G}}
|
|
\newcommand{\catp}{\cat{P}}
|
|
\newcommand{\catf}{\cat{F}}
|
|
\newcommand{\cati}{\cat{I}}
|
|
\newcommand{\cata}{\cat{A}}
|
|
\newcommand{\catabel}{\mathcal{A}b}
|
|
\newcommand{\catc}{\cat{C}}
|
|
\newcommand{\catb}{\cat{B}}
|
|
\newcommand{\catgi}{\cat{GI}}
|
|
\newcommand{\catgp}{\cat{GP}}
|
|
\newcommand{\catgf}{\cat{GF}}
|
|
\newcommand{\catgic}{\cat{GI}_C}
|
|
\newcommand{\catgib}{\cat{GI}_B}
|
|
\newcommand{\catib}{\cat{I}_B}
|
|
\newcommand{\catgibdc}{\cat{GI}_{\bdc}}
|
|
\newcommand{\catgicd}{\cat{GI}_{C^{\dagger}}}
|
|
\newcommand{\caticd}{\cat{I}_{C^{\dagger}}}
|
|
\newcommand{\catgc}{\cat{G}_C}
|
|
\newcommand{\catgpc}{\cat{GP}_C}
|
|
\newcommand{\catgpb}{\cat{GP}_B}
|
|
\newcommand{\catgpcd}{\cat{GP}_{C^{\dagger}}}
|
|
\newcommand{\catpcd}{\cat{P}_{C^{\dagger}}}
|
|
\newcommand{\catac}{\cat{A}_C}
|
|
\newcommand{\catab}{\cat{A}_B}
|
|
\newcommand{\catbc}{\cat{B}_C}
|
|
\newcommand{\catabdc}{\cat{A}_{\bdc}}
|
|
\newcommand{\catbbdc}{\cat{B}_{\bdc}}
|
|
\newcommand{\catbb}{\cat{B}_B}
|
|
\newcommand{\catacd}{\cat{A}_{\da{C}}}
|
|
\newcommand{\catbcd}{\cat{B}_{\da{C}}}
|
|
\newcommand{\catgfc}{\cat{GF}_C}
|
|
\newcommand{\catic}{\cat{I}_C}
|
|
\newcommand{\catibdc}{\cat{I}_{\bdc}}
|
|
\newcommand{\catpb}{\cat{P}_B}
|
|
\newcommand{\catpc}{\cat{P}_C}
|
|
\newcommand{\catfc}{\cat{F}'}
|
|
\newcommand{\opg}{\cat{G}}
|
|
\newcommand{\finrescat}[1]{\operatorname{res}\comp{\cat{#1}}}
|
|
\newcommand{\proprescat}[1]{\operatorname{res}\wti{\cat{#1}}}
|
|
\newcommand{\finrescatx}{\finrescat{X}}
|
|
\newcommand{\finrescaty}{\finrescat{Y}}
|
|
\newcommand{\finrescatv}{\finrescat{V}}
|
|
\newcommand{\fincorescatggicd}{\operatorname{cores}\comp{\catg(\caticd)}}
|
|
\newcommand{\finrescatw}{\finrescat{W}}
|
|
\newcommand{\finrescatpc}{\operatorname{res}\comp{\catpc}}
|
|
\newcommand{\finrescatpcr}{\operatorname{res}\comp{\catpc(R)}}
|
|
\newcommand{\finrescatpb}{\operatorname{res}\comp{\catpb}}
|
|
\newcommand{\finrescatpbr}{\operatorname{res}\comp{\catpb(R)}}
|
|
\newcommand{\finrescatgpb}{\operatorname{res}\comp{\catgpb}}
|
|
\newcommand{\finrescatgpbr}{\operatorname{res}\comp{\catgpb(R)}}
|
|
\newcommand{\propcorescatic}{\operatorname{cores}\wti{\catic}}
|
|
\newcommand{\propcorescatgic}{\operatorname{cores}\wti{\catgic}}
|
|
\newcommand{\fincorescatic}{\operatorname{cores}\comp{\catic}}
|
|
\newcommand{\fincorescaticr}{\operatorname{cores}\comp{\catic(R)}}
|
|
\newcommand{\fincorescatir}{\operatorname{cores}\comp{\cati(R)}}
|
|
\newcommand{\finrescatp}{\finrescat{P}}
|
|
\newcommand{\proprescatgpc}{\operatorname{res}\wti{\catgpc}}
|
|
\newcommand{\fincorescaticd}{\operatorname{cores}\comp{\caticd}}
|
|
\newcommand{\finrescatgp}{\finrescat{GP}}
|
|
\newcommand{\finrescatpcd}{\operatorname{res}\comp{\catp_{\da{C}}}}
|
|
\newcommand{\fincorescatggic}{\operatorname{cores}\comp{\catg(\catic)}}
|
|
\newcommand{\fincorescatibdc}{\operatorname{cores}\comp{\catibdc}}
|
|
\newcommand{\fincorescatibdcr}{\operatorname{cores}\comp{\catibdc(R)}}
|
|
\newcommand{\fincorescatgibdc}{\operatorname{cores}\comp{\catgibdc}}
|
|
\newcommand{\fincorescatgibdcr}{\operatorname{cores}\comp{\catgibdc(R)}}
|
|
\newcommand{\fincorescatibr}{\operatorname{cores}\comp{\catib(R)}}
|
|
\newcommand{\finrescatggpc}{\operatorname{res}\comp{\catg(\catpc)}}
|
|
\newcommand{\finrescatg}{\operatorname{res}\comp{\cat{G}}(R)}
|
|
\newcommand{\finrescatgpr}{\operatorname{res}\comp{\cat{GP}(R)}}
|
|
\newcommand{\finrescatpr}{\operatorname{res}\comp{\cat{P}(R)}}
|
|
\newcommand{\finrescatgpc}{\operatorname{res}\comp{\catgp_C(R)}}
|
|
\newcommand{\proprescatpc}{\operatorname{res}\wti{\catp_C(R)}}
|
|
\newcommand{\propcorescatpc}{\operatorname{cores}\wti{\catp_C(R)}}
|
|
\newcommand{\finrescatgpcd}{\operatorname{res}\comp{\catgp_{C^{\dagger}}(R)}}
|
|
\newcommand{\proprescatp}{\proprescat{P}}
|
|
\newcommand{\proprescatgp}{\proprescat{GP}}
|
|
\newcommand{\proprescatx}{\proprescat{X}}
|
|
\newcommand{\proprescaty}{\proprescat{Y}}
|
|
\newcommand{\proprescatv}{\proprescat{V}}
|
|
\newcommand{\proprescatw}{\proprescat{W}}
|
|
\newcommand{\fincorescat}[1]{\operatorname{cores}\comp{\cat{#1}}}
|
|
\newcommand{\propcorescat}[1]{\operatorname{cores}\wti{\cat{#1}}}
|
|
\newcommand{\fincorescatx}{\fincorescat{X}}
|
|
\newcommand{\fincorescati}{\fincorescat{I}}
|
|
\newcommand{\fincorescatgi}{\fincorescat{GI}}
|
|
\newcommand{\fincorescatgir}{\fincorescat{GI(R)}}
|
|
\newcommand{\fincorescatgic}{\operatorname{cores}\comp{\catgi_C(R)}}
|
|
\newcommand{\fincorescatgicd}{\operatorname{cores}\comp{\catgi_{C^{\dagger}}(R)}}
|
|
\newcommand{\propcorescati}{\propcorescat{I}}
|
|
\newcommand{\propcorescatgi}{\propcorescat{GI}}
|
|
\newcommand{\fincorescaty}{\fincorescat{Y}}
|
|
\newcommand{\fincorescatv}{\fincorescat{V}}
|
|
\newcommand{\fincorescatw}{\fincorescat{W}}
|
|
\newcommand{\propcorescatx}{\propcorescat{X}}
|
|
\newcommand{\propcorescaty}{\propcorescat{Y}}
|
|
\newcommand{\propcorescatv}{\propcorescat{V}}
|
|
\newcommand{\propcorescatw}{\propcorescat{W}}
|
|
\newcommand{\cpltrescat}[1]{\operatorname{res}\ol{\cat{#1}}}
|
|
\newcommand{\cpltcorescat}[1]{\operatorname{cores}\ol{\cat{#1}}}
|
|
\newcommand{\cpltrescatw}{\cpltrescat{W}}
|
|
\newcommand{\cpltcorescatw}{\cpltcorescat{W}}
|
|
\newcommand{\gw}{\opg(\catw)}
|
|
\newcommand{\gnw}[1]{\opg^{#1}(\catw)}
|
|
\newcommand{\gnx}[1]{\opg^{#1}(\catx)}
|
|
\newcommand{\gx}{\opg(\catx)}
|
|
\newcommand{\catao}{\cata^o}
|
|
\newcommand{\catxo}{\catx^o}
|
|
\newcommand{\catyo}{\caty^o}
|
|
\newcommand{\catwo}{\catw^o}
|
|
\newcommand{\catvo}{\catv^o}
|
|
|
|
|
|
% DIMENSIONS
|
|
|
|
\newcommand{\pdim}{\operatorname{pd}}
|
|
\newcommand{\pd}{\operatorname{pd}}
|
|
\newcommand{\gdim}{\mathrm{G}\text{-}\!\dim}
|
|
\newcommand{\gkdim}[1]{\mathrm{G}_{#1}\text{-}\!\dim}
|
|
\newcommand{\gcdim}{\gkdim{C}}
|
|
\newcommand{\injdim}{\operatorname{id}}
|
|
\newcommand{\id}{\operatorname{id}}
|
|
\newcommand{\fd}{\operatorname{fd}}
|
|
\newcommand{\fdim}{\operatorname{fd}}
|
|
\newcommand{\catpd}[1]{\cat{#1}\text{-}\pd}
|
|
\newcommand{\xpd}{\catpd{X}}
|
|
\newcommand{\xopd}{\catxo\text{-}\pd}
|
|
\newcommand{\xid}{\catid{X}}
|
|
\newcommand{\wpd}{\catpd{W}}
|
|
\newcommand{\ypd}{\catpd{Y}}
|
|
\newcommand{\gpd}{\catpd{G}}
|
|
\newcommand{\gid}{\catid{G}}
|
|
\newcommand{\catid}[1]{\cat{#1}\text{-}\id}
|
|
\newcommand{\yid}{\catid{Y}}
|
|
\newcommand{\vid}{\catid{V}}
|
|
\newcommand{\wid}{\catid{W}}
|
|
\newcommand{\pdpd}{\catpd\text{-}\pd}
|
|
\newcommand{\idid}{\catid\text{-}\id}
|
|
\newcommand{\pcpd}{\catpc\text{-}\pd}
|
|
\newcommand{\pbpd}{\catpb\text{-}\pd}
|
|
\newcommand{\icdagdim}{\caticd\text{-}\id}
|
|
\newcommand{\icdid}{\caticd\text{-}\id}
|
|
\newcommand{\ibdcid}{\catibdc\text{-}\id}
|
|
\newcommand{\icdim}{\catic\text{-}\id}
|
|
\newcommand{\icid}{\catic\text{-}\id}
|
|
\newcommand{\ibid}{\catib\text{-}\id}
|
|
\newcommand{\pcdim}{\catpc\text{-}\pd}
|
|
\newcommand{\gpcpd}{\catgpc\text{-}\pd}
|
|
\newcommand{\gfpd}{\catgf\text{-}\pd}
|
|
\newcommand{\gppd}{\catgp\text{-}\pd}
|
|
\newcommand{\gfcpd}{\catgfc\text{-}\pd}
|
|
\newcommand{\gpbpd}{\catgpb\text{-}\pd}
|
|
\newcommand{\gicid}{\catgic\text{-}\id}
|
|
\newcommand{\gibid}{\catgib\text{-}\id}
|
|
\newcommand{\gicdagdim}{\catgicd\text{-}\id}
|
|
\newcommand{\gicdid}{\catgicd\text{-}\id}
|
|
\newcommand{\ggpcpd}{\catg(\catpc)\text{-}\pd}
|
|
\newcommand{\ggicdid}{\catg(\caticd)\text{-}\id}
|
|
\newcommand{\ggicid}{\catg(\catic)\text{-}\id}
|
|
\newcommand{\cmdim}{\mathrm{CM}\text{-}\dim}
|
|
\newcommand{\cidim}{\mathrm{CI}\text{-}\!\dim}
|
|
\newcommand{\cipd}{\mathrm{CI}\text{-}\!\pd}
|
|
\newcommand{\cifd}{\mathrm{CI}\text{-}\!\fd}
|
|
\newcommand{\ciid}{\mathrm{CI}\text{-}\!\id}
|
|
|
|
|
|
% OTHER INVARIANTS
|
|
|
|
\newcommand{\Ht}{\operatorname{ht}}
|
|
\newcommand{\col}{\operatorname{col}}
|
|
\newcommand{\depth}{\operatorname{depth}}
|
|
\newcommand{\rank}{\operatorname{rank}}
|
|
\newcommand{\amp}{\operatorname{amp}}
|
|
\newcommand{\edim}{\operatorname{edim}}
|
|
\newcommand{\crs}{\operatorname{crs}}
|
|
\newcommand{\rfd}{\operatorname{Rfd}}
|
|
\newcommand{\ann}{\operatorname{Ann}}
|
|
\newcommand{\mspec}{\mathrm{m}\text{\spec}}
|
|
\newcommand{\soc}{\operatorname{Soc}}
|
|
\newcommand{\len}{\operatorname{length}}
|
|
\newcommand{\type}{\operatorname{type}}
|
|
\newcommand{\dist}{\operatorname{dist}}
|
|
\newcommand{\prox}{\operatorname{\sigma}}
|
|
\newcommand{\curv}{\operatorname{curv}}
|
|
\newcommand{\icurv}{\operatorname{inj\,curv}}
|
|
\newcommand{\grade}{\operatorname{grade}}
|
|
\newcommand{\card}{\operatorname{card}}
|
|
\newcommand{\cx}{\operatorname{cx}}
|
|
\newcommand{\cmd}{\operatorname{cmd}}
|
|
\newcommand{\Span}{\operatorname{Span}}
|
|
\newcommand{\CM}{\operatorname{CM}}
|
|
|
|
% FUNCTORS
|
|
|
|
\newcommand{\cbc}[2]{#1(#2)}
|
|
\newcommand{\ext}{\operatorname{Ext}}
|
|
\newcommand{\rhom}{\mathbf{R}\!\operatorname{Hom}}
|
|
\newcommand{\lotimes}{\otimes^{\mathbf{L}}}
|
|
\newcommand{\HH}{\operatorname{H}}
|
|
\newcommand{\Hom}{\operatorname{Hom}}
|
|
\newcommand{\coker}{\operatorname{Coker}}
|
|
\newcommand{\spec}{\operatorname{Spec}}
|
|
\newcommand{\s}{\mathfrak{S}}
|
|
\newcommand{\tor}{\operatorname{Tor}}
|
|
\newcommand{\im}{\operatorname{Im}}
|
|
\newcommand{\shift}{\mathsf{\Sigma}}
|
|
\newcommand{\othershift}{\mathsf{\Sigma}}
|
|
\newcommand{\da}[1]{#1^{\dagger}}
|
|
\newcommand{\Cl}{\operatorname{Cl}}
|
|
\newcommand{\Pic}{\operatorname{Pic}}
|
|
\newcommand{\proj}{\operatorname{Proj}}
|
|
\newcommand{\End}{\operatorname{End}}
|
|
\newcommand{\cone}{\operatorname{Cone}}
|
|
\newcommand{\Ker}{\operatorname{Ker}}
|
|
\newcommand{\xext}{\ext_{\catx}}
|
|
\newcommand{\yext}{\ext_{\caty}}
|
|
\newcommand{\vext}{\ext_{\catv}}
|
|
\newcommand{\wext}{\ext_{\catw}}
|
|
\newcommand{\aext}{\ext_{\cata}}
|
|
\newcommand{\ahom}{\Hom_{\cata}}
|
|
\newcommand{\aoext}{\ext_{\catao}}
|
|
\newcommand{\aohom}{\Hom_{\catao}}
|
|
\newcommand{\xaext}{\ext_{\catx\!\cata}}
|
|
\newcommand{\axext}{\ext_{\cata\catx}}
|
|
\newcommand{\ayext}{\ext_{\cata\caty}}
|
|
\newcommand{\avext}{\ext_{\cata\catv}}
|
|
\newcommand{\awext}{\ext_{\cata\catw}}
|
|
\newcommand{\Qext}{\ext_{\catw \cata}}
|
|
\newcommand{\pmext}{\ext_{\catp(R)\catm(R)}}
|
|
\newcommand{\miext}{\ext_{\catm(R)\cati(R)}}
|
|
\newcommand{\Qtate}{\comp{\ext}_{\catw \cata}}
|
|
\newcommand{\awtate}{\comp{\ext}_{\cata \catw}}
|
|
\newcommand{\avtate}{\comp{\ext}_{\cata \catv}}
|
|
\newcommand{\pmtate}{\comp{\ext}_{\catp(R) \catm(R)}}
|
|
\newcommand{\mitate}{\comp{\ext}_{\catm(R) \cati(R)}}
|
|
\newcommand{\pcext}{\ext_{\catpc}}
|
|
\newcommand{\pbext}{\ext_{\catpb}}
|
|
\newcommand{\gpcext}{\ext_{\catgpc}}
|
|
\newcommand{\icext}{\ext_{\catic}}
|
|
\newcommand{\gpbext}{\ext_{\catgpb}}
|
|
\newcommand{\gibdcext}{\ext_{\catgibdc}}
|
|
\newcommand{\ibdcext}{\ext_{\catibdc}}
|
|
\newcommand{\gicext}{\ext_{\catgic}}
|
|
\newcommand{\gpext}{\ext_{\catgp}}
|
|
\newcommand{\giext}{\ext_{\catgi}}
|
|
\newcommand{\gicdext}{\ext_{\catgicd}}
|
|
|
|
% IDEALS
|
|
|
|
\newcommand{\ideal}[1]{\mathfrak{#1}}
|
|
\newcommand{\m}{\ideal{m}}
|
|
\newcommand{\n}{\ideal{n}}
|
|
\newcommand{\p}{\ideal{p}}
|
|
\newcommand{\q}{\ideal{q}}
|
|
\newcommand{\fa}{\ideal{a}}
|
|
\newcommand{\fb}{\ideal{b}}
|
|
\newcommand{\fN}{\ideal{N}}
|
|
\newcommand{\fs}{\ideal{s}}
|
|
\newcommand{\fr}{\ideal{r}}
|
|
|
|
% OPERATIONS AND ACCENTS
|
|
|
|
\newcommand{\wt}{\widetilde}
|
|
\newcommand{\ti}{\tilde}
|
|
\newcommand{\comp}[1]{\widehat{#1}}
|
|
\newcommand{\ol}{\overline}
|
|
\newcommand{\wti}{\widetilde}
|
|
|
|
% OPERATORS
|
|
|
|
\newcommand{\ass}{\operatorname{Ass}}
|
|
\newcommand{\supp}{\operatorname{Supp}}
|
|
\newcommand{\minh}{\operatorname{Minh}}
|
|
\newcommand{\Min}{\operatorname{Min}}
|
|
|
|
% MATHBB
|
|
|
|
\newcommand{\bbz}{\mathbb{Z}}
|
|
\newcommand{\bbn}{\mathbb{N}}
|
|
\newcommand{\bbq}{\mathbb{Q}}
|
|
\newcommand{\bbr}{\mathbb{R}}
|
|
\newcommand{\bbc}{\mathbb{C}}
|
|
|
|
% ARROWS
|
|
|
|
\newcommand{\from}{\leftarrow}
|
|
\newcommand{\xra}{\xrightarrow}
|
|
\newcommand{\xla}{\xleftarrow}
|
|
\newcommand{\onto}{\twoheadrightarrow}
|
|
\newcommand{\res}{\xra{\simeq}}
|
|
|
|
|
|
% MAPS
|
|
|
|
\newcommand{\vf}{\varphi}
|
|
\newcommand{\ve}{\varepsilon}
|
|
\newcommand{\Qcomp}{\varepsilon_{\catw \cata}}
|
|
\newcommand{\awcomp}{\varepsilon_{\cata \catw}}
|
|
\newcommand{\avcomp}{\varepsilon_{\cata \catv}}
|
|
\newcommand{\xQcomp}{\vartheta_{\catx \catw \cata}}
|
|
\newcommand{\ayvcomp}{\vartheta_{\cata \caty \catv}}
|
|
\newcommand{\Qacomp}{\varkappa_{\catw \cata}}
|
|
\newcommand{\xaacomp}{\varkappa_{\catx \cata}}
|
|
\newcommand{\aaycomp}{\varkappa_{\cata\caty}}
|
|
\newcommand{\aavcomp}{\varkappa_{\cata\catv}}
|
|
\newcommand{\gpcpccomp}{\vartheta_{\catgpc\catpc}}
|
|
\newcommand{\gpcpbcomp}{\vartheta_{\catgpc\catpb}}
|
|
\newcommand{\gpcgpbcomp}{\vartheta_{\catgpc\catgpb}}
|
|
\newcommand{\gicibcomp}{\vartheta_{\catgic\catib}}
|
|
\newcommand{\gicgibcomp}{\vartheta_{\catgic\catgib}}
|
|
\newcommand{\giciccomp}{\vartheta_{\catgic\catic}}
|
|
\newcommand{\pccomp}{\varkappa_{\catpc}}
|
|
\newcommand{\gpccomp}{\varkappa_{\catgpc}}
|
|
\newcommand{\iccomp}{\varkappa_{\catic}}
|
|
\newcommand{\ibdccomp}{\varkappa_{\catibdc}}
|
|
\newcommand{\icdcomp}{\varkappa_{\caticd}}
|
|
\newcommand{\giccomp}{\varkappa_{\catgic}}
|
|
|
|
% MISCELLANEOUS
|
|
|
|
\newcommand{\y}{\mathbf{y}}
|
|
\newcommand{\te}{\theta}
|
|
\newcommand{\x}{\mathbf{x}}
|
|
\newcommand{\opi}{\operatorname{i}}
|
|
\newcommand{\route}{\gamma}
|
|
\newcommand{\sdc}[1]{\mathsf{#1}}
|
|
\newcommand{\nls}[1]{\mathsf{#1}}
|
|
\newcommand{\cl}{\operatorname{cl}}
|
|
\newcommand{\cls}{\operatorname{cls}}
|
|
\newcommand{\pic}{\operatorname{pic}}
|
|
\newcommand{\pics}{\operatorname{pics}}
|
|
\newcommand{\tri}{\trianglelefteq}
|
|
\newcommand{\Mod}{\operatorname{Mod}}
|
|
\newcommand{\bdc}{B^{\dagger_C}}
|
|
\newcommand{\e}{\mathbf{e}}
|
|
\newcommand{\f}{\mathbf{f}}
|
|
|
|
|
|
% RENEWED COMMANDS
|
|
|
|
\renewcommand{\geq}{\geqslant}
|
|
\renewcommand{\leq}{\leqslant}
|
|
\renewcommand{\ker}{\Ker}
|
|
\renewcommand{\hom}{\Hom}
|
|
|
|
|
|
\newcommand{\normal}{\lhd}
|
|
\newcommand{\normaleq}{\trianglelefteqslant}
|
|
\newcommand{\homrm}[1]{\hom_{_{#1}\catm}}
|
|
\newcommand{\hommr}[1]{\hom_{\catm_{#1}}}
|
|
\newcommand{\cplx}[1]{{#1}_{\bullet}}
|
|
\newcommand{\pext}{\mathrm{P}\!\ext}
|
|
\newcommand{\pextrm}[1]{\pext_{_{#1}\catm}}
|
|
\newcommand{\pextmr}[1]{\pext_{\catm_{#1}}}
|
|
\newcommand{\iext}{\mathrm{I}\!\ext}
|
|
\newcommand{\iextrm}[1]{\iext_{_{#1}\catm}}
|
|
\newcommand{\iextmr}[1]{\iext_{\catm_{#1}}}
|
|
\newcommand{\catmod}[1]{#1\text{-mod}}
|
|
\newcommand{\modcat}[1]{\text{mod-}#1}
|
|
|
|
|
|
\newcommand{\lcm}{\textnormal{lcm}}
|
|
\newcommand{\diff}{\backslash}
|
|
%\setlength{\parindent}{0mm}
|
|
|
|
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% LyX specific LaTeX commands.
|
|
\floatstyle{ruled}
|
|
\newfloat{algorithm}{tbp}{loa}
|
|
\providecommand{\algorithmname}{Algorithm}
|
|
\floatname{algorithm}{\protect\algorithmname}
|
|
|
|
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% Textclass specific LaTeX commands.
|
|
\numberwithin{equation}{section}
|
|
\numberwithin{figure}{section}
|
|
|
|
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% User specified LaTeX commands.
|
|
\usepackage{algpseudocode}
|
|
|
|
\usepackage{subcaption}
|
|
|
|
\numberwithin{equation}{section}
|
|
|
|
|
|
\makeatletter
|
|
|
|
|
|
\makeatletter
|
|
|
|
\newcommand{\h}{\mathcal{H}}
|
|
|
|
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% LyX specific LaTeX commands.
|
|
\floatstyle{ruled}
|
|
\newfloat{algorithm}{tbp}{loa}
|
|
\providecommand{\algorithmname}{Algorithm}
|
|
\floatname{algorithm}{\protect\algorithmname}
|
|
|
|
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% Textclass specific LaTeX commands.
|
|
\numberwithin{equation}{section}
|
|
\numberwithin{figure}{section}
|
|
|
|
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% User specified LaTeX commands.
|
|
\usepackage{algpseudocode}
|
|
|
|
\makeatother
|
|
|
|
\begin{document}
|
|
\begin{frontmatter}
|
|
|
|
\begin{fmbox}
|
|
\hfill\setlength{\fboxrule}{0px}\setlength{\fboxsep}{5px}\fbox{\includegraphics[width=2in]{moneroLogo.png}}
|
|
\dochead{Research bulletin \hfill MRL-0006}
|
|
\title{Difficulty Adjustment Algorithms in Cryptocurrency Protocols}
|
|
\date{12 October 2014}
|
|
\author[
|
|
addressref={mrl},
|
|
email={lab@monero.cc}
|
|
]{\fnm{Surae} \snm{Noether}}
|
|
\author[
|
|
addressref={mrl}
|
|
email={lab@monero.cc}
|
|
]{\fnm{Sarang} \snm{Noether}}
|
|
|
|
|
|
\address[id=mrl]{
|
|
\orgname{Monero Research Lab}
|
|
}
|
|
\end{fmbox}
|
|
|
|
\begin{abstractbox}
|
|
\begin{abstract}
|
|
As of this writing, the algorithm employed for difficulty adjustment in the CryptoNote reference code is known by the Monero Research Lab to be flawed. We describe and illustrate the nature of the flaw and recommend a solution. By dishonestly reporting timestamps, attackers can gain disproportionate control over network difficulty. We verify this route of attack by auditing the CryptoNote reference difficulty adjustment code, which, we reimplement in the Python programming language. We use a stochastic model of blockchain growth to test the CryptoNote reference difficulty formula against the more traditional Bitcoin difficulty formula. This allows us to test our difficulty formula against various hash rate scenarios.
|
|
|
|
This research bulletin has not undergone peer review, and reflects only the results of internal investigation.
|
|
\end{abstract}
|
|
\end{abstractbox}
|
|
\end{frontmatter}
|
|
|
|
\section{Introduction}\label{intro}
|
|
|
|
For blockchain-based currencies such as Monero and Bitcoin, both transaction verifications and mining rewards occur upon block arrivals. The rate of block arrival is tied to both network hash rate and the block difficulty score. A good block difficulty adjustment method tracks network hash rate such that block arrival rate is kept on target; this ensures that currency rewards for mining are paid on schedule and that transaction verification does not stall.
|
|
|
|
The difficulty score of the next block to be added to the blockchain depends on the sequence of timestamps of the blocks preceding it, together with those difficulty scores. We have no way of validating a timestamp, and so timestamps are vulnerable to manipulation; the sequence of timestamps need not even be ordered. Hence, mining reward scheduling and transaction verification are vulnerable to manipulation by way of difficulty manipulation.
|
|
|
|
To our knowledge, at the time of this publication, the study by Kraft in \cite{kraft2015difficulty} is the first and only analysis of blockchain difficulty adjustment. In part, \cite{kraft2015difficulty} formally establishes the relationship between nonhomogeneous Poisson processes, block arrival rate, and network hash rate; we recapitulate some of those arguments less formally here in Section \ref{blah}. Kraft also develops a model of block arrival as a function of exponential hash rate and derives the desired constraint to place upon the model to ensure block arrival times stably approach the target. Kraft refers to the desired constraint as the \textit{time-ratio update}, and develops a fixed point iteration based on time-ratio updating as a new difficulty adjustment method. Under an exponential network hash rate with a constant growth, Kraft showed that the proposed fixed point iteration yields block arrival time that exponentially approach the prescribed target. Although Kraft assumed that network hash rate is an exponential function of time, this result and many of the other results from \cite{kraft2015difficulty} may be naturally extended to piecewise continuous exponential functions (with either positive or negative growth rate).
|
|
|
|
In this research bulletin, we audit the difficulty assessment and adjustment components of the CryptoNote reference code, and we compare this code with the Bitcoin code; we conclude that the CryptoNote reference code is an inadequate solution to difficulty assessment and adjustment. We informally derive a stochastic process model of block arrival as a function of a piecewise constant network hash rate and we verify that, under our model, Kraft's time-ratio updating is a linear maximum-likelihood estimate of network hash rate based on block arrival rate. We propose a difficulty adjustment method for Monero similar to the one proposed by Kraft, but with a few differences. In particular, we utilize fixed point iterations and nonlinear dynamics; with nonlinear choices, we may ensure that difficulty adjustment is relatively insensitive to relatively small changes in block arrival rate, which are likely to correspond with stochastic noise.
|
|
|
|
We use stochastic simulations to compare our derived difficulty adjustment with the CryptoNote reference code and with the Bitcoin code under various piecewise constant hash rate scenarios. In particular, we investigate very large steps up and down. We consider this to be a ``worst-case scenario'' analysis, i.e.\ Lex Luthor's mining pool controlling half the network hash rate switches away to another coin.
|
|
Rather than Kraft's approach, we implement our model of block arrival rate under a piecewise constant network hash rate for a few reasons. Piecewise constant functions are dense in $L^2$ on any compact interval, so these are mathematically convenient functions to work with. Simulating nonhomogeneous Poisson processes with piecewise constant rates of arrival is not difficult. Also, by assuming computers are either mining full-bore or they are not mining at all, we are actually closer to realistic dynamics under a piecewise constant scenario.
|
|
|
|
We also consider a sort of ``timewarp'' attack and demonstrate that the current CryptoNote reference code is vulnerable to a certain mode of attack. By this attack, a user manipulating their timestamps may choose to have no impact on difficulty whatsoever, or may choose to disproportionately increase their impact on difficulty.
|
|
|
|
The model we present herein is simplified in at least two critical ways compared to the true behavior of the network. First, we assume that there are no propagation delays of block discoveries, and second, we assume the usual Nakamoto parent coin selection rule. That is to say, although we will make some comments about parent coin selection rule in Section \ref{modelDef}, we will not investigate variants of parent coin selection rules such as Sompolinsky's GHOST rule in this document.
|
|
|
|
The modeler is stuck with a dilemma. On one hand, as we have done herein, way may assume no propagation delays occur in the cryptocurrency network, which leads to unrealistic behavior. With such an assumption, everyone has perfectly accurate data and thus every user will select the same parent coin, regardless of parent coin selection rule. Furthermore, in Poisson processes, arrivals may not occur simultaneously, and so we never have more than one chain in the blocktree. On the other hand, we may incorporate assumptions about propagation delays in a cryptocurrency network, but these assumptions are equivalent to assumptions about the cryptocurrency network structure (which is unobservable) and speed (which is estimable). In the former case, we are sacrificing realistic competitive behavior so that we are not forced to make unrealistic assumptions about network structure. In the latter case, we are gaining some realistic competitive behaviors but we are assuming much about network structure.
|
|
|
|
One could mitigate the problem of making assumptions about network structure by making empirical measurements such as average and standard deviation of block transmission times, inferring data about the network structure. However, if a researcher were to go down the route of estimating network structure, she would also need to study how rules regarding parent coin selection can impact blockchain dynamics in the presence of competing chains. Studying these things in the context of forking blockchains and selfish mining would certainly be a project worthy of effort. However, this is beyond the scope of this document, which is primarily concerned with the behavior of difficulty scores in response to varying network hash rates. Hence, we go with the former assumption, with no propagation delays and with the Nakamoto parent coin selection rule.
|
|
|
|
Before beginning, we define for the reader some notation we shall use in the sequel. When block arrival rate is viewed as a nonhomogeneous Poisson process, we denote the instantaneous rate of block arrivals on the network as $\lambda(t)$. We wish to keep instantaneous block arrival rate close to our target block arrival rate, $\lambda^*$, which is a known constant. We denote instantaneous network hash rate as $H(t)$, which is unknown and not directly observable (although we run simulations with piecewise constant $H(t)$). We may occasionally refer to a cryptographic hash function $\mathcal{H}$, and a nonce, $x$. We consider the blockchain to be a sequence of blocks, $\mathcal{B}_0$, $\mathcal{B}_1$, $\ldots$. Each block, $\mathcal{B}_i$, consists of a difficulty score and a (possibly false) timestamp:
|
|
|
|
\[\mathcal{B}_0 = (t_0, d_0), \mathcal{B}_1 = (t_1, d_1), \mathcal{B}_2 = (t_2, d_2), \ldots\]
|
|
|
|
We begin counting the genesis block as height $0$, so the block of height $n-1$ is the $n^{th}$ block to arrive; the number of inter-arrival times corresponds with block height this way. The block of height $n$ as $\mathcal{B}_{n}$, the difficulty of the block of height $n$ as $d_{n}$. We may occasionally refer to a nonce, which we denote $x$.
|
|
|
|
For some $m \geq 2$, we will generate a sequence of sample block arrival rates, $\hat{\lambda}_{i}$, for $i=m+1, m+2, m+3, \ldots$, each with sample size $m$. If $m=2$, we are only considering the inter-arrival time between the latest two blocks, for example. From these, we compute network hash rate estimates, $\hat{H}_i$. For some $\ell \leq m$, we also compute $\overline{H}_i$, for each $i = \ell, \ell+1, \ell+2, \ldots$, the moving average of the network hash rate estimates with sample size $\ell$, $\overline{H}_i = \frac{1}{\ell}\sum_{j=1}^{\ell} \hat{H}_{n-j}$. If $\ell = 1$, then the moving average is simplified to our latest estimate of instantaneous hash rate. Notice that $m/\lambda^*$ is the expected length of time to observe $m$ arrivals from a homogeneous Poisson process with rate $\lambda^*$; so, for example, if our block arrival target is $\lambda^{*} = (60.0 s)^{-1}$ and we wish to use the last five minutes of data to compute our sample block arrival rate, we set $m=5$. On the other hand, if we wish to use the last two hours of data, we set $m=120$. Also note that the expected block arrival rate from a homogeneous Poisson process with constant rate $\lambda^*$ is not the expected block arrival rate from the nonhomogeneous Poisson process with non-constant rate $\lambda(t)$, which is proportional to the time-varying network hash rate.
|
|
|
|
|
|
TODO: Verify that, under our model, Kraft's time-ratio updating is a linear maximum-likelihood estimate of network hash rate based on block arrival rate.
|
|
|
|
TODO: Ensure that difficulty adjustment is relatively insensitive to relatively small changes in block arrival rate
|
|
|
|
TODO: Use stochastic simulations to compare our derived difficulty adjustment with the CryptoNote reference code and with the Bitcoin code under various piecewise constant hash rate scenarios. In particular, we investigate very large steps up and down.
|
|
|
|
TODO: Consider a ``timewarp'' attack and demonstrate that the current CryptoNote reference code is vulnerable to a certain mode of attack. By this attack, a user manipulating their timestamps may choose to have no impact on difficulty whatsoever, or may choose to disproportionately increase their impact on difficulty.
|
|
|
|
\section{Blockchain growth model}\label{modelDef}
|
|
|
|
In this section, we define a stochastic process that models the growth of the blockchain over time, and we justify why this model is a good representation of blockchain growth. We also make a few comments about parent coin selection rules. In Section \ref{hashrateattack}, we make a general criticism of traditional blockchain methods and describe a possibly novel route of attack based solely on blockchain dynamics.
|
|
|
|
First, we describe our model, which may be represented formally in the following way; let us worry about justification in a moment. We use as input an unknown, \emph{a priori}, positive hash rate function $H(t) > 0$ with support containing the interval $[0,T)$ for some $T > 0$, where $T$ is a constant denoting the time we stop modeling the network. We could apply more assumptions about $H(t)$ if we feel this is not sufficient. For example, we could also presume that $H(t)$ is bounded below away from zero, say $H(t) > 1$, but this won't change our model very much. We could also presume $H(t)$ is piecewise constant (modeling user hash rates as either on the network at full speed or not), or we could assume $H(t)$ is generated by some other stochastic process, which may or may not be dependent upon the current state of the blockchain. For a general derivation, we simply assume that $H(t)$ is positive with support containing $[0,T)$.
|
|
|
|
We set initial difficulty $d_0 = 1$. A nonhomogeneous Poisson process governing block arrivals with counting process $N(t)$ is observed. Recall that $N(t)$ corresponds to the total number of block arrivals on the time interval $[0,t)$. Denote the instantaneous rate of block arrivals at time $t < T$ as $\lambda(t) = H(t)/d_{N(t)}$. The nonhomogeneous Poisson process gives rise to block arrival times $t_0, t_1, t_2, \ldots$, which may then be manipulated by a malicious user, giving rise to \emph{manipulated} block arrival times $\hat{t}_0, \hat{t}_1, \ldots$. The value of the denominator in the block arrival rate, $d_{N(t)}$, is block difficulty. This value is generated from a function that is determined by the manipulated block arrival times and the difficulties of the top blocks. That is to say, we have some function $\phi$ and we set
|
|
\[d_{N(t)} = \phi((\hat{t}_i, d_i)_{i=0}^{N(t)-1})\]
|
|
|
|
We define a blockchain in this context, then, as the stochastic process consisting of the sequence of ordered pairs $(\hat{t}_i, d_i)_{i=0}^{N(t)}$, where the difficulty $d_n = \phi((\hat{t}_i, d_i)_{i=0}^{n-1})$. Observe two facts: first, the \emph{latest} block (the block with $t_i = \max_j (t_j)$) is not necessarily the top block (which has height $n-1$) because the manipulated timestamps need not occur in the same order as their indices. Second, we have defined this notion so generally that it encompasses a variety of difficulty adjustment methods. For example, Bitcoin has a difficulty adjustment period of $2016$ blocks. That is to say, for Bitcoin, the function $\phi$ is dependent upon $n$, so that if $n \cong 0 (mod p)$ for some integer $p$, then for $k=0, 1, 2, \ldots, 2015$, $d_n = d_{n+k}$.
|
|
|
|
To justify this model, recall how the proof-of-work competition for block validation works. Users collect transactions into blocks for validation and try to hash nonces together with block data in order to find a hash smaller than a certain target. That is to say, if a user on the network finds some $x$ such that $d_{n}\cdot \mathcal{H}(\mathcal{B} + x) < \texttt{fixed\_target}$, they have earned the right to declare a block as valid and they usually receive a block reward in the form of a coinbase transaction. Everyone who is working off of the same copy of the blockchain as each other will compute difficulty, $d_{n}$ in the same way. When a user finds such an $x$, they publish $\mathcal{B}$, $x$, and a possibly untrustworthy timestamp. They then recompute their difficulty. When a user hears about a new block, they add it to their copy of the blockchain and recompute their difficulty before trying more hashes.
|
|
|
|
Since the goal of the proof-of-work game is to find $x$ such that $d_{n} \cdot \mathcal{H}(\mathcal{B} + x) < \texttt{fixed\_target}$ and since the output of a good hash function is, in practice, indistinguishable from a uniform distribution, this implies the probability that any given nonce is a success is precisely$\texttt{fixed\_target}/d_{n}$. That is to say, each trial testing a nonce is a Bernoulli trial (weighted coin flip) with probability of success inversely proportional to the difficulty of the next block to be added. Without loss of generality, we may choose $\texttt{fixed\_target}=1$; doing so calibrates the difficulty score such that $d_n=1$ corresponds with a success on each and every nonce. Hence, each trial testing a nonce is a Bernoulli trial (weighted coin flip) with probabiltiy of success $1/d_n$.
|
|
|
|
Arrivals of heads-up coins in a sequence of coin flips, under suitable conditions, can be well approximated with a Poisson process. If we are given a constant $\lambda > 0$, a sequence of probabilities $\left\{p_n\right\}$ satisfying $n p_n \longrightarrow \lambda$ and a few other suitable conditions, then a binomial random variable $\text{Bin}(n,p_n)$ can be roughly approximated with the Poisson distribution with rate $\lambda$. The suitable conditions are technical but a good rule of thumb is that when $n \geq 20$ and $p \leq 1/20$, or if $n \geq 100$ and $np \leq 10$, we may use the Poisson approximation. In our case, we are talking about cryptocurrency networks with $n$ typically much greater than thousands of hashes per second, and with probability of success far lower than $1$ in $20$ nonces. Hence, we may approximate the proof-of-work block competition for validation by a Poisson process with rate $\lambda = np$. More technical arguments can be made toward this equivalence, but we shall be satisfied and proceed with a Poisson process model.
|
|
|
|
Note that, in the Poisson process described above, $n$ is proportional to our global network hashrate, $H(t)$, and $p=1/d_n$. The block arrival rate will be $\lambda(t) = np = \beta H(t)/d_n$. Consider $\beta$. If difficulty is $1$, every nonce is a success, and if $H(t) = 1 H/s$, then a block will arrive with probability $1$ by time $t=1.0 s$, yielding a block arrival rate of $1.0 s^{-1}$. Hence, we have $1.0 s^{-1} = \beta \frac{1.0 H/s}{1}$. We conclude $\beta = 1$ and we are free to use the relationship
|
|
\[\lambda = \frac{H}{d}\]
|
|
Furthermore, since users assign difficulty $d_n$ according to some function of the preceding difficulties and timestamps, this is a nearly complete justification of our model in question. As mentioned in Section \ref{intro}, we largely ignore parent-coin selection rules in this document, although many interesting questions may arise (see Section \ref{furtherQuestions}). For completeness, however, we shall discuss parent-coin selection for a moment.
|
|
|
|
Many decision rules for choosing parent coins could be constructed. One common misconception of the Bitcoin whitepaper, \cite{nakamoto2008bitcoin}, is that Satoshi Nakamoto did not propose mining on the longest chain, but, in fact, the chain with the largest cumulative difficulty. In the case of Bitcoin, for which thousands of blocks in a row have the same difficulty score, this is often (although not always) equivalent to selecting a chain with the highest block height. Alternative proposals have been made, e.g\ in \cite{sompolinsky2013accelerating} Sompolinsky and Zohar recommend a greedy algorithm seeking the heaviest subtree approach. Users determine their parent coin by climbing the blocktree from the genesis block upward, and each time they are faced with a branch, they take the branch leading to the heaviest subtree. When they have finished climbing the blocktree, they have found their parent coin.
|
|
|
|
These approaches are, in fact, identical if we view as a generalized Nakamoto rule: assign a generalized ``score'' to each block and choose the parent block by selecting the block with the largest \textit{cumulative} score of all preceding blocks. The only difference between these two methods is how the scores are computed. The original Nakamoto recommendation was to use difficulty of block $\mathcal{B}$ as the score function, whereas the recommendation by Sompolinsky and Zohar is to use the cumulative heaviness (in terms of difficulty) of the subtree that has a block $\mathcal{B}$ as its root. From this perspective, we see that the generalized Nakamoto rule may be sensitive to both choices of score function and difficulty equation.
|
|
|
|
This is all we shall say on parent-coin selection rules in this document. With that, we have justified the model of choice All that remains is the focus of this document, which is to decide how difficulty is to be computed under the model described above.
|
|
|
|
\subsection{Consequences of the model: Stalling}\label{hashrateattack}
|
|
|
|
In this section, we discuss how the model presented in Section \ref{modelDef} is sensitive to sudden changes in hash rate.
|
|
|
|
Consider a sudden change in hash rate; any traditional blockchain scheme is vulnerable to stalling if a large portion of the network suddenly withdraws its participation. We are creating a discrete object, the blockchain, from a continuous-time stochastic process; if the underlying forces giving rise to the creation of the blockchain are in a very rapid state of change between blocks, the results may be disastrous. Consider the scenario in which true network hash rate drops by several orders of magnitude very suddenly. For example, if Lex Luthor has control of a very large proportion of some cryptocurrency network, say $99.9\%$, Lex has some options. He can certainly re-write the history of the blockchain and give himself all of the money (the usual $50\%$ attack route). But he could also simply decide to switch off all of his machines.
|
|
|
|
Difficulty $d_{n+1}$ remains unchanged, as it is based on an estimate of previous block arrival rates. After all, difficulty only updates and adjusts upon receipt of a new block timestamp. Now, however, block arrival rate is very close to zero blocks per second. Now the problem is that the network comes to a standstill; blocks are not arriving because difficulty was very high and a very large actor in the mining space took their equipment offline. Difficulty will not change before the next block arrives, which could take an arbitrarily long period of time. No one is mining because blocks are very rare, and no one is on the network, so blocks remain rare forever, killing transaction processing capabilities. Of course, a slow decay or smaller jumps in hash rate will not have this effect.
|
|
|
|
One standard assumption in cryptocurrencies is that no single attacker controls more than $50\%$ of the network (otherwise, the ledger may be re-written, and presumably the currency will lose all value). Hence, we wish to construct a difficulty algorithm robust against a sudden halving or doubling of net hash rate. If hash rate is cut in half but difficulty remains the same, block inter-arrival times will be doubled, and if hash rate is doubled, the inter-arrival times will be halved. Hence, under the standard $50\%$ attacker assumption, no network will be stalled forever. However, with sufficiently long block adjustment periods, this can still be disastrous for a currency. Consider a situation in which the Bitcoin network hash rate is cut in half immediately after a difficutly adjustment. Transactions will now be processed in $20$ minute blocks, rather than $10$ minute blocks, and thus transaction processing rate on the network will be halved. Furthermore, due to the two week adjustment period in the Bitcoin difficulty adjustment code, we can expect the network to remain in this state \emph{for two weeks}. This would be an agonizingly long period of time for transaction processing speeds to be cut in half. Of course, the Bitcoin network need not be worried about such a scenario, for the size of the Bitcoin network is a good insulator against such attacks.
|
|
|
|
Due to this, we are interested in measuring the robustness of the two difficulty equations of interest (the CryptoNote reference code and our new difficulty equation) when exposed to sudden large changes in hash rate. In Section \ref{compareToDropsAndJumps}, we will investigate hash rate functions that are piecewise constant, and we will investigae how rapidly the two difficulty adjustment algorithms respond to hash rate changes of varying magnitude.
|
|
|
|
\subsection{Consequences of the model: Orphaned blocks}\label{orphanedBlocks}
|
|
In this seciton, we discuss how block arrival rate relates to the production of orphan blocks. Indeed, we lament the speed of light's inevitable restriction on propagation of data and the resulting orphaned coins. Regardless of parent coin selection rule, regardless of network structure, no matter the speed of our network, we will still occasionally see computers producing orphan blocks. If all users mine honestly, this is the primary source of orphan blocks, but there is nothing to prevent users from colluding in a selfish mining attack, as described in \cite{eyal2014majority}. Due to some conflict in the community over the term ``orphaned block,'' a more descriptive term perhaps could be \textit{dead branches of the blocktree}.
|
|
|
|
Consider the following example modeled after Bitcoin with a block target of $\lambda_{\text{target}}=1/600$ blocks per second. For the sake of argument, presume the network is a complete graph with a constant $N$ nodes and with the same transmission speed between any two nodes. We could model transmission time in the network between node $i$ and node $j$ as $\mu + e_{ij}$ where each $e_{ij}$ is a random variable with $E(e_{ij}) = 0$ and the mean propagation time is $\mu$. In in \cite{decker2013information}, Decker and Wattenhofer measured the average Bitcoin propagation time between nodes to be $\mu \approx 6.5$ seconds. Although the mean propagation times reported in that publication are somewhat out of date, they are a sufficient starting point for this discussion. When a node finds a successful nonce at time $t$, they announce this fact on the network. By time $t+6.5$, only half the network (on average) has heard of this new block. What about a computer in the second half of the network, the blind half? A node in that half of the network hashing between time $t$ and $t+6.5$ will have chosen a different parent coin because it does not have as much information as the nodes in the first half of the network. Furthermore, one or the other branch will become part of the main chain eventually; these two events are mutually exclusive. Hence the total number of coin flips wasted is, with high probability, bounded by
|
|
\[\text{Orphaned flips} \geq E\left[\text{Number of heads found by half the network in }[t,t+\mu)\right]\]
|
|
and if hash power is distributed approximately uniformly, we can take a simple average. This approximation works out to be $\mu\cdot \lambda_{\text{target}}$ flips, where $\lambda_{\text{target}}$ denotes our target block arrival rate. In this example, this would be about $0.01083$ orphaned blocks on average. In expectation, a Bitcoin miner can expect about $1.083$ of their blocks orphaned for every $100$ blocks mined (once every $16$-ish hours). In probability, a Bitcoin miner can be $95\%$ confident that at least one block has been orphaned for every $275$ blocks mined. An individual with $1\%$ of the Bitcoin network's hashing power can expect, on average, to receive $1$ out of every $100$ blocks rewards, which would occur with rate $\lambda_{\text{target}}/100 = 1/1000$ blocks per minute. Hence, to obtain $275$ blocks would require about $190$ days, and such a miner can be $95\%$ confident she will see one orphaned block every $190$ days.
|
|
|
|
On the other hand, if we have a coin modeled after Monero, with a block target time of $\lambda_{\text{target}}=1.0$ blocks per minute (compared with Bitcoin's $\lambda_{\text{target}} = 0.1$ blocks per minute), but we choose an otherwise similar setup as above, we will see about $0.1083$ orphaned blocks per mined block on average, corresponding to a $6.5$-second propagation time. In expectation, a Monero miner can expect $1.083$ blocks orphaned/wasted for every $10$ mined blocks. In probability, a Monero miner can be $95\%$ confident that at least one block has been orphaned every $26$ blocks mined. If a miner has $1\%$ of the Monero network's hashing power, such a miner can expect to receive $1$ out of every $100$ block rewards, which would occur with rate $\lambda_{\text{target}}/100 = 1/100$ blocks per minute. With $95\%$ probability, such a miner can be $95\%$ confident she will see one orphaned block every $1.8$ days.
|
|
|
|
Before drawing any conclusions, note that these estimates rely upon \textit{expectations}, which we justified by assuming uniformity in hashing power. This is, of course, very false in the case of Bitcoin, which can see many orders of magnitude difference in performance between various mining rigs. In the case of Monero, which has a somewhat egalitarian proof-of-work algorithm, this assumption is less problematic. Either way, these values should not be taken as particularly precise. Rather, these values are intended give a broad idea of a ``first-glimpse'' into orphan coin analysis. Having said that, notice that the rate of orphan block arrivals is proportional to the target block arrival rate: the arrival rate of Monero blocks is ten times the arrival rate of Bitcoin blocks, Monero miners can expect around $10\%$ of blocks to be orphan blocks, and Bitcoin miners can expect around $1\%$ of blocks to be orphan blocks. Hence, setting target block arrival rates lower (say, one block every two minutes or three minutes) will dramatically reduce the rate of orphans on the Monero network.
|
|
|
|
|
|
|
|
|
|
\section{Difficulty Adjustment Formula}\label{myCode}
|
|
|
|
In this section we use the model defined in Section \ref{modelDef} to determine our difficulty adjustment function and we present it as a sequence of steps. Recall that the overall goal is to keep the true rate $\lambda(t)$ to be approximately $\lambda^{*}$, the target block arrival time. Say that a user is given a blockchain of height $n-1$, which has a sequence of (possibly out-of-order) timestamps and difficulties
|
|
\[\mathcal{B}_0 = (t_0, d_0), \mathcal{B}_1 = (t_1, d_1), \mathcal{B}_2 = (t_2, d_2), \ldots, \mathcal{B}_{n-1} = (t_{n-1},d_{n-1})\]
|
|
Before mining, the user shall place the timestamps in order, i.e.\ the \emph{order statistics}, and consider the top $m$ of these:
|
|
\[t_{(n-1)} > t_{(n-2)} > \ldots > t_{(n-m)}\]
|
|
Define $\Delta_{n-1} T = t_{(n-1)} - t_{(n-m)}$ for each $n \geq m$. Notice that each $\Delta_{n} T$ is computed from the order statistics of a different sequence, and timestamps may occur out of order. We now compute the sample block arrival rate at height $n-1$
|
|
\[\hat{\lambda}_{n-1} := m/\Delta_{n-1}T\]
|
|
This is a running estimate of the instantaneous block arrival rate based on the maximum likelihood of a Poisson process with a constant rate/intensity. Each time a new block arrives, the \emph{order statistics} are recomputed, and the sample block arrival rate is recomputed from the new \emph{order statistics}. Our goal is to keep these samples very close to our target, $\lambda^*$. Further, since $\lambda = H/d$, each of these block arrival rates come equipped with an estimate of network hash rate:
|
|
\[\hat{H}_{n-1} = d_{n-1} \hat{\lambda}_{n-1} = m d_{n-1}/\Delta_{n-1} T\]
|
|
|
|
While time series analysis tools may be used to make predictions of $\hat{H}_n$, we suspect that such predictions will not outperform simpler approaches\footnote{This may be the subject of a future research bulletin.}. In our case, for difficulty adjustment, it is unnecessary to predict future values of hash rate (which is a complicated problem) if we are capable of producing an accurate estimate of instantaneous hash rate (which is a less complicated problem). Thus we choose our sequence of difficulties $d_n$ so as to allow difficulty to track the simple moving average of these network hashrate estimates; this is, in fact, equivalent to presuming network hash rate will remain static and match our current estimate until the next block arrives. We compute the sample mean of the last $\ell$ estimates of network hash rate
|
|
\begin{align*}
|
|
\overline{H}_{n-1} =& \frac{1}{\ell}\sum_{i=1}^{\ell} \hat{H}_{n-i}\\
|
|
=& \frac{1}{\ell} \sum_{i=1}^{\ell} d_{n-i} \hat{\lambda}_{n-i}\\
|
|
=& \frac{1}{\ell} \sum_{i=1}^{\ell} m d_{n-i} /\Delta_{n-i} T
|
|
\end{align*}
|
|
Since $\lambda = H/d$, we may keep $\lambda(t)$ close to $\lambda^*$ by choosing
|
|
\begin{align}
|
|
d_{n+1} =& \overline{H}_n/\lambda^* \notag \\
|
|
=& \frac{\overline{H}_n}{\overline{H}_{n-1}} \frac{\overline{H}_{n-1}}{\lambda^*}\notag \\
|
|
d_{n+1}=& \frac{\overline{H}_n}{\overline{H}_{n-1}} d_{n}\label{diffeq}
|
|
\end{align}
|
|
That is to say, hash rate and difficulty should move in lockstep. At this point, we should be somewhat comforted. If average network hash rate has doubled, difficulty should probably double, and if it has halved, difficulty should halve. Notice that if we were to set $\ell = 1$, we would abandon the simple moving average and simply have the instantaneous estimate
|
|
\[d_{n+1} = \frac{\hat{H}_{n}}{\hat{H}_{n-1}} d_n\]
|
|
Returning to the more general case, we have
|
|
\begin{align*}
|
|
d_{n+1}=& \frac{\overline{H}_n}{\overline{H}_{n-1}} d_{n}\\
|
|
=& \frac{\frac{1}{\ell} \sum_{i=1}^{\ell} d_{n-i+1} \hat{\lambda}_{n-i+1}}{\frac{1}{\ell} \sum_{i=1}^{\ell} d_{n-i} \hat{\lambda}_{n-i}} d_n\\
|
|
=& \frac{\sum_{i=1}^{\ell} m d_{n-i+1}/\Delta_{n-i+1} T}{\sum_{i=1}^{\ell} md_{n-i}/\Delta_{n-i} T}d_n\\
|
|
=& \frac{\sum_{i=1}^{\ell} d_{n-i+1}/\Delta_{n-i+1} T}{\sum_{i=1}^{\ell} d_{n-i}/\Delta_{n-i} T}d_n
|
|
\end{align*}
|
|
We may expand this a bit to write it as a difference equation:
|
|
\begin{align*}
|
|
d_{n+1}=& \frac{\frac{d_{n}}{\Delta_n T} + \frac{d_{n-1}}{\Delta_{n-1} T} + \cdots + \frac{d_{n-\ell + 1}}{\Delta_{n-\ell + 1} T}}{\frac{d_{n-1}}{\Delta_{n-1} T} + \cdots + \frac{d_{n-\ell }}{\Delta_{n-\ell } T}}d_n\\
|
|
=& \left(1 + \frac{\frac{d_n}{\Delta_n T} - \frac{d_{n-\ell}}{\Delta_{n-\ell} T}}{\frac{d_{n-1}}{\Delta_{n-1} T} + \cdots + \frac{d_{n-\ell }}{\Delta_{n-\ell } T}} \right) d_n\\
|
|
d_{n+1} - d_n =& \frac{\frac{d_n}{\Delta_n T} - \frac{d_{n-\ell}}{\Delta_{n-\ell} T}}{\frac{d_{n-1}}{\Delta_{n-1} T} + \cdots + \frac{d_{n-\ell }}{\Delta_{n-\ell } T}} d_n
|
|
\end{align*}
|
|
We will use these formulae interchangeably as our difficulty adjustment function, but for shorthand, we may remember the general inductive rule
|
|
\[d_{n+1}= \frac{\overline{H}_n}{\overline{H}_{n-1}} d_{n}\]
|
|
|
|
We now describe how to implement this difficulty adjustment algorithm. Presume we are given a blockchain of the form
|
|
\[(t_0, d_0), (t_1, d_1), \ldots, (t_{n-1}, d_{n-1})\]
|
|
and a difficulty score of the next block to be added, $d_n$. Furthermore, presume some timestamp, $t_n$, has just been announced on the network and we wish to compute $d_{n+1}$. Denote the sequence of hash rate estimates
|
|
\[\texttt{HASH} = \hat{H}_{n-1}$, $\hat{H}_{n-2}, \ldots \hat{H}_{n-k}\]
|
|
where $k = \min(n,m)$. Denote the average of this sequence
|
|
\[\texttt{AVG\_HASH} = \frac{1}{600} \sum_{i=1}^{600} \hat{H}_{n-i}\]
|
|
|
|
Recall that we reject a block (and its timestamp, $t_n$) as illegitimate if its timestamp is too far away from the latest timestamp. In particular, the CryptoNote reference code rejects a block if $\text{median}(t_{n-i})_{i=1}^{m} > t_n$ or if $t_n > 7200 + \max(t_{n-i})_{i=1}^{m}$. We see no reason to change this window of acceptance. We execute the following procedure for any $n \geq 1$:
|
|
\begin{enumerate}[(1)]
|
|
\item Append $(t_n, d_n)$ to the blockchain if the timestamp $t_n$ is legitimate.
|
|
\item Store the top $m$ blocks' timestamps temporarily as, say, $\hat{t}_1$, $\hat{t}_2$, $\ldots$, $\hat{t}_{m}$. If $n < m$, store as many as we have.
|
|
\item Sort these stored timestamps into their order statistics, $\hat{t}_{(1)} < \hat{t}_{(2)} < \ldots < \hat{t}_{(m)}$.
|
|
\item Compute the span of time represented by these timestamps, $\Delta_{n} T = \hat{t}_{(m)} - \hat{t}_{(1)}$.
|
|
\item If $n < m$, then set $\hat{\lambda}_{n} := n/\Delta_n T$. Otherwise, set $\hat{\lambda}_{n} := m/\Delta_n T$.
|
|
\item Compute the estimated hash rate during this span of time, $\hat{H}_{n} = \hat{\lambda}_{n} d_{n}$.
|
|
\item Prepend $\hat{H}_n$ to $\texttt{HASH}$.
|
|
\item If $n \geq m$, remove $\hat{H}_{n-m}$ from $\texttt{HASH}$.
|
|
\item Compute the new average of the hash rate estimates. If $n < m$, set $\texttt{NEW\_AVG\_HASH} = \frac{1}{n} \sum_{i=0}^{n-1} \hat{H}_{n-i}$. Otherwise, set $\texttt{NEW\_AVG\_HASH} = \frac{1}{m} \sum_{i=0}^{m-1} \hat{H}_{n-i}$.
|
|
\item If $n < m$, set new difficulty $d_{n+1} = 1$. Otherwise, set new difficulty to $d_{n+1} = d_{n} \cdot \texttt{NEW\_AVG\_HASH}/\texttt{AVG\_HASH}$.
|
|
\item Re-set $\texttt{AVG\_HASH} \leftarrow \texttt{NEW\_AVG\_HASH}$.
|
|
\item Wait until a new block timestamp, $t_{n+1}$, to arrive and go back to step (1) when it does.
|
|
\end{enumerate}
|
|
|
|
|
|
\section{The Current CryptoNote Code}\label{refCode}
|
|
|
|
In Appendix \ref{referenceCode}, we present a re-implementation by Sarang Noether of the reference CryptoNote difficulty assessment method in python. We describe that difficulty assessment method here and make comparisons between their difficulty adjustment algorithm and ours.
|
|
|
|
The CryptoNote reference code computes the next difficulty score in the following manner:
|
|
\begin{enumerate}[(1)]
|
|
\item Store the top $720$ blocks' timestamps temporarily as, say, $\hat{t}_1$, $\hat{t}_2$, $\ldots$, $\hat{t}_{720}$.
|
|
\item Store the top $720$ blocks' difficulty scores temporarily as, say,
|
|
$\hat{d}_1$, $\hat{d}_2$, $\ldots$, $\hat{d}_{720}$.
|
|
\item Sort the top timestamps in increasing order as $\hat{t}_{(1)}$, $\hat{t}_{(2)}$, $\ldots$, $\hat{t}_{(720)}$
|
|
\item Eliminate outlying bottom $1/12$ and outlying top $1/12$ of each of these lists, leaving $(\hat{t}_{(61)}, \hat{t}_{(62)}, \ldots, \hat{t}_{(660)})$ and $(\hat{d}_{61}, \hat{d}_{62}, \ldots, \hat{d}_{660})$.
|
|
\item Compute the span of time represented by these timestamps, $\Delta t = \hat{t}_{(660)} - \hat{t}_{(60)}$.
|
|
\item Compute the sum of the difficulties, $D = \sum_{i=61}^{660} \hat{d}_i$.
|
|
\item The difficulty of the next block, $\mathcal{B}_{H+1}$, given some target block arrival rate $\lambda$ blocks per unit time, is then computed from the formula
|
|
\begin{align}
|
|
\hat{d}_{721} :=&\frac{D/\lambda + \Delta T - 1}{\Delta T} \label{diffFormInitial}
|
|
\end{align}
|
|
\end{enumerate}
|
|
|
|
We make some observations about this approach. First, as usual, timestamps added to the blockchain need not occur in order. In fact, we have no reason to expect that they will. Sorting the timestamps seems reasonable (in fact, we do this in our proposed algorithm in Section \ref{myCode}). Further, removing the outlying elements from the list also seems reasonable. However, notice that the timestamps were sorted separately from the difficulties. Hence, while an outlying timestamp may be removed, it's associated difficulty score is still included in the computation and the associated difficulty score of some other block is subsequently removed as an outlier (although the timestamp associated with the removed difficulty score may not have been an outlier originally!). We discuss this problem in detail in Section \ref{reorderingproblems}.
|
|
|
|
Second, notice that the formula for $\hat{d}_{721}$ may seem a bit strange. Recall that, given $n$ sample inter-arrival times $S_1, S_2, \ldots, S_n$, then the maximum likelihood estimate of the Poisson rate is $\hat{\lambda} = n/\sum_i S_i$ or rather, if we have $n$ arrivals in a time interval of width $\Delta T$, then $\hat{\lambda}^{-1} = n/\Delta T$. These are the formulas used in Section \ref{myCode}. Also notice that we may replace a sum, $D = \sum_i d_i$, with a product of the mean, $D = n\overline{d}$. Since the CryptoNote reference code considers the middle $600$ blocks after slicing outliers, we may write
|
|
\begin{align*}
|
|
\hat{d}_{721} =&\frac{D/\lambda + \Delta T - 1}{\Delta T}\\
|
|
=& \frac{\hat{\lambda}}{\lambda} \overline{d} + \frac{\Delta T - 1}{\Delta T}
|
|
\end{align*}
|
|
|
|
|
|
In the CryptoNote reference code, we have a block arrival target of $60.0 s$ per block, so to observe $600$ blocks should take, on average, $36000 s$, so $\frac{\Delta T - 1}{\Delta T} \approx 0.99997$ and this formula may be written as, approximately,
|
|
\begin{align}
|
|
\hat{d}_{721} =& \frac{\hat{\lambda}}{\lambda} D + 1 \label{diffFormApprox}
|
|
\end{align}
|
|
We provide further discussion of this in Section \ref{clockproblems}.
|
|
|
|
\subsection{Criticisms of the CryptoNote Reference Code: Sorting Timestamps Alone}\label{reorderingproblems}
|
|
|
|
To see the gravity of the mistake made in re-ordering timestamps in isolation from the difficulties, consider throwing out the outlying measurements incorrectly from a more usual example. Let's say we are measuring the height and weight of everyone in Huntington-Ashland, West Virginia, the fattest city in America. We line everyone up and give them a number according to their position in line. This is their \emph{index}. We then measure their heights and their weights, proceeding from index $1$, the first person in line, down to \emph{index} $P$, the last person in line, where $P$ is the population of the town. These measurements yield the list of data $(H_1, W_1)$, $(H_2, W_2)$, and so on up to $(H_P, W_P)$. Of course, $H_i$ and $W_i$ correspond to the height and weight, respectively, of the $i^{th}$ person in line. Furthermore, let us presume that we wish to discard the top $1/12$ of the list and the bottom $1/12$ of the list, leaving $5/6$ of the population in the list.
|
|
|
|
However, before analyzing weight by removing outliers, presume we follow a similar procedure as the CryptoNote reference code. We order heights separately from weights:
|
|
\begin{align*}
|
|
\hat{H}_1 =& \min\left\{H_1, H_2, \ldots, H_P\right\}\\
|
|
\hat{H}_2 =& \min\left\{H_1, H_2, \ldots, H_P \mid H_i \neq \hat{H}_1\right\}\\
|
|
\hat{H}_3 =& \min\left\{H_1, H_2, \ldots, H_P \mid H_i \neq \hat{H}_1, \hat{H}_2\right\}\\
|
|
\vdots
|
|
\end{align*}
|
|
Now, $\hat{H}_1$ corresponds to the height of the shortest person in town, but $W_1$ still corresponds to the weight of the first person in line. Similarly, $\hat{H}_P$ corresponds to the height of the tallest person in town, but $W_P$ still corresponds to the weight of the last person in line. The approach presented by the CryptoNote reference code would then exclude all pairs $(\hat{H}_i, W_i)$ if the index $i$ is in the first $P/12$ or in the last $P/12$ indices.
|
|
|
|
What data do we have remaining if we do this? We have $\hat{H}_{P/12 + 1}$, $\hat{H}_{P/12 + 2}$, $\ldots$, $\hat{H}_{5P/6}$, which correspond to the heights of the middle quantile. However, we also have $W_{P/12 + 1}, \ldots, W_{5P/6}$, which corresponds to the weights of the people \emph{who were in the middle of the line}. While our initial goal was to discard outliers based on their height and weight measurements, what we ended up doing was discarding outliers based on their position in line. While this may have yielded interesting statistical information about how rapidly overweight people make it into a queue, it is not relevant to relating height with weight.
|
|
|
|
With this simple description, we have demonstrated that the methods used in the CryptoNote reference code for discarding outliers are, at best, inappropriately applied.
|
|
|
|
\subsection{Criticisms of the CryptoNote Reference Code: Lack of Equilibrium Solutions with Constant Hash Rate}\label{clockproblems}
|
|
|
|
Our intuitive notion of difficulty says we should expect difficulty to go up when hash rate has gone up, and for difficulty to go down when hash rate has gone down. If hash rate has not changed, then difficulty should not change. Intuitively, the CryptoNote difficulty assessment formula, Equation \ref{diffFormInitial}, does a satisfactory job. Indeed, when hash rate doubles, the sample rate $\hat{\lambda}$ doubles, and so the difficulty of the next block will be approximately double the average of the sample blocks.
|
|
|
|
However, there is a critical problem with the formula, which has to do with equilibrium solutions to the difficulty equation. Again, our intuitive notion is that, if hash rate remains unchanged, then difficulty should remain unchanged, and vice versa. Of course, we may not directly measure hash rate, but we can directly measure rate of block arrivals. So what happens if blocks have been arriving on target? This would be the best possible indication that hash rate has remained unchanged.
|
|
|
|
If blocks have been arriving on target, then $\hat{\lambda} \approx \lambda$ and so difficulty (approximately) goes up by one according to Equation \ref{diffFormInitial}. On the other hand, if difficulty is being held (approximately) constant at equilibrium, then $\hat{d}_{721} \approx \overline{d}$ and
|
|
\begin{align*}
|
|
\hat{d}_{721} =& \frac{\hat{\lambda}}{\lambda} \overline{d} + 1\\
|
|
\overline{d} \approx& \frac{\hat{\lambda}}{\lambda} \overline{d} + 1\\
|
|
(1 - \frac{\hat{\lambda}}{\lambda})\overline{d} \approx& 1
|
|
\end{align*}
|
|
But then we must conclude that our sample rate, $\hat{\lambda}$, must be quite different from our target rate, $\lambda$, otherwise we would conclude $0 = 1$, which is absurd. Indeed, difficulty may never be held constant when blocks are arriving on target given Equation \ref{diffFormInitial}.
|
|
|
|
This analysis of an equilibrium solution suggests that we have verified that $\hat{\lambda} = \lambda$ does not directly correspond with a static difficulty in the CryptoNote reference code, violating our intuition about difficulty.
|
|
|
|
\section{Comparing Difficulty Adjustment Performance}\label{comp}
|
|
|
|
In Appendix \ref{mySimulationCode}, we present Python code that takes as input a piecewise constant hashrate function and produces a stochastically generated blockchain in the form of a sequence of timestamp-difficulty ordered pairs. In this section, we use this code to compare the CryptoNote reference code difficulty equation (see Section \ref{refCode}) with the proposed difficulty equation (see Section \ref{myCode}). Recalling our interest in comparing the robustness of these equations against sudden changes in hash rate, and recalling that $\lambda = H/d$ in the model from Section \ref{modelDef}, we investigate several hash rate scenarios with piecewise constant hash rate functions. The overall goal of the difficulty equations is to keep observed block arrival rates, $\hat{\lambda}$, close to target block arrival rates, $\lambda^{*}$. Hence, this provides our metric for goodness of a difficulty equation: we compute the relative error between sample block arrival rate and target block arrival rate for each of our difficulty adjustment equations and for each hash rate function under investigation.
|
|
|
|
We look into four hash rate scenarios. In the first hash rate scenario, we investigate a piecewise constant hash function of the form $H(t) = \alpha^{\lfloor \beta t \rfloor}$ where $\lfloor x \rfloor$ denotes the usual floor function, i.e.\ $\lfloor x \rfloor = \max\left\{i \in \mathbb{Z} \mid i \leq x\right\}$, and where we take $\alpha > 1$ controls overall growth rate and $1/\beta > 0$ controls the period of time between hash rate jumps. This would be an exponential growth model. In the second hash rate scenario, we investigate a logistic growth model with $H(t) = K(1 + Ae^{-k\lfloor \beta t\rfloor })^{-1}$ where $K$ is the carrying capacity of the network, $k$ is the overall population growth rate, the constant $A$ satisfies $A = (K-H(0))/H(0)$, and where $1/\beta > 0$ represents the period of time between hash rate jumps. In the third hash rate scenario, we investigate a hash rate that starts small, jumps to a very large value for a sufficiently long enough period of time for difficulty to come to equilibrium, and then drops back to a very small value for the remainder of the simulation (i.e.\ a tophat function). This way, we can investigate two population-based hash rate models, and we can also investigate the scenario presented in Section \ref{hashrateattack}. In the final hash rate scenario, we investigate a square wave function to investigate what sort of effect periodic behavior in mining has upon network hash rate.
|
|
|
|
\subsection{Exponential Growth in Hash Rate}\label{expComp}
|
|
|
|
We find that, for large hash rates, the CryptoNote reference code tracks hash rate very well. However, the term $(\Delta T - 1)/\Delta T$ leads to a roughly linear increase in hash rate over time when hash rate is small or constant. This is true regardless of the choice of $\beta$, which controls how long hash rate is held constant before changing. In all investigations of exponential growth in hash rate, we find that the difficulty equation presented in Section \ref{myCode} does a better job of tracking true network hash rate compared to the CryptoNote reference code difficulty equation presented in Section \ref{refCode}. Not only is the relative error between observed block arrival rate and target block arrival rate generally smaller when we use our difficulty equation instead of the CryptoNote reference code, our difficulty equation appears to approach true network hash rate at a faster rate than the CryptoNote reference code.
|
|
|
|
We numerically validate these claims by investigating two hash rate functions, $H_1(t) = 1.001^{\lfloor t/600 \rfloor}$, representing a $0.1\%$ increase in hash rate every ten minutes, and $H_2(t) = 1.005^{\lfloor t/120 \rfloor}$, representing a $0.5\%$ increase in hash rate every two minutes. Using a given hash rate function, we generate one blockchain using our new difficulty adjustment equation described in Section \ref{myCode}:
|
|
\[\underline{\mathcal{B}}_{new} = (\mathcal{B}_{new,0}, \mathcal{B}_{new,1}, \mathcal{B}_{new,2}, \ldots, \mathcal{B}_{new,n_{new}-1})\]
|
|
and we generate another blockchain using the CryptoNote reference code difficulty adjustment equation described in Section \ref{refCode}:
|
|
\[\underline{\mathcal{B}}_{old} = (\mathcal{B}_{old,0}, \mathcal{B}_{old,1}, \mathcal{B}_{old,2}, \ldots, \mathcal{B}_{old,n_{old}-1})\]
|
|
From these blockchain objects, we may compute the sample block arrival rates, $\hat{\lambda}$, at each new timestamp, and we may compare the relative error between $\hat{\lambda}$ and the target block arrival rate, $\lambda^{*}$ as a function of the latest timestamp (which may not be the top timestamp).
|
|
|
|
\subsection{Logistic Growth in Hash Rate}\label{logComp}
|
|
|
|
We again find that, for large hash rates, the CryptoNote reference code does a decent job. However, when hash rate is small (i.e.\ the beginning of a logistic population) or held constant (i.e.\ the end of a logistic population), the linear growth in the CryptoNote reference difficulty equation becomes steadily more apparent as time goes on; the relative error increases regularly over time. The effect of this linear term on the upswing portion of the logistic curve is difficult to detect, but is still present. Not only is the relative error between observed block arrival rate and target block arrival rate generally smaller when we use our difficulty equation instead of the CryptoNote reference code, our difficulty equation appears to approach true network hash rate at a faster rate than the CryptoNote reference code.
|
|
|
|
We numerically validate these claims by investigating two hash rate functions. We first investigate $H_1(t) = 10^\alpha/(1+A^{-k \lfloor \beta t \rfloor}$, where $A = (10^{\alpha} - 10^{\gamma})/10^{\gamma}$, representing a logistic growth model with carrying capacity $10^\alpha$, initial hash rate $10^{\gamma}$, growth rate $k$, and $1/\beta$ controls how often hash rate jumps. In particular, with a network carrying capacity of $1.0$ petahash per second ($\alpha = 15$) and an initial network hash rate of $1.0$ gigahash per second ($\gamma = 9$), with $\beta = 1/6 h^{-1}$ so that hash rate changes every six hours, and with a maximum growth rate of $k=1.0$ megahash per second per day. We also investigate $H_2(t)$ of the same form with a terahash per second carrying capacity ($\alpha =12$), an initial hash rate of $1.0$ gigahash per second ($\gamma = 9$), and with $\beta = 1/6 h^{-1}$ so that hash rate changes every six hours, and with a maximum growht rate of $10^7$ hashes per second, a ten fold increase over $H_1(t)$.
|
|
|
|
Using these given hash rate functions, we generate one blockchain using our new difficulty adjustment equation described in Section \ref{myCode}:
|
|
\[\underline{\mathcal{B}}_{new} = (\mathcal{B}_{new,0}, \mathcal{B}_{new,1}, \mathcal{B}_{new,2}, \ldots, \mathcal{B}_{new,n_{new}-1})\]
|
|
and we generate another blockchain using the CryptoNote reference code difficulty adjustment equation described in Section \ref{refCode}:
|
|
\[\underline{\mathcal{B}}_{old} = (\mathcal{B}_{old,0}, \mathcal{B}_{old,1}, \mathcal{B}_{old,2}, \ldots, \mathcal{B}_{old,n_{old}-1})\]
|
|
From these blockchain objects, we may compute the sample block arrival rates, $\hat{\lambda}$, at each new timestamp, and we may compare the relative error between $\hat{\lambda}$ and the target block arrival rate, $\lambda^{*}$ as a function of the latest timestamp (which may not be the top timestamp).
|
|
|
|
\subsection{Big Swings in Hash Rate}\label{topHatComp}
|
|
|
|
This is the fun, true test of the difficulty equations. We set a small constant $h_0 > 0$ and a large constant $h_1 > 0$. We choose a time interval $[t_1, t_2)$ on which network hash rate will blast up from $H(t) = h_0$ hashes per second to $H(t) = h_1$ hashes per second, and then back down to $H(t) = h_0$ again.
|
|
\[H(t) = \begin{cases} h_0 & t \in [0,t_1)\\
|
|
h_1 & t \in [t_1, t_2)\\
|
|
h_0 & t > t_2\end{cases}\]
|
|
|
|
Using this function, we can assess the rate at which sample block arrival rate approaches target block arrival rate after a large jump in network hash rate, and we can also assess how gracefully a difficulty equation can respond to the converse situation when hash rate drops. We investigate only values of $h_1, h_0$ such that $h_0/h_1 > 1/100$. This would correspond to a user who controls $99\%$ of the network turning off their equipment, and will lead to block arrival rates dropping by two orders of magnitude. Investigating smaller values of $h_0/h_1$ is unnecessary, for the consequence would be a large stall and no more blocks.
|
|
|
|
We again see that our difficulty adjustment equation almost always has a smaller relative error between sample and target block arrival rates, and that these values approach each other more rapidly with our difficulty adjustment equation than with the CryptoNote reference code difficulty equation.
|
|
|
|
\subsubsection{Square Wave Hash Rate}\label{squareWaveComp}
|
|
|
|
If hash rate behaves in a square wave, we may imagine this as applying a periodic forcing function to a difference equation. We investigate this scenario for several reasons. One reason is the possibility that resonance could cause a blowup in network hash rate. Indeed, if a difference equation has a natural frequency, $\omega$, and a periodic forcing function is applied with period $P$ that is an integer multiple of $\omega$, we run the risk of resonance. Another reason to investigate periodic hash rates is because the CryptoNote reference code uses a dubious method of discarding outliers. Indeed, in the CryptoNote reference code, the top $720$ blocks are studied and the top and bottom $60$ blocks from that list are discarded. Hence, a miner could hop on the network, mine for $60$ blocks, and then hop off the network; such a miner would never personally experience an increase in difficulty as a response to their activity. This suggests some time delay properties between hash rates and difficulties which may be revealed with a periodic forcing function through the expressed phase angle.
|
|
|
|
To these ends, we investigate periodic square wave functions. We choose $h_1 > h_0 > 0$ and a period, $P$. We choose some timepoint $0 \leq t^{*} < P$ to represent the time in which network activity is ``low.'' We then use a periodic extension of the function
|
|
\[H_0(t) = \begin{cases} h_0 & t \in [0, t^*)\\
|
|
h_1 & t \in [t^*, P)\end{cases}\]
|
|
with period $P$ as our hash rate function. We investigate ranges of $1 > h_0/h_1 > 1/100$ as in the previous case, and we investigate several values for the period $P$ ranging from $P=1.0 s$ to $P=2m/\lambda^{*}$, which represents the amount of time it takes to receive $m$ blocks, where $m > \ell$ is the sample size used to estimate block arrival rates. If the frequency of the periodic hash rate is an integer multiple of difficulty adjustment period, or outlier discarding periods, or even block arrival rate sample size, this window should be wide enough to detect an effect.
|
|
|
|
\subsection{Results}
|
|
In response to exponential or logarithmic hash rate growth, the new difficulty adjustment equation performs uniformly better than the CryptoNote reference code; not only is relative error between sample block arrival rate and target block arrival rate usually smaller, it decays to zero more rapidly with the new difficulty equation. The CryptoNote reference code performs worse in the case of logistic growth, but sample block arrival rate is still usually kept to within a tolerance of BLAH BLAH INSERT NUMBER HERE compared to target block arrival rate.
|
|
|
|
In response to the top hat function, the two difficulty equations perform nearly as well as one another, although the new difficulty equation performed moderately better in terms of responding to a large decrease in hash rate. In response to the periodic function, we saw some interesting behaviors.
|
|
|
|
The most surprising result is that the CryptoNote reference code, which has only a weak resemblance to our difficulty equation (which was derived directly from a blockchain growth model), it does a fair to good job of keeping arrival rates roughly constant in time. The primary concern about this algorithm, it turns out, is not the response to general hash rate trends. In fact, it is a far bigger problem that users may exploit the way that outliers are discarded in order to mine without seeing difficulty change due to their own activity. Nevertheless, our difficulty equation is still uniformly better than the reference code.
|
|
|
|
\section{Further Questions}\label{furtherQuestions}
|
|
|
|
In this section, we discuss several routes by which the above work could be expanded by interested and motivated readers. In general, it is to our benefit to produce many models and assess their precision and accuracy, and to compare and contrast multiple hypotheses to determine which model does the best job of representing the ground truth. In practice, exploring every possibility is unreasonable. We hope that others extend this work to develop yet better difficulty adjustment methods.
|
|
|
|
\subsection{Time Series Models of Hash Rate}\label{timeSeries}
|
|
In deriving our difficulty equation, we presumed hash rate of the future will match the average hash rate of the past. The holy grail of any adaptive algorithm is prediction. Time series analysis tools may be used to make statistical predictions of instantaneous hash rate estimate $\hat{H}_{n}$ based on previous observations $\hat{H}_{n-1}, \hat{H}_{n-2}, \ldots$. An ambitious user could seek a transformation of this time series that is covariance-stationary and use a SARIMA model, for example, to try to predict the next hash rate.
|
|
|
|
In this document, although we are tempted to use heavy statistical machinery to solve the problem, parsimony suggests that we seek a simpler solution before we seek a more complicated solution. Models such as SARIMA should only be used if ARIMA models are not doing particularly well, which should only be used if ARMA models are not doing very well, which should only be used if autoregressive (AR) or moving average (MA) models are insufficient. Furthermore, our difficulty adjustment algorithm will be unsupervised, whereas time series analysis problems usually require a modeler to actively tweak and play with data; time series analysis often does not perform well when fully automated and runs the risk of over-fitting. In our case, we decided upon using simple moving averages, but an efficient implementation of a fully automated SARIMA model that punishes overfitting would be a very interesting application of statistical analysis on a network.
|
|
|
|
\subsection{Population Growth Models of Hash Rate}\label{popGrowth}
|
|
|
|
In this document, we generated hash rate functions in a deterministic way; we picked formulae for the hash rate functions directly. We did this in order to see how difficulty equations responded to specific scenarios. However, we could generate hash rate functions stochastically using population models and various stochastic implementations of those models.
|
|
|
|
For example, we may investigate the logistic growth, which obeys the differential equation $H^{\prime} = kH(1-H/H_{max})$ where $k$ is maximum growth rate and $H_{max}$ is the carrying capacity of the network. The solution to this equation takes the form $H(t) = K/(1+Ae^{-kt})$ where $A = (K-H(0))/H(0)$, which is the form used in Section $\ref{logComp}$. However, rather than determining hash rate determinsitically from this function by plugging in the current time, we could implement a Gillespie algorithm from \cite{gillespie1977exact} to create a stochastic hash rate function whose expectation is the logistic solution above. This would allow a researcher to investigate how different assumptions about hash rate growth over time may influence blockchain growth.
|
|
|
|
One may ask, ``why does the author recommend a complication of the hash rate function here, but in Section \ref{timeSeries} wishes to argue for parsimony?'' The answer to this is that in Section \ref{timeSeries}, we are discussing the way a user on the network estimates hash rate. This is the mechanism we are designing, a difficulty equation, which necessarily needs to be simple for computaitonal reasons. On the other hand, here, in Section \ref{popGrowth}, we are discussing which hash rate function we should use as input to our blockchain when testing a difficulty equation for robustness and accuracy. We are asking ``how should I test our designed mechanism?'' Testing our difficulty adjustment equation against a complicated hash rate is a stress test on the mechanism we have designed.
|
|
|
|
\subsection{Graph Theoretic Modeling and Parent Coin Selection Rules}
|
|
One other obvious route of extension is to model the cryptocurrency network itself and investigate the relationships between difficulty adjustment, parent coin selection rules, and network structure. One can representing a computer network with a weighted graph, where the weight between nodes represents a propagation delay. The choice of graph is important, for a five-node network with $1 s$ separation between each pair of connected nodes, can result in a maximum five second propagation delay (if the nodes are lined up serially), or could result in a maximum one second propagation delay (if the nodes are arranged as a pentagram in a complete graph with five nodes, $K_5$). Any graph choice must be justified; equating the average and standard deviation in propagation delay may be sufficient for some statistical purposes, but by no means capture the details of a complicated network.
|
|
|
|
The presence of propagation delays will necessarily lead to competing chains, which necessitates a choice of parent coin selection rules, such as the ``longest chain'' rule first proposed by Nakamoto in \cite{nakamoto2008bitcoin}, and the so-called GHOST rule, first proposed by Sompolinsky in \cite{sompolinsky2013accelerating}. The GHOST rule selects the parent coin by seeking the heaviest observed sub-tree at each fork in the blocktree, and may exhibit certain advantages over the Nakamoto rule. It is possible that the choice of graph structure, parent coin selection rule, and difficulty adjustment formula are deeply inter-related. Given a particular graph structure and difficulty adjustment formula, a cryptocurrency network may prefer the GHOST rule over the Nakamoto rule, or vice versa. Given a particular graph structure and a particular parent coin selection rule, some difficulty adjustment formulae may make no sense, and some may perform very well. Finally, given a particular parent coin selection rule and difficulty adjustment formula, there may be a graph structure for which these choices perform optimally. These questions could lead to very efficient creations.
|
|
|
|
\subsection{Stalling in the Event of Sudden Hash Rate Drop}
|
|
|
|
The problem with stalling cryptocurrency networks in the event of a large hashrate drop can be mitigated in at least one of two ways. The reason the network may stall is because the current estimate of difficulty remains unchanged until a block is added. The usual maximum likelihood estimate (MLE) of a Poisson rate doesn't use all the data we have available. The first way we recommend mitigating a stalling event involves exploiting the fact that we know how long it has been since the last block arrival. This is additional information we have available that is not yet being used. The second way we recommend mitigating a stalling event involves contact with a third party to obtain information about their own, distinct blockchain, whcih is also additional information we have available. Either or both of these avenues could present interesting lines of inquiry.
|
|
|
|
Our first avenue is to incorporate the right-censored maximum likelihood estimate of a Poisson process rate. Indeed, in our derivation in Section \ref{myCode}, we assume network hash rate remains static and match our current estimate until more information arrives in the form of the next block to be added. However, at any given time, we actually have more information avaiable; that is to say, we know that some block \emph{will be arriving} at some point \emph{after the current time}. Assume that all users are reporting their timestamps honestly. Rather than declaring a blockchain to be a sequence of timestamp-difficulty ordered pairs together with a future difficulty of the form
|
|
\[(t_0, d_0), (t_1, d_1), \ldots, (t_{n-1}, d_{n-1}), d_n\]
|
|
and with no \emph{a priori} concept of time attached to the blockchain object (outside of it's timestamps), we could alternatively define a blockchain in the following way. Presuming some height $n$, we may define a blockchain to consist of a sequence of timestamp-difficulty ordered pairs together with a time, $t$, and a random variable, $\tau$, of the form
|
|
\[(t_0, d_0), (t_1, d_1), \ldots, (t_{n-1}, d_{n-1}), (\tau, d_n)\]
|
|
where the only information we have about $\tau$ is the distributional fact that $\tau > t$ with probability $1$. If we wish to incorporate data about timestamp manipulation, this may be stored in any distributional assumptions we place on $\tau$.
|
|
|
|
The way we could implement this is to take our list of known timestamps, $t_0, t_1, \ldots, t_{n-1}$, include the current time, $t^*$, and sort the results into their ordered statistics, $t_{(0)} < t_{(1)} < \ldots < t_{(n)}$. We then simply use the MLE as usual from here. If all timestamps are being reported honestly on a network with no propagation delays, then we automatically have the ordering $t_0 < t_1 < t_2 < \cdots < t_{n-1} < t^*$. However, since timestamps may occur out of order, it is possible that $t_{(n-i)} < t^{*} < t_{(n)}$ for some $i=1, 2, \ldots$.
|
|
|
|
One problem with this approach is that now every user may use their own time, $t^*$, to determine current difficulty. As a consequence, two users with the same copy of the blockchain may compute different values for ``next difficulty'' and there is no way to verify which, if any, are correct. Users with the same data will only compute the same difficulty if they also have synchronized clocks. To worsen matters, it seems this approach leaves open the possibility of a vulnerability. It will be in the users' best interest to obtain a smaller estimate of block arrival rates, if possible, in order to obtain a small estimate of hash rate, and therefore a smaller difficulty. Hence, using the right-censored maximum likelihood approach, each user would set their local clock very far ahead in the future to take advantage of low-difficulty mining. This would cause an upward drift in timestamps compared to the true time. This upward drift may be bounded and slow, in which case the risk of blocks being rejected by the rest of the network will prevent the behavior from getting out of control. On the other hand, the upward drift may be very fast or unbounded. Timestamps on the top of the blockchain will eventually be so far out of whack with reality that no honest miners will get their blocks accepted by relaying nodes.
|
|
|
|
Our second avenue is to tie the estimate of block arrival rate on the network with the block arrival rates on other networks. For example, if a user has observed $n$ Bitcoin blocks validated since the last Monero block has been validated, she may reasonably presume that either (a) she has a connectivity issue with the Monero network, (b) the Monero network hash rate has suddenly dropped by a factor of $10n$, or (c) the Bitcoin network hash rate has suddenly increased by a factor of $10n$. Due to the size of the Bitcoin network, option (c) is almost certainly false in general. This seems like an elegant solution, but any time we use a third party, verifiability and security become issues. Working out the details of either of these proposals could lead to an enormous improvement to the resilience of any cryptocurrency network.
|
|
|
|
|
|
\subsection{Damping}
|
|
|
|
We may wish to damp the amount of change we apply to our difficulty in the case that we believe that timestamps have recently been manipulated. To this end, recall the convenient property of Poisson processes with rate $\lambda$: the mean inter-arrival time is $1/\lambda$ and the variance in the inter-arrival times is $1/\lambda^2$. Hence, if mean inter-arrival time and standard deviation of inter-arrival times are very different, this suggests that timestamps have been manipulated and the process underlying the timestamp creation is not a homogeneous Poisson process.
|
|
|
|
Hence, testing the squared mean against the variance is a test for whether a process is Poisson or not. Given a sequence of inter-arrival times, say $\left\{S_1, S_2, \ldots, S_n\right\}$, a test of whether these inter-arrival times came from a genuine Poisson process is to compare the squared sample mean $\overline{S}^2 = \left(n^{-1}\sum_{i=1}^{n} S_i\right)^2$ with the (unbiased) sample variance $\text{Var}(S) = (n-1)^{-1}\sum_{i=1}^{n} (S_i - \overline{S})^2$. Indeed, taking the whole Monero blockchain as a sample, we observe about a $22\%$ difference between mean and the standard deviation! This suggests that, throughout the history of Monero, quite a bit of timestamp manipulation has occurred, although at this point it is not clear to the Monero Research Lab how to quantify the amount or degree of manipulation.
|
|
|
|
When the squared mean and variance are vastly different, we distrust the notion that the underlying process is Poisson. Furthermore, if this lack of Poisson-icity is due to an attacker manipulating timestamps, then the mean inter-arrival time, which only utilizes $T_{(1)}$ and $T_{(N)}$, the first and last times of arrival, is particularly vulnerable to a \textit{single} attacker changing their timestamp. On the other hand, the sample standard deviation utilizes all inter-arrival times equally in it's computation. Hence, rather than using $\hat{\lambda} = 1/\overline{S}$, another approach is to use $\hat{\lambda} = 1/\sqrt{\text{Var}(S)}$.
|
|
|
|
Hence, rather than computing $d_{n+1}$ by a multiplicative factor $\overline{H}_n/\overline{H}_{n-1}$, consider an equivalent additive change in difficulty, $d_{n+1} - d_n = (\frac{\overline{H}_n - \overline{H}_{n-1}}{\overline{H}_{n-1}})d_n$. The idea of this approach is to scale this change in difficulty by a factor $\alpha$ which will depend on both the squared sample mean and the sample variance of inter-arrival times. If squared sample mean and sample variance are approximately equal, we will set $\alpha \approx 1$, and if sample meean and sample standard deviation are different, $\alpha \to 0$. To this end, let $\overline{S}^2$ denote the squared sample mean of inter-arrival times and denote $\text{Var}(S)$ denote the sample standard deviation. Many choices of $\alpha$ are reasonable. For example, we may choose
|
|
\[\alpha = \text{Exp}\left[-\left|\overline{S}^2-\text{Var}(S)\right|/\overline{S}^2\right]\]
|
|
Notice that, since $\overline{S}^2$ is never zero, since all blocks must arrive with at least one second separating them, this expression is well defined. Also notice that $\alpha$ has the property that when $\overline{S}^2 = \text{Var}(S)$, we have that $\alpha = 1$, and whenever $\left|\overline{S}^2 - \text{Var}(S)\right|/\overline{S}^2 \to \infty$, we have that $\alpha \to 0$. The scaling in the denominator of the exponent provides a probabilistic guarantee that the momentum term $\alpha$ stays reasonable close to $1$ under manipulation-free circumstances.
|
|
|
|
We can consider $\alpha$ a momentum term, where a close match between a true Poisson process and the observed inter-arrival times yields almost no momentum, or we can consider $\alpha$ a trust value, where a close match yields a high trust level. Using this interpretation, a $22\%$ variance between squared sample mean and sample variance would yield an $80.25\%$ trust rating. On the other hand, a mere five percent difference between squared sample mean, $\overline{S}^2$, and sample variance, $\text{Var}(S)$, will provide a trust value of $0.95122$, or rather, about a $95\%$ trust rating. An unreasonably large difference between squared sample mean and sample variance, say $50\%$, would yield a trust value of $0.5 \leq -\ln(\alpha)$, or rather $\alpha \geq 0.606$ or a $60.6\%$ trust rating.
|
|
|
|
As we have previously observed, there is, historically, a $22\%$ difference between squared sample mean, $\overline{S}^2$, and sample variance, $\text{Var}(S)$. This provides a good reference point for examining this momentum term. A $22\%$ difference would provide $-\ln(\alpha) \leq 0.22$, yielding $0.8025 \leq \alpha$. That is to say, our momentum term would cause our difficulty to adjust at around $80\%$ it's maximal rate.
|
|
|
|
%%%%%%%%%%%%%%%%%%%%%%%%
|
|
|
|
|
|
\newpage
|
|
\begin{appendices}
|
|
\chapter{CryptoNote Difficulty Reference Code}\label{referenceCode}
|
|
|
|
\begin{lstlisting}[language=Python,basicstyle=\small,breaklines=true]
|
|
# DIFFICULTY.py
|
|
# Gives difficulty information over time for blocks with given timestamps
|
|
# Input: block_file WINDOW CUT LAG CHECK_WINDOW [MODE]
|
|
# block_file format: one line per integer timestamp
|
|
# WINDOW: blocks to be used when computing difficulty (720 in practice)
|
|
# CUT: blocks on each side of the block window to exclude (60 in practice)
|
|
# LAG: how far behind we want to be (15 in practice)
|
|
# CHECK_WINDOW: number of blocks to use for median cutoff (60 in practice)
|
|
# [MODE]: 0 = don't sort difficulties (default); 1 = sort difficulties
|
|
# Output: block information, one line per block
|
|
# line format: block_id timestamp next_difficulty cumulative_difficulty
|
|
|
|
import sys
|
|
from math import floor
|
|
|
|
window = int( sys.argv[2] )
|
|
cut = int( sys.argv[3] )
|
|
lag = int( sys.argv[4] )
|
|
check_window = int( sys.argv[5] )
|
|
target = 60
|
|
mode = 0
|
|
try:
|
|
if int( sys.argv[6] ) == 1:
|
|
mode = 1
|
|
except:
|
|
pass
|
|
|
|
print "# window is " + str( window ) + " and cut is " + str( cut )
|
|
|
|
# Read timestamps into an integer array
|
|
block_file = open( sys.argv[1], 'r' )
|
|
timestamps = []
|
|
for line in block_file:
|
|
timestamps.append( floor( float( line.strip() ) ) )
|
|
cumulative_difficulties = []
|
|
|
|
print "# read " + str( len( timestamps ) ) + " blocks"
|
|
|
|
# Apply the median rule
|
|
for i in range( check_window, len( timestamps ) ):
|
|
try:
|
|
median = sum( timestamps[i-check_window:i] )
|
|
if timestamp[i] < median:
|
|
timestamp.pop( i )
|
|
except:
|
|
pass
|
|
|
|
# Compute the difficulty for the next block
|
|
def next_difficulty( timestamps, cumulative_difficulties ):
|
|
if ( len( timestamps ) > window ):
|
|
timestamps = timestamps[0:window]
|
|
cumulative_difficulties = cumulative_difficulties[0:window]
|
|
|
|
length = len( timestamps )
|
|
|
|
# Run some sanity checks
|
|
if len( timestamps ) > window or len( cumulative_difficulties ) > window or len( timestamps ) != len( cumulative_difficulties ):
|
|
raise Exception( "Incorrect number of blocks" )
|
|
if length <= 1:
|
|
return 1
|
|
if window < 2:
|
|
raise Exception( "Window is too small" )
|
|
if ( 2 * cut > window - 2 ):
|
|
raise Exception( "Cut is too large" )
|
|
|
|
timestamps.sort()
|
|
if mode == 1:
|
|
cumulative_difficulties.sort()
|
|
|
|
# Compute the cut indices
|
|
if ( length <= ( window - 2 * cut ) ):
|
|
cut_begin = 0
|
|
cut_end = length
|
|
else:
|
|
cut_begin = int( floor( ( len( timestamps ) - ( window - 2 * cut ) + 1 ) / 2 ) )
|
|
cut_end = cut_begin + window - 2 * cut
|
|
|
|
time_span = timestamps[cut_end-1] - timestamps[cut_begin]
|
|
if time_span == 0:
|
|
time_span = 1
|
|
|
|
total_work = cumulative_difficulties[cut_end-1] - cumulative_difficulties[cut_begin]
|
|
if total_work < 0:
|
|
raise Exception( "Cannot have negative total work" )
|
|
|
|
# Assume high is zero; that is, no overflow
|
|
low = total_work * target
|
|
if low + time_span - 1 < low:
|
|
return 0
|
|
else:
|
|
return int( floor( ( low + time_span - 1 ) / time_span ) )
|
|
|
|
# Start feeding blocks into the difficulty algorithm
|
|
print "# block timestep next_difficulty cumulative_difficulty"
|
|
for i in range( len( timestamps ) ):
|
|
offset = ( i + 1 ) - min( i + 1, window + cut )
|
|
if offset == 0:
|
|
offset = 1
|
|
|
|
difficulty = next_difficulty( timestamps[offset:i], cumulative_difficulties[offset:i] )
|
|
if i == 0:
|
|
cumulative_difficulties.append( 0 )
|
|
else:
|
|
cumulative_difficulties.append( cumulative_difficulties[i-1] + difficulty )
|
|
|
|
# Output
|
|
print ' '.join( map( str, [ i, timestamps[i], difficulty, cumulative_difficulties[i] ] ) )
|
|
\end{lstlisting}
|
|
|
|
|
|
\chapter{New Difficulty Reference Code}\label{referenceCode}
|
|
|
|
\begin{lstlisting}[language=Python,basicstyle=\small,breaklines=true]
|
|
|
|
\end{lstlisting}
|
|
|
|
\end{appendices}
|
|
|
|
\medskip{}
|
|
|
|
\bibliographystyle{plain}
|
|
\bibliography{biblio.bib}
|
|
|
|
\end{document}
|