<?xml version="1.0" encoding="utf-8"?>
<?xml-model href="rfc7991bis.rnc"?>  <!-- Required for schema
      validation and schema-aware editing --> 

<!DOCTYPE rfc [
  <!ENTITY filename "draft-eastlake-fnv-20">
  <!ENTITY nbsp     "&#160;">
  <!ENTITY zwsp     "&#8203;">
  <!ENTITY nbhy     "&#8209;">
  <!ENTITY wj       "&#8288;">
]>
<!-- <?xml-stylesheet type="text/xsl" href="rfc2629.xslt" ?> --> 
<!-- This third-party XSLT can be enabled for direct transformations
in XML processors, including most browsers --> 
<!-- If further character entities are required then they should be
added to the DOCTYPE above. Use of an external entity file is not
recommended. --> 
<?rfc strict="yes" ?>
<!-- give errors regarding ID-nits and DTD validation -->
<!-- control the table of contents (ToC) -->
<?rfc toc="yes"?>

<rfc
  xmlns:xi="http://www.w3.org/2001/XInclude"
  category="info"
  docName="&filename;"
  ipr="trust200902"
  obsoletes=""
  submissionType="IETF"
  xml:lang="en"
  version="3">
<!-- 
    * docName should be the name of your draft * category should be
    one of std, bcp, info, exp, historic * ipr should be one of
    trust200902, noModificationTrust200902, noDerivativesTrust200902,
    pre5378Trust200902 * updates can be an RFC number as NNNN *
    obsoletes can be an RFC number as NNNN
-->


<!-- ____________________FRONT_MATTER____________________ -->
<front>
   <title abbrev="FNV">The FNV Non-Cryptographic Hash
   Algorithm</title>
   <!--  The abbreviated title is required if the full title is
        longer than 39 characters --> 

   <seriesInfo name="Internet-Draft"
               value="&filename;"/>

   <author fullname="Glenn S. Fowler" initials="G."
           surname="Fowler">
     <organization>Google</organization>
     <address>
       <email>glenn.s.fowler@gmail.com</email>
     </address>
   </author>

   <author fullname="Landon Curt Noll" initials="L."
           surname="Noll">
     <organization>Cisco Systems</organization>
     <address>
       <postal>
         <street>170 West Tasman Drive</street>
         <city>San Jose</city>
         <region>California</region>
         <code>95134</code>
         <country>USA</country>
       </postal>        
       <phone>+1-408-424-1102</phone>
       <email>fnv-ietf5-mail@asthe.com</email>
       <uri>http://www.isthe.com/chongo/index.html</uri>
     </address>
   </author>

   <author fullname="Kiem-Phong Vo" initials="K."
           surname="Vo">
     <organization>Google</organization>
     <address>
       <email>phongvo@gmail.com</email>
     </address>
   </author>

   <author fullname="Donald E. Eastlake 3rd" initials="D."
           surname="Eastlake">
     <organization>Futurewei Technologies</organization>
     <address>
       <postal>
         <street>2386 Panoramic Circle</street>
         <city>Apopka</city>
         <region>Florida</region>
         <code>32703</code>
         <country>USA</country>
       </postal>        
       <phone>+1-508-333-2270</phone>
       <email>d3e3e3@gmail.com</email>
       <email>donald.eastlake@futurewei.com</email>
     </address>
   </author>

   <author fullname="Tony Hansen" initials="T."
           surname="Hansen">
     <organization>AT&amp;T Laboratories</organization>
     <address>
       <postal>
         <street>200 Laurel Avenue South</street>
         <city>Middletown</city>
         <region>New Jersey</region>
         <code>07748</code>
         <country>USA</country>
       </postal>        
      <email>tony@att.com</email>
     </address>
   </author>

   <date year="2023" month="7" day="10"/>

   <area>Security</area>
   <workgroup>Internet Engineering Task Force</workgroup>
   <!-- "Internet Engineering Task Force" is fine for individual
        submissions.  If this element is not present, the default is
        "Network Working Group", which is used by the RFC Editor as a
        nod to the history of the RFC Series. --> 

   <keyword></keyword>
   <!-- Multiple keywords are allowed.  Keywords are incorporated
        into HTML output files for use by search engines. --> 

<abstract>
  <t>FNV (Fowler/Noll/Vo) is a fast, non-cryptographic hash algorithm
  with good dispersion. The purpose of this document is to make
  information on FNV and open source code performing FNV conveniently
  available to the Internet community.</t>
</abstract>

</front>


<!-- ____________________MIDDLE_MATTER____________________ -->
<middle>
    
<section> <!-- 1. -->
  <name>Introduction</name>

  <t>The FNV hash algorithm is based on an idea sent as reviewer
  comments to the <xref target="IEEE"/> POSIX P1003.2 committee by
  Glenn Fowler and Phong Vo in 1991. In a subsequent ballot round
  Landon Curt Noll suggested an improvement on their algorithm. Some
  people tried this hash and found that it worked rather well. In an
  EMail message to Landon, they named it the "Fowler/Noll/Vo" or FNV
  hash. <xref target="FNV"/></t>

  <t>FNV hashes are designed to be fast while maintaining a low
  collision rate. The high dispersion of the FNV hashes makes them
  well suited for hashing nearly identical strings such as URLs,
  hostnames, filenames, text, IP addresses, etc. Their speed allows
  one to quickly hash lots of data while maintaining a reasonably low
  collision rate. However, they are generally not suitable for
  cryptographic use. (See Section 7.1.)</t>

  <t>The FNV hash is widely used, for example in <xref
  target="RFC7357"/>, <xref target="RFC7873"/>, <xref
  target="IEEE8021Qbp"/>, DNS servers, the Twitter service, database
  indexing hashes, major web search / indexing engines, netnews
  history file Message-ID lookup functions, anti-spam filters, a
  spellchecker programmed in Ada 95, flatassembler's open source x86
  assembler - user-defined symbol hashtree, non-cryptographic file
  fingerprints, computing Unique IDs in DASM (DTN (Delay Tolerant
  Networking) Applications for Symbian Mobile-phones), Microsoft's
  hash_map implementation for VC++ 2005, the realpath cache in PHP 5.x
  (php-5.2.3/TSRM/tsrm_virtual_cwd.c), and many other uses. </t>

  <t>A study has recommended FNV in connection with the IPv6 Flow
  Label field <xref target="IPv6flow"/>. There was a proposal to use
  FNV for BFD sequence number generation <xref target="BFDseq"/>.</t>

  <t>FNV hash algorithms and source code have been released into the
  public domain. The authors of the FNV algorithm took deliberate
  steps to disclose the algorithm in a public forum soon after it was
  invented. More than a year passed after this public disclosure and
  the authors deliberately took no steps to patent the FNV algorithm.
  Therefore, it is safe to say that the FNV authors have no patent
  claims on the FNV algorithm as published. </t>

  <t>If you use an FNV function in an application, you are kindly
  requested to send an EMail about it to: fnv-mail@asthe.com</t>

</section> <!-- end 1. Introduction -->


<section> <!-- 2. -->
  <name>FNV Basics</name>

<t>This document focuses on the FNV-1a function whose pseudo-code is
as follows:</t>

<sourcecode type="pseudocode">
  hash = offset_basis
  for each octet_of_data to be hashed
      hash = hash xor octet_of_data
      hash = hash * FNV_Prime
  return hash
</sourcecode>

<t>In the pseudo-code above, hash is a power-of-two number of bits
(32, 64, ... 1024) and offset_basis and FNV_Prime depend on the size
of hash.</t>

<t>The FNV-1 algorithm is the same, including the values of
offset_basis and FNV_Prime, except that the order of the two lines
with the "xor" and multiply operations are reversed. Operational
experience indicates better hash dispersion for small amounts of data
with FNV-1a. FNV-0 is the same as FNV-1 but with offset_basis set to
zero. FNV-1a is suggested for general use.</t>

<section> <!-- 2.1 -->
  <name>FNV Primes</name>

<t>The theory behind FNV_Prime's is beyond the scope of this document
but the basic property to look for is how an FNV_Prime would impact
dispersion. Now, consider any n-bit FNV hash where n is >= 32 and also
a power of 2, in particular n = 2**s. For each such n-bit FNV hash, an
FNV_Prime p is defined as:</t>

<ul>
  <li><t>When s is an integer and 4 &lt; s &lt; 11, then
  FNV_Prime is the smallest prime p of the form:</t>

<artwork align="center" type="ascii-art">
256**int((5 + 2**s)/12) + 2**8 + b
</artwork>
</li>

  <li><t>where b is an integer such that:</t>

<artwork align="center" type="ascii-art">
0 &lt; b &lt; 2**8
</artwork>
</li>

<li><t>The number of one-bits in b
is 4 or 5</t></li>

<li><t>and where</t>
<artwork align="center" type="ascii-art">
( p mod (2**40 - 2**24 - 1) ) &gt; (2**24 + 2**8 + 2**7)
</artwork>
</li>
</ul>

<t>Experimentally, FNV_Primes matching the above constraints tend to
have better dispersion properties. They improve the polynomial
feedback characteristic when an FNV_Prime multiplies an intermediate
hash value. As such, the hash values produced are more scattered
throughout the n-bit hash space.</t>

<t>The case where s &lt; 5 is not considered because the resulting
hash quality is too low. Such small hashes can, if desired, be derived
from a 32 bit FNV hash by XOR folding (see Section 3). The case where
s &gt; 10 is not considered because of the doubtful utility of such
large FNV hashes and because the criteria for such large FNV_Primes is
more complex, due to the sparsity of such large primes, and would
needlessly clutter the criteria given above.</t>

<t>Per the above constraints, an FNV_Prime should have only 6 or 7
one-bits in it. Therefore, some compilers may seek to improve the
performance of a multiplication with an FNV_Prime by replacing the
multiplication with shifts and adds.  However, note that the
performance of this substitution is highly hardware-dependent and
should be done with care. FNV_Primes were selected primarily for the
quality of resulting hash function, not for compiler optimization.</t>

</section>

<section> <!-- 2.2 -->
  <name>FNV offset_basis</name>

<t>The offset_basis values for the n-bit FNV-1a algorithms are
computed by applying the n-bit FNV-0 algorithm to the 32 octets
representing the following character string in ASCII <xref
target="RFC0020"/>:</t>

<artwork align="center" type="ascii-art">
chongo &lt;Landon Curt Noll&gt; /\../\
</artwork>

<t>The \'s in the above string are not C-style escape characters. In
C-string notation, these 32 octets are:</t>

<artwork align="center" type="ascii-art">
"chongo &lt;Landon Curt Noll&gt; /\\../\\"
</artwork>

<t>That string was used because the person testing FNV with non-zero
offset_basis values was looking at an email message from Landon and
was copying his standard email signature line; however, they couldn't
see very well and copied it incorrectly. In fact, he uses</t>

<artwork align="center" type="ascii-art">
chongo (Landon Curt Noll) /\oo/\
</artwork>

<t>but, since it doesn't matter, no effort has been made to correct
this.</t>

<t>In the general case, almost any offset_basis will serve so long as
it is non-zero. The choice of a non-standard offset_basis may be
beneficial in defending against some attacks that try to induce hash
collisions.</t>

</section>

<section>  <!-- 2.3 -->
  <name>FNV Endianism</name>

<t>For persistent storage or interoperability between different
hardware platforms, an FNV hash shall be represented in the little
endian format. That is, the FNV hash will be stored in an array
hash[N] with N bytes such that its integer value can be retrieved as
follows:</t>

<sourcecode type="C">
  unsigned char   hash[N];
  for ( i = N-1, value = 0; i &gt;= 0; --i )
      value = ( value &lt;&lt; 8 ) + hash[i];
</sourcecode>

<t>Of course, when FNV hashes are used in a single process or a group
of processes sharing memory on processors with compatible endian-ness,
the natural endian-ness of those processors can be used regardless of
its type, little, big, or some other exotic form.</t>

<t>The code provided in Section 6 has FNV hash functions that return a
little endian byte vector. Because they are slightly more efficient,
code returning FNV hashes of 32-bit or 64-bit size as integers, on
computers supporting integers of those sides, are also provided. Such
integers are compatible with the same size byte vectors on little
endian computers but use of the functions returning integers on big
endian or other non-little-endian machines will be byte-reversed or
otherwise incompatible with the byte vectors.</t>

</section>
</section>

<section>  <!-- 3. -->
  <name>Other Hash Sizes and XOR Folding</name>

<t>Many hash uses require a hash that is not one of the FNV sizes for
which constants are provided in Section 5.  If a larger hash size is
needed, please contact the authors of this document.</t>

<t>Most hash applications make use of a hash that is a fixed size
binary field. Assume that k bits of hash are desired and k is less
than 1024 but not one of the sizes for which constants are provided in
Section 5. The recommended technique is to take the smallest FNV hash
of size S, where S is larger than k, and calculate the desired
k-bit-hash using xor folding as shown below. The final bit masking
operation is logically unnecessary if the size of the variable
k-bit-hash is exactly k bits.</t>

<sourcecode type="pseudocode">
  temp = FNV_S ( data-to-be-hashed )
  k-bit-hash = ( temp xor temp>>k ) bitwise-and ( 2**k - 1 )
</sourcecode>

<t>Hash functions are a trade-off between speed and strength. For
example, a somewhat stronger hash may be obtained for exact FNV sizes
by calculating an FNV twice as long as the desired output ( S = 2*k )
and performing such data folding using a k equal to the size of the
desired output. However, if a much stronger hash, for example one
suitable for cryptographic applications, is wanted, algorithms
designed for that purpose, such as those in <xref target="RFC6234"/>,
should be used.</t>

<t>If it is desired to obtain a hash result that is a value between 0
and max, where max+1 is a not a power of two, simply choose an FNV
hash size S such that 2**S > max. Then calculate the following:</t>

<artwork align="center" type="ascii-art">
FNV_S mod ( max+1 )
</artwork>
    
<t>The resulting remainder will be in the range desired but will
suffer from a bias against large values with the bias being larger if
2**S is only a little bigger than max. If this bias is acceptable, no
further processing is needed. If this bias is unacceptable, it can be
avoided by retrying for certain high values of hash, as follows,
before applying the mod operation above:</t>

<sourcecode type="pseudocode">
  X = ( int( ( 2**S - 1 ) / ( max+1 ) ) ) * ( max+1 )
  while ( hash >= X )
      hash = ( hash * FNV_Prime ) + offset_basis
</sourcecode>

</section> <!-- 3 -->

<section>  <!-- 4. -->
  <name>Hashing Multiple Values Together</name>
 
<t>It is common for there to be a few different component values, say
three strings X, Y, and Z, where a hash over all of them is desired.
The simplest thing to do is to concatenate them and compute the hash
of that concatenation, as in</t>

<artwork align="center" type="ascii-art">
hash ( X | Y | Z )
</artwork>

<t>where the vertical bar character ("|") represents string
concatenation.  Note that, for FNV, the same hash results if X, Y, and
Z are actually concatenated and the FNV hash applied to the resulting
string or if FNV is calculated on an initial substring and the result
used as the offset_basis when calculating the FNV hash of the
remainder of the string.  This can be done several times. Assuming
FNVoffset_basis ( v, w ) is FNV of w using v as the offset_basis, then
in the example above, fnvx = FNV ( X ) could be calculated and then
fnvxy = FNVoffset_basis ( fnvx, Y ), and finally fnvxyz =
FNVoffset_basis ( fnvxy, Z). The resulting fnvxyz would be the same as
FNV ( X | Y | Z ).</t>

<t>Cases are also common where such a hash needs to be repeatedly
calculated where the component values vary but some vary more
frequently than others.  For example, assume some sort of computer
network traffic flow ID, such as the IPv6 flow ID <xref
target="RFC6437"/>, is to be calculated for network packets based on
the source and destination IPv6 address and the Traffic Class <xref
target="RFC8200"/>. If the Flow ID is calculated in the originating
host, the source IPv6 address would likely always be the same or
perhaps assume one of a very small number of values. By placing this
quasi-constant IPv6 source address first in the string being FNV
hashed, FNV ( IPv6source ) could be calculated and used as the
offset_basis for calculating FNV of the IPv6 destination address and
Traffic Class for each packet. As a result, the per packet hash would
be over 17 bytes rather than over 33 bytes saving computational
resources. The code in this document includes functions facilitating
the use of a non-standard offset_basis.</t>

</section>  <!-- 4. -->

<section> <!-- 5. -->
  <name>FNV Constants</name>

  <t>The FNV Primes are as follows:</t>
  
<table>
  <thead>
<tr><th>Size FNV Prime = Exression</th></tr>
<tr><th align="right">= Decimal</th></tr>
<tr><th align="right">= Hexadecimal</th></tr>
  </thead>
  <tbody>
    
<tr><td>32 bit FNV_Prime = 2**24 + 2**8 + 0x93</td></tr>
<tr><td align="right">= 16,&zwsp;777,&zwsp;619</td></tr>
<tr><td align="right">= 0x01000193</td></tr>

<tr><td>64 bit FNV_Prime = 2**40 + 2**8 + 0xB3</td></tr>
<tr><td align="right">= 1,&zwsp;099,&zwsp;511,&zwsp;628,&zwsp;211</td></tr>
<tr><td align="right"> = 0x00000100 000001B3</td></tr>

<tr><td>128 bit FNV_Prime = 2**88 + 2**8 + 0x3B</td></tr>
<tr><td align="right">=
309,&zwsp;485,&zwsp;009,&zwsp;821,&zwsp;345,&zwsp;068,&zwsp;724,&zwsp;781,&zwsp;371</td></tr>
<tr><td align="right">= 0x00000000 01000000 00000000
0000013B</td></tr>

<tr><td>256 bit FNV_Prime = 2**168 + 2**8 + 0x63</td></tr>
<tr><td align="right">=
374,&zwsp;144,&zwsp;419,&zwsp;156,&zwsp;711,&zwsp;147,&zwsp;060,&zwsp;143,&zwsp;317,&zwsp;175,&zwsp;368,&zwsp;453,&zwsp;031,&zwsp;918,&zwsp;731,&zwsp;002,&zwsp;211</td></tr>
<tr><td align="right">= 0x0000000000000000
0000010000000000 0000000000000000 0000000000000163</td></tr>

<tr><td>512 bit FNV_Prime = 2**344 + 2**8 + 0x57</td></tr>
<tr><td align="right">= 35,&zwsp;835,&zwsp;915,&zwsp;874,&zwsp;844,&zwsp;867,&zwsp;368,&zwsp;919,&zwsp;076,&zwsp;489,&zwsp;095,&zwsp;108,&zwsp;449,&zwsp;946,&zwsp;327,&zwsp;955,&zwsp;754,&zwsp;392,&zwsp;558,&zwsp;399,&zwsp;825,&zwsp;615,&zwsp;420,&zwsp;669,&zwsp;938,&zwsp;882,&zwsp;575,&zwsp;126,&zwsp;094,&zwsp;039,&zwsp;892,&zwsp;345,&zwsp;713,&zwsp;852,&zwsp;759</td></tr>
<tr><td align="right">= 0x0000000000000000
0000000000000000 0000000001000000 0000000000000000 0000000000000000
0000000000000000 0000000000000000 0000000000000157</td></tr>

<tr><td>1024 bit FNV_Prime = 2**680 + 2**8 + 0x8D</td></tr>
<tr><td align="right">= 5,&zwsp;016,&zwsp;456,&zwsp;510,&zwsp;113,&zwsp;118,&zwsp;655,&zwsp;434,&zwsp;598,&zwsp;811,&zwsp;035,&zwsp;278,&zwsp;955,&zwsp;030,&zwsp;765,&zwsp;345,&zwsp;404,&zwsp;790,&zwsp;744,&zwsp;303,&zwsp;017,&zwsp;523,&zwsp;831,&zwsp;112,&zwsp;055,&zwsp;108,&zwsp;147,&zwsp;451,&zwsp;509,&zwsp;157,&zwsp;692,&zwsp;220,&zwsp;295,&zwsp;382,&zwsp;716,&zwsp;162,&zwsp;651,&zwsp;878,&zwsp;526,&zwsp;895,&zwsp;249,&zwsp;385,&zwsp;292,&zwsp;291,&zwsp;816,&zwsp;524,&zwsp;375,&zwsp;083,&zwsp;746,&zwsp;691,&zwsp;371,&zwsp;804,&zwsp;094,&zwsp;271,&zwsp;873,&zwsp;160,&zwsp;484,&zwsp;737,&zwsp;966,&zwsp;720,&zwsp;260,&zwsp;389,&zwsp;217,&zwsp;684,&zwsp;476,&zwsp;157,&zwsp;468,&zwsp;082,&zwsp;573</td></tr>
<tr><td align="right">= 0x0000000000000000
0000000000000000 0000000000000000 0000000000000000 0000000000000000
0000010000000000 0000000000000000 0000000000000000 0000000000000000
0000000000000000 0000000000000000 0000000000000000 0000000000000000
0000000000000000 0000000000000000 000000000000018D</td></tr>

  </tbody>
</table>

<t>The FNV offset_basis values are as follows:</t>

<table>
  <thead>
<tr><th>Size offset_basis</th></tr>
<tr><th align="right">= Decimal</th></tr>
<tr><th align="right">= Hexadecimal</th></tr>
  </thead>
  <tbody>

<tr><td>32 bit offset_basis</td></tr>
<tr><td align="right">= 2,&zwsp;166,&zwsp;136,&zwsp;261</td></tr>
<tr><td>= 0x811C9DC5</td></tr>

<tr><td>64 bit offset_basis</td></tr>
<tr><td align="right">=
14,&zwsp;695,&zwsp;981,&zwsp;039,&zwsp;346,&zwsp;656,&zwsp;037</td></tr>
<tr><td align="right">= 0xCBF29CE4 84222325</td></tr>

<tr><td>128 bit offset_basis</td></tr>
<tr><td align="right">=
144,&zwsp;066,&zwsp;263,&zwsp;297,&zwsp;769,&zwsp;815,&zwsp;596,&zwsp;495,&zwsp;629,&zwsp;667,&zwsp;062,&zwsp;367,&zwsp;629</td></tr>
<tr><td align="right">= 0x6C62272E 07BB0142 62B82175 6295C58D</td></tr>

<tr><td>256 bit offset_basis</td></tr>
<tr><td align="right">=
100,&zwsp;029,&zwsp;257,&zwsp;958,&zwsp;052,&zwsp;580,&zwsp;907,&zwsp;070,&zwsp;968,&zwsp;620,&zwsp;625,&zwsp;704,&zwsp;837,&zwsp;092,&zwsp;796,&zwsp;014,&zwsp;241,&zwsp;193,&zwsp;945,&zwsp;225,&zwsp;284,&zwsp;501,&zwsp;741,&zwsp;471,&zwsp;925,&zwsp;557</td></tr>
<tr><td align="right">= 0xDD268DBCAAC55036 2D98C384C4E576CC
C8B1536847B6BBB3 1023B4C8CAEE0535</td></tr>

<tr><td>512 bit offset_basis</td></tr> <tr><td align="right">= 9,&zwsp;659,&zwsp;303,&zwsp;129,&zwsp;496,&zwsp;669,&zwsp;498,&zwsp;009,&zwsp;435,&zwsp;400,&zwsp;716,&zwsp;310,&zwsp;466,&zwsp;090,&zwsp;418,&zwsp;745,&zwsp;672,&zwsp;637,&zwsp;896,&zwsp;108,&zwsp;374,&zwsp;329,&zwsp;434,&zwsp;462,&zwsp;657,&zwsp;994,&zwsp;582,&zwsp;932,&zwsp;197,&zwsp;716,&zwsp;438,&zwsp;449,&zwsp;813,&zwsp;051,&zwsp;892,&zwsp;206,&zwsp;539,&zwsp;805,&zwsp;784,&zwsp;495,&zwsp;328,&zwsp;239,&zwsp;340,&zwsp;083,&zwsp;876,&zwsp;191,&zwsp;928,&zwsp;701,&zwsp;583,&zwsp;869,&zwsp;517,&zwsp;785</td></tr>
<tr><td align="right">= 0xB86DB0B1171F4416 DCA1E50F309990AC
AC87D059C9000000 0000000000000D21 E948F68A34C192F6 2EA79BC942DBE7CE
182036415F56E34B AC982AAC4AFE9FD9</td></tr>

<tr><td>1024 bit offset_basis</td></tr> <tr><td align="right">= 14,&zwsp;197,&zwsp;795,&zwsp;064,&zwsp;947,&zwsp;621,&zwsp;068,&zwsp;722,&zwsp;070,&zwsp;641,&zwsp;403,&zwsp;218,&zwsp;320,&zwsp;880,&zwsp;622,&zwsp;795,&zwsp;441,&zwsp;933,&zwsp;960,&zwsp;878,&zwsp;474,&zwsp;914,&zwsp;617,&zwsp;582,&zwsp;723,&zwsp;252,&zwsp;296,&zwsp;732,&zwsp;303,&zwsp;717,&zwsp;722,&zwsp;150,&zwsp;864,&zwsp;096,&zwsp;521,&zwsp;202,&zwsp;355,&zwsp;549,&zwsp;365,&zwsp;628,&zwsp;174,&zwsp;669,&zwsp;108,&zwsp;571,&zwsp;814,&zwsp;760,&zwsp;471,&zwsp;015,&zwsp;076,&zwsp;148,&zwsp;029,&zwsp;755,&zwsp;969,&zwsp;804,&zwsp;077,&zwsp;320,&zwsp;157,&zwsp;692,&zwsp;458,&zwsp;563,&zwsp;003,&zwsp;215,&zwsp;304,&zwsp;957,&zwsp;150,&zwsp;157,&zwsp;403,&zwsp;644,&zwsp;460,&zwsp;363,&zwsp;550,&zwsp;505,&zwsp;412,&zwsp;711,&zwsp;285,&zwsp;966,&zwsp;361,&zwsp;610,&zwsp;267,&zwsp;868,&zwsp;082,&zwsp;893,&zwsp;823,&zwsp;963,&zwsp;790,&zwsp;439,&zwsp;336,&zwsp;411,&zwsp;086,&zwsp;884,&zwsp;584,&zwsp;107,&zwsp;735,&zwsp;010,&zwsp;676,&zwsp;915</td></tr>
<tr><td align="right">= 0x0000000000000000 005F7A76758ECC4D
32E56D5A591028B7 4B29FC4223FDADA1 6C3BF34EDA3674DA 9A21D90000000000
0000000000000000 0000000000000000 0000000000000000 0000000000000000
0000000000000000 000000000004C6D7 EB6E73802734510A 555F256CC005AE55
6BDE8CC9C6A93B21 AFF4B16C71EE90B3</td></tr>

  </tbody>
</table>

</section>

<section>  <!-- 6. -->
  <name>The Source Code</name>

<t>(((THIS CODE IS BEING WORKED ON)))</t>

<t>The following sub-sections provide reference C source code and a
test driver for FNV-1a.</t>

<t>Alternative source code, including 32 and 64 bit FNV-1 and FNV-1a
in x86 assembler, is currently available at <xref target="FNV"/>.</t>

<t>Section 6.2 provides the test driver.</t>

<section>  <!-- 6.1 -->
  <name>FNV-1a C Code</name>

  <t>This section provides the direct FNV-1a function for each of the
  lengths for which it is specified in this document.  The functions
  provided are listed below. The xxx in the function names is "32",
  "64", "128", "256", "512", or "1024" depending on the length of the
  FNV. The FNV-32 functions return a 32 bit integer. The FNV-64 bit
  functions return a 64-bit integer or an 8 byte vector depending on
  whether 64-bit integers are or are not supports. All longer FNV
  functions return a byte vector. Versions returning an integer may
  NOT compatible between systems of different endian-ness.</t>

  <t>If you want to copy the source code from this document, note that
  it is indented by three spaces in the ".txt" version. It may be
  simplest to copy from the ".html" version of this documents.</t>

  <dl>
    
  <dt>FNVxxxstring, FNVxxxblock:</dt><dd>These are simple functions
  for directly returning the FNV hash of a zero terminated byte string
  not including the zero and the FNV hash of a counted block of
  bytes. Note that for applications of FNV-32 where 32-bit integers
  are supported and FNV-64 where 64-bit integers are supported and an
  integer data type output is acceptable, the code is sufficiently
  simple that, to maximize performance, use of open coding or macros
  may be more appropriate than calling a subroutine.</dd>

  <dt>FNVxxxinit, FNVxxxinitBasis:</dt><dd>These functions and the
  next two sets of functions below provide facilities for
  incrementally calculating FNV hashes. They all assume a data
  structure of type FNVxxxcontext that holds the current state of
  the hash. FNVxxxinit initializes that context to the standard
  offset_basis. FNVxxxinitBasis takes an offset_basis value as a
  parameter and may be useful for hashing concatenations, as described
  in Section 4, as well as for simply using a non-standard
  offset_basis.</dd>

  <dt>FNVxxxblockin, FNVxxxstringin:</dt><dd>These functions hash a
  sequence of bytes into an FNVxxxcontext that was originally
  initialized by FNVxxxinit or FNVxxxinitBasis. FNVxxxblockin
  hashes in a counted block of bytes. FNVxxxstringin hashes in a
  zero terminated byte string not incuding the final zero.</dd>

  <dt>FNVxxxresult:</dt><dd>This function extracts the final FNV hash
  result from an FNVxxxcontext.</dd>
  
</dl>

<t>The following code is a private header file used by all the FNV
functions further below and which states the terms for use and
redistribution of all of this code.</t>

<sourcecode type="C" markers="true" name="fnv-private.h">
/************************ fnv-private.h ************************/
/****************** See RFC NNNN for details *******************/
/* Copyright (c) 2016, 2023 IETF Trust and the persons identified as
 * authors of the code.  All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 *
 * *  Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 *
 * *  Redistributions in binary form must reproduce the above
 *    copyright notice, this list of conditions and the following
 *    disclaimer in the documentation and/or other materials provided
 *    with the distribution.
 *
 * *  Neither the name of Internet Society, IETF or IETF Trust, nor
 *    the names of specific contributors, may be used to endorse or
 *    promote products derived from this software without specific
 *    prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND
 * CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
 * INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
 * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS
 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED
 * TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
 * DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
 * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR
 * TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
 * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
 * SUCH DAMAGE.
 */

#ifndef _FNV_PRIVATE_H_
#define _FNV_PRIVATE_H_

/*
 *      Six FNV-1a hashes are defined with these sizes:
 *              FNV32          32 bits, 4 bytes
 *              FNV64          64 bits, 8 bytes
 *              FNV128         128 bits, 16 bytes
 *              FNV256         256 bits, 32 bytes
 *              FNV512         512 bits, 64 bytes
 *              FNV1024        1024 bits, 128 bytes
 */

/* Private stuff used by this implementation of the FNV
 * (Fowler, Noll, Vo) non-cryptographic hash function FNV-1a.
 * External callers don't need to know any of this.  */

enum {  /* State value bases for context->Computed */
    FNVinited = 22,
    FNVcomputed = 76,
    FNVemptied = 220,
    FNVclobber = 122 /* known bad value for testing */
};

/* Deltas to assure distinct state values for different lengths */
enum {
   FNV32state = 1,
   FNV64state = 3,
   FNV128state = 5,
   FNV256state = 7,
   FNV512state = 11,
   FNV1024state = 13
};

#endif
</sourcecode>


<t>The following code is a simple header file to define the return
value error codes for the FNV routines.</t>


<sourcecode type="C" markers="true" name="FNVErrorCodes.h">
/************************ FNVErrorCodes.h *************************/
/****************** See RFC NNNN for details. *********************/
/*
 * Copyright (c) 2016, 2023 IETF Trust and the persons identified as
 * authors of the code.  All rights reserved.
 * See fnv-private.h for terms of use and redistribution.
 */

#ifndef _FNV_ErrCodes_
#define _FNV_ErrCodes_
/*******************************************************************
 *  All FNV functions provided return as integer as follows:
 *       0 -> success
 *      >0 -> error as listed below
 */
enum {    /* success and errors */
    fnvSuccess = 0,
    fnvNull,          /* Null pointer parameter */
    fnvStateError,    /* called Input after Result or before Init */
    fnvBadParam       /* passed a bad parameter */
};
#endif /* _FNV_ErrCodes_ */
</sourcecode>


<t>The following code is a simple header file to include all the
specific length FNV header files.</t>


<sourcecode type="C" markers="true" name="FNV.h">
/***************************** FNV.h ******************************/
/****************** See RFC NNNN for details. *********************/
/*
 * Copyright (c) 2016, 2023 IETF Trust and the persons identified as
 * authors of the code.  All rights reserved.
 * See fnv-private.h for terms of use and redistribution.
 */

#ifndef _FNV_H_
#define _FNV_H_

#include "FNV32.h"
#include "FNV64.h"
#include "FNV128.h"
#include "FNV256.h"
#include "FNV512.h"
#include "FNV1024.h"

#endif /* _FNV_H_ */
</sourcecode>


<t>The following code is a simple header file to control configuration
related to big integer and big endian support.</t>


<sourcecode type="C" markers="true" name="FNVconfig.h">
<![CDATA[
/************************** FNVconfig.h ***************************/
/****************** See RFC NNNN for details. *********************/
/*
 * Copyright (c) 2016, 2023 IETF Trust and the persons identified as
 * authors of the code.  All rights reserved.
 * See fnv-private.h for terms of use and redistribution.
 */

#ifndef _FNVconfig_H_
#define _FNVconfig_H_

/*
 *  Description:
 *      This file provides configuration ifdefs for the
 *      FNV-1a non-cryptographic hash algorithms.
 *
 * >>>>>>>> IMPORTANT CONFIGURATION ifdefs: <<<<<<<<<< */

/*      FNV_64bitIntegers - Define this if your system supports
 *          64-bit arithmetic including 32-bit x 32-bit 
 *          multiplication producing a 64-bit product. If
 *          undefined, it will be assumed that 32-bit arithmetic
 *          is supported including 16-bit x 16-bit multiplication
 *          producing a 32-bit result.
 */
// #define FNV_64bitIntegers

/*
 *
 *      FNV_BigEndian - Define this ONLY if your system uses big
 *          endian representation AND your FNV hashes need to
 *          interoperate with little endian systems. If you #define
 *          this symbol when not needed, it will unnecessarily slow
 *          down and increase the code size of the FNV functions.
 */
// #define FNV_BigEndian


/*
 *      The following allow the FNV test program to override the
 *      above configuration settings.
 */

#ifdef FNV_TEST_PROGRAM
# ifdef TEST_FNV_64bitIntegers
#  ifndef FNV_64bitIntegers
#   define FNV_64bitIntegers
#  endif
# else
#  undef FNV_64bitIntegers
# endif
# ifndef FNV_64bitIntegers /* causes an error if uint64_t is used */
#  undef uint64_t
#  define uint64_t no_64_bit_integers
# endif
# ifdef TEST_FNV_BigEndian
#  ifndef FNV_BigEndian
#   define FNV_BigEndian
#  endif
# else
#  undef FNV_BigEndian
# endif
#endif

#endif /* _FNVconfig_H_ */
]]></sourcecode>


<section>
  <name>FNV32 Code</name>

<t>The header and C source for 32-bit FNV-1a returning a 32-bit
integer.</t>


<sourcecode type="C" markers="true" name="FNV32.h">
<![CDATA[
/**************************** FNV32.h *****************************/
/******************* See RFC NNNN for details *********************/
/*
 * Copyright (c) 2016, 2023 IETF Trust and the persons identified as
 * authors of the code.  All rights reserved.
 * See fnv-private.h for terms of use and redistribution.
*/

#ifndef _FNV32_H_
#define _FNV32_H_

#include "FNVconfig.h"

#include <stdint.h>
/* #define FNV32size (32/8)   not used */

/* If you do not have the ISO standard stdint.h header file, then you
 * must typedef the following types:
 *
 *    type              meaning
 *  uint32_t         unsigned 32 bit integer
 *  uint8_t          unsigned 8 bit integer (i.e., unsigned char)
 */

#include "FNVErrorCodes.h"

/*
 *  This structure holds context information for an FNV32 hash
 */
typedef struct FNV32context_s {
    int Computed;  /* state */
    uint32_t Hash;
} FNV32context;

/*
 *  Function Prototypes
 *    FNV32string: hash a zero terminated string not including
 *                 the terminating zero
 *    FNV32block: hash a specified length byte vector
 *    FNV32init:  initializes an FNV32 context
 *    FNV32initBasis: initializes an FNV32 context with a
 *                    provided basis
 *    FNV32blockin:  hash in a specified length byte vector
 *    FNV32stringin: hash in a zero terminated string not
 *                   including the zero
 *    FNV32result: returns the hash value
 *
 *    Hash is returned as a 32-bit unsigned integer
 */

#ifdef __cplusplus
extern "C" {
#endif

/* FNV32 */
extern int FNV32string ( const char *in,
                         uint32_t * const out );
extern int FNV32block ( const void *in,
                        long int inlength,
                        uint32_t * const out );
extern int FNV32init ( FNV32context * const );
extern int FNV32initBasis ( FNV32context * const,
                            uint32_t basis );
extern int FNV32blockin ( FNV32context * const,
                          const void *in,
                          long int inlength );
extern int FNV32stringin ( FNV32context * const,
                           const char *in );
extern int FNV32result ( FNV32context * const,
                         uint32_t * const out );

#ifdef __cplusplus
}
#endif

#endif /* _FNV32_H_ */
]]></sourcecode>


<sourcecode type="C" markers="true" name="FNV32.c">
<![CDATA[
/*************************** FNV32.c ***************************/
/***************** See RFC NNNN for details. *******************/
/* Copyright (c) 2016, 2023 IETF Trust and the persons identified
 * as authors of the code.  All rights reserved.
 * See fnv-private.h for terms of use and redistribution.
 */

/* This code implements the FNV (Fowler, Noll, Vo)
 *  non-cryptographic hash function FNV-1a for 32-bit hashes.
 */

#ifndef _FNV32_C_
#define _FNV32_C_

#include "fnv-private.h"
#include "FNV32.h"

/* 32 bit FNV_prime = 2^24 + 2^8 + 0x93 */
#define FNV32prime 0x01000193
#define FNV32basis 0x811C9DC5

#ifdef FNV_BigEndian
/* Local prototype */
static void FNV32reverse ( uint32_t *out, uint32_t hash );
#endif

/* FNV32 hash a zero terminated string not including the zero
*****************************************************************/
int FNV32string ( const char *in, uint32_t * const out )
{
    uint32_t    temp;
    uint8_t     ch;

    if ( !in || !out )
        return fnvNull; /* Null input pointer */

    temp = FNV32basis;
    while ( (ch = *in++) )
        temp = FNV32prime * ( temp ^ ch );
#ifdef FNV_BigEndian
    FNV32reverse ( out, temp );
#else
    *out = temp;
#endif
    return fnvSuccess;
}   /* end FNV32string */

/* FNV32 hash a counted block
 ****************************************************************/
int FNV32block ( const void *vin,
                 long int length,
                 uint32_t * const out )
{
    const uint8_t *in = (const uint8_t*)vin;
    uint32_t    temp;

    if ( !in || !out )
        return fnvNull; /* Null input pointer */

    if ( length < 0 )
        return fnvBadParam;
    for ( temp = FNV32basis; length > 0; length-- )
        temp = FNV32prime * ( temp ^ *in++ );
#ifdef FNV_BigEndian
    FNV32reverse ( out, temp );
#else
    *out = temp;
#endif
    return fnvSuccess;
}   /* end FNV32block */

#ifdef FNV_BigEndian

/* Store a Big Endian result back as Little Endian
 ***************************************************************/
static void FNV32reverse ( uint32_t *out, uint32_t hash )
{
    uint32_t    temp;

    temp = hash & 0xFF;
    hash >>= 8;
    temp = ( temp << 8 ) + ( hash & 0xFF );
    hash >>= 8;
    temp = ( temp << 8 ) + ( hash & 0xFF );
    hash >>= 8;
    *out = ( temp << 8 ) + ( hash & 0xFF );
}   /* end FNV32reverse */

#endif /* FNV_BigEndian */


/***************************************************************
 *       Set of init, input, and output functions below        *
 *       to incrementally compute FNV32                        *
 ***************************************************************/

/* initialize context
 ***************************************************************/
int FNV32init ( FNV32context * const ctx )
{
    return FNV32initBasis ( ctx, FNV32basis );
}   /* end FNV32init */

/* initialize context with a provided basis
 ***************************************************************/
int FNV32initBasis ( FNV32context * const ctx, uint32_t basis )
{
    if ( !ctx )
        return fnvNull;

    ctx->Hash = basis;
    ctx->Computed = FNVinited+FNV32state;
    return fnvSuccess;
}   /* end FNV32initBasis */

/* hash in a counted block
 ***************************************************************/
int FNV32blockin ( FNV32context * const ctx,
                   const void *vin,
                   long int length )
{
    const uint8_t *in = (const uint8_t*)vin;
    uint32_t    temp;

    if ( !ctx || !in )
        return fnvNull;

    if ( length < 0 )
        return fnvBadParam;
    switch ( ctx->Computed )
        {
        case FNVinited+FNV32state:
            ctx->Computed = FNVcomputed+FNV32state;
        case FNVcomputed+FNV32state:
            break;
        default:
            return fnvStateError;
        }
    for ( temp = ctx->Hash; length > 0; length-- )
        temp = FNV32prime * ( temp ^ *in++ );
    ctx->Hash = temp;
    return fnvSuccess;
}   /* end FNV32blockin */

/* hash in a zero terminated string not including the zero
 ***************************************************************/
int FNV32stringin ( FNV32context * const ctx,
                    const char *in )
{
    uint32_t    temp;
    uint8_t     ch;

    if ( !ctx || !in )
        return fnvNull;

    switch ( ctx->Computed )
        {
        case FNVinited+FNV32state:
            ctx->Computed = FNVcomputed+FNV32state;
        case FNVcomputed+FNV32state:
            break;
        default:
            return fnvStateError;
        }
    temp = ctx->Hash;
    while ( (ch = (uint8_t)*in++) )
        temp = FNV32prime * ( temp ^ ch );
    ctx->Hash = temp;
    return fnvSuccess;
}   /* end FNV32stringin */

/* return hash
 ***************************************************************/
int FNV32result ( FNV32context * const ctx,
                  uint32_t * const out )
{
    if ( !ctx || !out )
        return fnvNull;

    if ( ctx->Computed != FNVcomputed+FNV32state )
        return fnvStateError;
    ctx->Computed = FNVemptied+FNV32state;
#ifdef FNV_BigEndian
    FNV32reverse ( out, ctx->Hash );
#else
    *out = ctx->Hash;
#endif
    ctx->Hash = 0;
    return fnvSuccess;
}   /* end FNV32result */

#endif    /* _FNV32_C_ */
]]></sourcecode>


</section>

<section>
  <name>FNV64 Code</name>

<t>The header and C source for 64-bit FNV-1a. Return a 64-bit integers
are supported, otherwise return an 8-byte vectir if bytes.</t>


<sourcecode type="C" markers="true" name="FNV64.h">
<![CDATA[
/**************************** FNV64.h *****************************/
/****************** See RFC NNNN for details. *********************/
/*
 * Copyright (c) 2016, 2023 IETF Trust and the persons identified as
 * authors of the code.  All rights reserved.
 * See fnv-private.h for terms of use and redistribution.
 */

#ifndef _FNV64_H_
#define _FNV64_H_

/*
 *  Description:
 *      This file provides headers for the 64-bit version of the
 *      FNV-1a non-cryptographic hash algorithm.
 */

#include "FNVconfig.h"

#include <stdint.h>
#define FNV64size (64/8)

/* If you do not have the ISO standard stdint.h header file, then you
 * must typedef the following types:
 *
 *    type             meaning
 *  uint64_t        unsigned 64 bit integer (ifdef FNV_64bitIntegers)
 *  uint32_t        unsigned 32 bit integer
 *  uint16_t        unsigned 16 bit integer
 *  uint8_t         unsigned 8 bit integer (i.e., unsigned char)
 */

#include "FNVErrorCodes.h"

/*
 *  This structure holds context information for an FNV64 hash
 */
#ifdef FNV_64bitIntegers
    /* version if 64 bit integers supported */

typedef struct FNV64context_s {
        int Computed;  /* state */
        uint64_t Hash;
} FNV64context;

#else
    /* version if 64 bit integers NOT supported */

typedef struct FNV64context_s {
        int Computed;  /* state */
        uint16_t Hash[FNV64size/2];
} FNV64context;

#endif /* FNV_64bitIntegers */

/*
 *  Function Prototypes
 *    FNV64string: hash a zero terminated string not including
 *                 the terminating zero
 *    FNV64block: FNV64 hash a specified length byte vector
 *    FNV64init: initializes an FNV64 context
 *    FNV64initBasis: initializes an FNV64 context with a
 *                    provided basis
 *    FNV64blockin: hash in a specified length byte vector
 *    FNV64stringin: hash in a zero terminated string not
 *                   incluing the zero
 *    FNV64result: returns the hash value
 *
 *    Hash is returned as a 64-bit integer if supported, otherwise
 *         as a vector of 8-bit unsigned integers
 */

#ifdef __cplusplus
extern "C" {
#endif

/* FNV64 */
extern int FNV64init ( FNV64context * const );
extern int FNV64blockin ( FNV64context * const,
                          const void * in,
                          long int length );
extern int FNV64stringin ( FNV64context * const,
                           const char * in );

#ifdef FNV_64bitIntegers
  extern int FNV64string ( const char *in,
                           uint64_t * const out );
  extern int FNV64block ( const void *in,
                          long int length,
                          uint64_t * const out );
  extern int FNV64initBasis ( FNV64context * const,
                            uint64_t basis );
  extern int FNV64result ( FNV64context * const,
                           uint64_t * const out );
#else
  extern int FNV64string ( const char *in,
                           uint8_t out[FNV64size] );
  extern int FNV64block ( const void *in,
                          long int length,
                          uint8_t out[FNV64size] );
  extern int FNV64initBasis ( FNV64context * const,
                            const uint8_t basis[FNV64size] );
  extern int FNV64result ( FNV64context * const,
                           uint8_t out[FNV64size] );
#endif /* FNV_64bitIntegers */

#ifdef __cplusplus
}
#endif

#endif /* _FNV64_H_ */
]]></sourcecode>


<sourcecode type="C" markers="true" name="FNV64.c">
<![CDATA[
/**************************** FNV64.c *****************************/
/******************* See RFC NNNN for details *********************/
/* Copyright (c) 2016, 2023 IETF Trust and the persons identified as
 * authors of the code.  All rights reserved.
 * See fnv-private.h for terms of use and redistribution.
 */

/* This file implements the FNV (Fowler, Noll, Vo) non-cryptographic
 * hash function FNV-1a for 64-bit hashes.
 */

#ifndef _FNV64_C_
#define _FNV64_C_

#include "fnv-private.h"
#include "FNV64.h"

/******************************************************************
 *       START VERSION FOR WHEN YOU HAVE 64 BIT ARITHMETIC        *
 ******************************************************************/
#ifdef FNV_64bitIntegers

#ifdef FNV_BigEndian
/* Local prototype */
static void FNV64reverse ( uint64_t * out, uint64_t hash );
#endif /* FNV_BigEndian */

/* 64 bit FNV_prime = 2^40 + 2^8 + 0xb3 */
#define FNV64prime 0x00000100000001B3
#define FNV64basis 0xCBF29CE484222325

/* FNV64 hash a null terminated string  (64 bit)
 ******************************************************************/
int FNV64string ( const char *in, uint64_t * const out )
{
    uint64_t    temp;
    uint8_t     ch;

    if ( !in || !out )
        return fnvNull; /* Null input pointer */

    temp = FNV64basis;
    while ( (ch = *in++) )
        temp = FNV64prime * ( temp ^ ch );
#ifdef FNV_BigEndian
    FNV64reverse ( out, temp );
#else
    *out = temp;
#endif
    return fnvSuccess;
}   /* end FNV64string */

/* FNV64 hash a counted block  (64 bit)
 ******************************************************************/
int FNV64block ( const void *vin,
                 long int length,
                 uint64_t * const out )
{
    const uint8_t *in = (const uint8_t*)vin;
    uint64_t    temp;

    if ( !in || !out )
        return fnvNull; /* Null input pointer */

    if ( length < 0 )
        return fnvBadParam;
    for ( temp = FNV64basis; length > 0; length-- )
        temp = FNV64prime * ( temp ^ *in++ );
#ifdef FNV_BigEndian
    FNV64reverse ( out, temp );
#else
    *out = temp;
#endif
    return fnvSuccess;
}   /* end FNV64block */

#ifdef FNV_BigEndian

/* Store a Big Endian result back as Little Endian
 ***************************************************************/
static void FNV64reverse ( uint64_t *out, uint64_t hash )
{
    uint64_t    temp;
    int         i;

    temp = hash & 0xFF;
    for ( i = FNV64size - 1; i > 0; i-- )
        {
        hash >>= 8;
        temp = ( temp << 8 ) + ( hash & 0xFF );
        }
    *out = temp;
}   /* end FNV64reverse */

#endif /* FNV_BigEndian */


/******************************************************************
 *        Set of init, input, and output functions below          *
 *        to incrementally compute FNV64                          *
 ******************************************************************/

/* initialize context  (64 bit)
 ******************************************************************/
int FNV64init ( FNV64context * const ctx )
{
    return FNV64initBasis ( ctx, FNV64basis );
}       /* end FNV64init */

/* initialize context with a provided basis  (64 bit)
 ******************************************************************/
int FNV64initBasis ( FNV64context * const ctx, uint64_t basis )
{
    if ( !ctx )
        return fnvNull;

    ctx->Hash = basis;
    ctx->Computed = FNVinited+FNV64state;
    return fnvSuccess;
}       /* end FNV64initBasis */

/* hash in a counted block  (64 bit)
 ******************************************************************/
int FNV64blockin ( FNV64context * const ctx,
                   const void *vin,
                   long int length )
{
    const uint8_t *in = (const uint8_t*)vin;
    uint64_t    temp;

    if ( !ctx || !in )
        return fnvNull;

    if ( length < 0 )
        return fnvBadParam;
    switch ( ctx->Computed )
        {
        case FNVinited+FNV64state:
            ctx->Computed = FNVcomputed+FNV64state;
        case FNVcomputed+FNV64state:
            break;
        default:
            return fnvStateError;
        }
    for ( temp = ctx->Hash; length > 0; length-- )
        temp = FNV64prime * ( temp ^ *in++ );
    ctx->Hash = temp;
    return fnvSuccess;
}   /* end FNV64input */

/* hash in a zero terminated string not including the zero (64 bit)
 ******************************************************************/
int FNV64stringin ( FNV64context * const ctx,
                    const char *in )
{
    uint64_t        temp;
    uint8_t         ch;

    if ( !ctx || !in )
        return fnvNull;

    switch ( ctx->Computed )
        {
        case FNVinited+FNV64state:
            ctx->Computed = FNVcomputed+FNV64state;
        case FNVcomputed+FNV64state:
            break;
        default:
             return fnvStateError;
         }
    temp = ctx->Hash;
    while ( (ch = *in++) )
        temp = FNV64prime * ( temp ^ ch );
    ctx->Hash = temp;
    return fnvSuccess;
}   /* end FNV64stringin */

/* return hash  (64 bit)
 ******************************************************************/
int FNV64result ( FNV64context * const ctx,
                  uint64_t * const out )
{
    if ( !ctx || !out )
        return fnvNull;

    if ( ctx->Computed != FNVcomputed+FNV64state )
        return fnvStateError;
    ctx->Computed = FNVemptied+FNV64state;
#ifdef FNV_BigEndian
    FNV64reverse ( out, ctx->Hash );
#else
    *out = ctx->Hash;
#endif
    ctx->Hash = 0;
    return fnvSuccess;
}   /* end FNV64result */

/****************************************************************
 *       END VERSION FOR WHEN YOU HAVE 64 BIT ARITHMETIC        *
 ****************************************************************/
#else    /*  FNV_64bitIntegers */
/****************************************************************
 *    START VERSION FOR WHEN YOU ONLY HAVE 32-BIT ARITHMETIC    *
 ****************************************************************/

/* 64 bit FNV_prime = 2^40 + 2^8 + 0xb3 */
/* #define FNV64prime 0x00000100000001B3 */
#define FNV64primeX 0x01B3
#define FNV64shift 8

/* #define FNV64basis 0xCBF29CE484222325 */
#define FNV64basis0 0xCBF2
#define FNV64basis1 0x9CE4
#define FNV64basis2 0x8422
#define FNV64basis3 0x2325

/* FNV64 hash a null terminated string  (32 bit)
 ******************************************************************/
int FNV64string ( const char *in, uint8_t out[FNV64size] )
{
    FNV64context     ctx;
    int              err;

    if ( ( err = FNV64init (&ctx) ) != fnvSuccess )
        return err;
    if ( ( err = FNV64stringin (&ctx, in) ) != fnvSuccess )
        return err;
    return FNV64result (&ctx, out);
}   /* end FNV64string */

/* FNV64 hash a counted block  (32 bit)
 ******************************************************************/
int FNV64block ( const void *in,
                 long int length,
                 uint8_t out[FNV64size] )
{
    FNV64context     ctx;
    int              err;

    if ( ( err = FNV64init (&ctx) ) != fnvSuccess )
        return err;
    if ( ( err = FNV64blockin (&ctx, in, length) ) != fnvSuccess )
        return err;
    return FNV64result (&ctx, out);
}   /* end FNV64block */


/******************************************************************
 *        Set of init, input, and output functions below          *
 *        to incrementally compute FNV64                          *
 ******************************************************************/

/* initialize context  (32 bit)
 ******************************************************************/
int FNV64init ( FNV64context * const ctx )
{
    if ( !ctx )
        return fnvNull;

    ctx->Hash[0] = FNV64basis0;
    ctx->Hash[1] = FNV64basis1;
    ctx->Hash[2] = FNV64basis2;
    ctx->Hash[3] = FNV64basis3;
    ctx->Computed = FNVinited+FNV64state;
    return fnvSuccess;
}   /* end FNV64init */

/* initialize context  (32 bit)
 ******************************************************************/
int FNV64initBasis ( FNV64context * const ctx,
                     const uint8_t basis[FNV64size] )
{
    if ( !ctx )
        return fnvNull;

#ifdef FNV_BigEndian
    ctx->Hash[0] = basis[1] + ( basis[0]<<8 );
    ctx->Hash[1] = basis[3] + ( basis[2]<<8 );
    ctx->Hash[2] = basis[5] + ( basis[4]<<8 );
    ctx->Hash[3] = basis[7] + ( basis[6]<<8 );
#else
    ctx->Hash[0] = basis[0] + ( basis[1]<<8 );
    ctx->Hash[1] = basis[2] + ( basis[3]<<8 );
    ctx->Hash[2] = basis[4] + ( basis[5]<<8 );
    ctx->Hash[3] = basis[6] + ( basis[7]<<8 );
#endif
    ctx->Computed = FNVinited+FNV64state;
    return fnvSuccess;
}   /* end FNV64initBasis */

/* hash in a counted block  (32 bit)
 ******************************************************************/
int FNV64blockin ( FNV64context * const ctx,
                   const void *vin,
                   long int length )
{
    const uint8_t *in = (const uint8_t*)vin;
    uint32_t   temp[FNV64size/2];
    uint32_t   temp2[2];
    int        i;

    if ( !ctx || !in )
        return fnvNull;

    if ( length < 0 )
        return fnvBadParam;
    switch ( ctx->Computed )
        {
        case FNVinited+FNV64state:
            ctx->Computed = FNVcomputed+FNV64state;
        case FNVcomputed+FNV64state:
            break;
        default:
            return fnvStateError;
        }
    for ( i=0; i<FNV64size/2; ++i )
         temp[i] = ctx->Hash[i];
    for ( ; length > 0; length-- )
        {
        /* temp = FNV64prime * ( temp ^ *in++ ); */
        temp2[1] = temp[3] << FNV64shift;
        temp2[0] = temp[2] << FNV64shift;
        temp[3] = FNV64primeX * ( temp[3] ^ *in++ );
        temp[2] *= FNV64primeX;
        temp[1] = temp[1] * FNV64primeX + temp2[1];
        temp[0] = temp[0] * FNV64primeX + temp2[0];
        temp[2] += temp[3] >> 16;
        temp[3] &= 0xFFFF;
        temp[1] += temp[2] >> 16;
        temp[2] &= 0xFFFF;
        temp[0] += temp[1] >> 16;
        temp[1] &= 0xFFFF;
        }
    for ( i=0; i<FNV64size/2; ++i )
        ctx->Hash[i] = temp[i];
    return fnvSuccess;
}   /* end FNV64blockin */

/* hash in a string  (32 bit)
 ******************************************************************/
int FNV64stringin ( FNV64context * const ctx,
                    const char *in )
{
    uint32_t   temp[FNV64size/2];
    uint32_t   temp2[2];
    int        i;
    uint8_t    ch;

    if ( !ctx || !in )
        return fnvNull;

    switch ( ctx->Computed )
        {
        case FNVinited+FNV64state:
            ctx->Computed = FNVcomputed+FNV64state;
        case FNVcomputed+FNV64state:
            break;
        default:
             return fnvStateError;
         }
    for ( i=0; i<FNV64size/2; ++i )
         temp[i] = ctx->Hash[i];
    while ( ( ch = (uint8_t)*in++ ) != 0)
        {
        /* temp = FNV64prime * ( temp ^ ch ); */
        temp2[1] = temp[3] << FNV64shift;
        temp2[0] = temp[2] << FNV64shift;
        temp[3] = FNV64primeX * ( temp[3] ^ *in++ );
        temp[2] *= FNV64primeX;
        temp[1] = temp[1] * FNV64primeX + temp2[1];
        temp[0] = temp[0] * FNV64primeX + temp2[0];
        temp[2] += temp[3] >> 16;
        temp[3] &= 0xFFFF;
        temp[1] += temp[2] >> 16;
        temp[2] &= 0xFFFF;
        temp[0] += temp[1] >> 16;
        temp[1] &= 0xFFFF;
        }
    for ( i=0; i<FNV64size/2; ++i )
        ctx->Hash[i] = temp[i];
    return fnvSuccess;
}   /* end FNV64stringin */

/* return hash  (32 bit)
 ******************************************************************/
int FNV64result ( FNV64context * const ctx,
                  uint8_t out[FNV64size] )
{
    int    i;

    if ( !ctx || !out )
        return fnvNull;

    if ( ctx->Computed != FNVcomputed+FNV64state )
        return fnvStateError;
    for ( i=0; i<FNV64size/2; ++i )
        {
#ifdef FNV_BigEndian
        out[7-2*i] = ctx->Hash[i];
        out[6-2*i] = ctx->Hash[i] >> 8;
#else
        out[2*i] = ctx->Hash[i];
        out[2*i+1] = ctx->Hash[i] >> 8;
#endif
        ctx -> Hash[i] = 0;
        }
    ctx->Computed = FNVemptied+FNV64state;
    return fnvSuccess;
}   /* end FNV64result */

#endif    /*  FNV_64bitIntegers */
/******************************************************************
 *       END VERSION FOR WHEN YOU ONLY HAVE 32-BIT ARITHMETIC     *
 ******************************************************************/

#endif    /* _FNV64_C_ */
]]></sourcecode>


</section>

<section>
  <name>FNV128 Code</name>

<t>The header and C source for 128-bit FNV-1a returning a byte
vector.</t>


<sourcecode type="C" markers="true" name="FNV128.h">
<![CDATA[
/*************************** FNV128.h *************************/
/**************** See RFC NNNN for details. *******************/
/*
 * Copyright (c) 2016, 2023 IETF Trust and the persons identified
 * as authors of the code.  All rights reserved.
 * See fnv-private.h for terms of use and redistribution.
 */

#ifndef _FNV128_H_
#define _FNV128_H_

/*
 *  Description:
 *      This file provides headers for the 128-bit version of the
 *      FNV-1a non-cryptographic hash algorithm.
 */

#include "FNVconfig.h"

#include <stdint.h>
#define FNV128size (128/8)

/* If you do not have the ISO standard stdint.h header file, then
 * you must typedef the following types:
 *
 *    type              meaning
 *  uint64_t    unsigned 64 bit integer (ifdef FNV_64bitIntegers)
 *  uint32_t    unsigned 32 bit integer
 *  uint16_t    unsigned 16 bit integer
 *  uint8_t     unsigned 8 bit integer (i.e., unsigned char)
 */

#include "FNVErrorCodes.h"

/*
 *  This structure holds context information for an FNV128 hash
 */
#ifdef FNV_64bitIntegers
    /* version if 64 bit integers supported */
typedef struct FNV128context_s {
        int Computed;  /* state */
        uint32_t Hash[FNV128size/4];
} FNV128context;

#else
    /* version if 64 bit integers NOT supported */

typedef struct FNV128context_s {
        int Computed;  /* state */
        uint16_t Hash[FNV128size/2];
} FNV128context;

#endif /* FNV_64bitIntegers */

/*
 *  Function Prototypes
 *    FNV128string: hash a zero terminated string not including
 *                  the terminating zero
 *    FNV128block: FNV128 hash a specified length byte vector
 *    FNV128init: initializes an FNV128 context
 *    FNV128initBasis: initializes an FNV128 context with a
 *                     provided basis
 *    FNV128blockin: hash in a specified length byte vector
 *    FNV128stringin: hash in a zero terminated string not
 *                    including the zero
 *    FNV128result: returns the hash value
 *
 *    Hash is returned as an array of 8-bit unsigned integers
 */

#ifdef __cplusplus
extern "C" {
#endif

/* FNV128 */
extern int FNV128string ( const char *in,
                          uint8_t out[FNV128size] );
extern int FNV128block ( const void *in,
                         long int length,
                         uint8_t out[FNV128size] );
extern int FNV128init ( FNV128context * const );
extern int FNV128initBasis ( FNV128context * const,
                             const uint8_t basis[FNV128size] );
extern int FNV128blockin ( FNV128context * const,
                           const void *in,
                           long int length );
extern int FNV128stringin ( FNV128context * const,
                            const char *in );
extern int FNV128result ( FNV128context * const,
                          uint8_t out[FNV128size] );

#ifdef __cplusplus
}
#endif

#endif /* _FNV128_H_ */
]]></sourcecode>


<sourcecode type="C" markers="true" name="FNV128.c">
<![CDATA[
/***************************** FNV128.c ***************************/
/******************** See RFC NNNN for details ********************/
/* Copyright (c) 2016, 2023 IETF Trust and the persons identified as
 * authors of the code.  All rights
 * See fnv-private.h for terms of use and redistribution.
 */

/* This file implements the FNV (Fowler, Noll, Vo) non-cryptographic
 * hash function FNV-1a for 128-bit hashes.
 */

#ifndef _FNV128_C_
#define _FNV128_C_

#include "fnv-private.h"
#include "FNV128.h"

/* common code for 64 and 32 bit modes */

/* FNV128 hash a null terminated string  (64/32 bit)
 ******************************************************************/
int FNV128string ( const char *in, uint8_t out[FNV128size] )
{
    FNV128context    ctx;
    int              err;

    if ( (err = FNV128init ( &ctx )) != fnvSuccess )
        return err;
    if ( (err = FNV128stringin ( &ctx, in )) != fnvSuccess )
        return err;
    return FNV128result (&ctx, out);
}   /* end FNV128string */

/* FNV128 hash a counted block  (64/32 bit)
 ******************************************************************/
int FNV128block ( const void *in,
                  long int length,
                  uint8_t out[FNV128size] )
{
    FNV128context    ctx;
    int              err;

   if ( (err = FNV128init ( &ctx )) != fnvSuccess )
       return err;
   if ( (err = FNV128blockin ( &ctx, in, length )) != fnvSuccess )
       return err;
    return FNV128result ( &ctx, out );
}   /* end FNV128block */


/******************************************************************
 *        START VERSION FOR WHEN YOU HAVE 64 BIT ARITHMETIC       *
 ******************************************************************/
#ifdef FNV_64bitIntegers

/* 128 bit FNV_prime = 2^88 + 2^8 + 0x3b */
/* 0x00000000 01000000 00000000 0000013B */
#define FNV128primeX 0x013B
#define FNV128shift 24

/* 0x6C62272E 07BB0142 62B82175 6295C58D */
#define FNV128basis0 0x6C62272E
#define FNV128basis1 0x07BB0142
#define FNV128basis2 0x62B82175
#define FNV128basis3 0x6295C58D

/******************************************************************
 *         Set of init, input, and output functions below         *
 *         to incrementally compute FNV128                        *
 ******************************************************************/

/* initialize context  (64 bit)
 ******************************************************************/
int FNV128init ( FNV128context * const ctx )
{
    if ( !ctx )
        return fnvNull;

    ctx->Hash[0] = FNV128basis0;
    ctx->Hash[1] = FNV128basis1;
    ctx->Hash[2] = FNV128basis2;
    ctx->Hash[3] = FNV128basis3;
    ctx->Computed = FNVinited+FNV128state;
    return fnvSuccess;
}   /* end FNV128init */

/* initialize context with a provided basis  (64 bit)
 ******************************************************************/
int FNV128initBasis ( FNV128context * const ctx,
                      const uint8_t basis[FNV128size] )
{
    int       i;
    const uint8_t   *ui8p;
    uint32_t  temp;

    if ( !ctx )
        return fnvNull;

#ifdef FNV_BigEndian
    ui8p = basis;
    for ( i=0; i < FNV128size/4; ++i )
        {
        temp = (*ui8p++)<<8;
        temp = (temp + *ui8p++)<<8;
            temp = (temp + *ui8p++)<<8;
        ctx->Hash[i] = temp +  *ui8p;
        }
#else
    ui8p = basis + ( FNV128size/4 - 1 );
    for ( i=0; i < FNV128size/4; ++i )
        {
            temp = (*ui8p--)<<8;
            temp = (temp + *ui8p--)<<8;
            temp = (temp + *ui8p--)<<8;
        ctx->Hash[i] = temp +  *ui8p;
        }
#endif
    ctx->Computed = FNVinited+FNV128state;
    return fnvSuccess;
}   /* end FNV128initBasis */

/* hash in a counted block  (64 bit)
 ******************************************************************/
int FNV128blockin ( FNV128context * const ctx,
                    const void *vin,
                    long int length )
{
    const uint8_t *in = (const uint8_t*)vin;
    uint64_t    temp[FNV128size/4];
    uint64_t    temp2[2];
    int         i;

    if ( !ctx || !in )
        return fnvNull;

    if ( length < 0 )
        return fnvBadParam;
    switch ( ctx->Computed )
        {
        case FNVinited+FNV128state:
            ctx->Computed = FNVcomputed+FNV128state;
        case FNVcomputed+FNV128state:
            break;
        default:
            return fnvStateError;
        }
    for ( i=0; i<FNV128size/4; ++i )
         temp[i] = ctx->Hash[i];
    for ( ; length > 0; length-- )
        {
        /* temp = FNV128prime * ( temp ^ *in++ ); */
        temp2[1] = temp[3] << FNV128shift;
        temp2[0] = temp[2] << FNV128shift;
        temp[3] = FNV128primeX * ( temp[3] ^ *in++ );
        temp[2] *= FNV128primeX;
        temp[1] = temp[1] * FNV128primeX + temp2[1];
        temp[0] = temp[0] * FNV128primeX + temp2[0];
        temp[2] += temp[3] >> 32;
        temp[3] &= 0xFFFFFFFF;
        temp[1] += temp[2] >> 32;
        temp[2] &= 0xFFFFFFFF;
        temp[0] += temp[1] >> 32;
        temp[1] &= 0xFFFFFFFF;
        }
    for ( i=0; i<FNV128size/4; ++i )
        ctx->Hash[i] = temp[i];
    return fnvSuccess;
}   /* end FNV128input */

/* hash in a string  (64 bit)
 ******************************************************************/
int FNV128stringin ( FNV128context * const ctx, const char *in )
{
    uint64_t   temp[FNV128size/4];
    uint64_t   temp2[2];
    int        i;
    uint8_t    ch;

    if ( !ctx || !in )
        return fnvNull;

    switch ( ctx->Computed )
        {
        case FNVinited+FNV128state:
            ctx->Computed = FNVcomputed+FNV128state;
        case FNVcomputed+FNV128state:
            break;
        default:
             return fnvStateError;
         }
    for ( i=0; i<FNV128size/4; ++i )
         temp[i] = ctx->Hash[i];
    while ( (ch = (uint8_t)*in++) )
        {
        /* temp = FNV128prime * ( temp ^ ch ); */
        temp2[1] = temp[3] << FNV128shift;
        temp2[0] = temp[2] << FNV128shift;
        temp[3] = FNV128primeX * ( temp[3] ^ *in++ );
        temp[2] *= FNV128primeX;
        temp[1] = temp[1] * FNV128primeX + temp2[1];
        temp[0] = temp[0] * FNV128primeX + temp2[0];
        temp[2] += temp[3] >> 32;
        temp[3] &= 0xFFFFFFFF;
        temp[1] += temp[2] >> 32;
        temp[2] &= 0xFFFFFFFF;
        temp[0] += temp[1] >> 32;
        temp[1] &= 0xFFFFFFFF;
        }
    for ( i=0; i<FNV128size/4; ++i )
        ctx->Hash[i] = temp[i];
    return fnvSuccess;
}   /* end FNV128stringin */

/* return hash  (64 bit)
 ******************************************************************/
int FNV128result ( FNV128context * const ctx,
                   uint8_t out[FNV128size] )
{
    int     i;

    if ( !ctx || !out )
        return fnvNull;

    if ( ctx->Computed != FNVcomputed+FNV128state )
        return fnvStateError;
    for ( i=0; i<FNV128size/4; ++i )
        {
#ifdef FNV_BigEndian
        out[15-4*i] = ctx->Hash[i];
        out[14-4*i] = ctx->Hash[i] >> 8;
        out[13-4*i] = ctx->Hash[i] >> 16;
        out[12-4*i] = ctx->Hash[i] >> 24;
#else
        out[4*i] = ctx->Hash[i];
        out[4*i+1] = ctx->Hash[i] >> 8;
        out[4*i+2] = ctx->Hash[i] >> 16;
        out[4*i+3] = ctx->Hash[i] >> 24;
#endif
        ctx -> Hash[i] = 0;
        }
    ctx->Computed = FNVemptied+FNV128state;
    return fnvSuccess;
}   /* end FNV128result */

/*****************************************************************
 *        END VERSION FOR WHEN YOU HAVE 64 BIT ARITHMETIC        *
 *****************************************************************/
#else    /*  FNV_64bitIntegers */
/*****************************************************************
 *     START VERSION FOR WHEN YOU ONLY HAVE 32-BIT ARITHMETIC    *
 *****************************************************************/

/* 128 bit FNV_prime = 2^88 + 2^8 + 0x3b */
/* 0x00000000 01000000 00000000 0000013B */
#define FNV128primeX 0x013B
#define FNV128shift 8

/* 0x6C62272E 07BB0142 62B82175 6295C58D */
uint16_t FNV128basis[FNV128size/2] =
         { 0x6C62, 0x272E, 0x07BB, 0x0142,
           0x62B8, 0x2175, 0x6295, 0xC58D };

/******************************************************************
 *         Set of init, input, and output functions below         *
 *         to incrementally compute FNV128                        *
 ******************************************************************/

/* initialize context  (32 bit)
 ******************************************************************/
int FNV128init ( FNV128context *ctx )
{
    int     i;

    if ( !ctx )
        return fnvNull;

    for ( i=0; i<FNV128size/2; ++i )
        ctx->Hash[i] = FNV128basis[i];
    ctx->Computed = FNVinited+FNV128state;
    return fnvSuccess;
}   /* end FNV128init */

/* initialize context with a provided basis  (32 bit)
 ******************************************************************/
int FNV128initBasis ( FNV128context *ctx,
                      const uint8_t basis[FNV128size] )
{
    int      i;
    const uint8_t  *ui8p;
    uint32_t temp;

    if ( !ctx )
        return fnvNull;

#ifdef FNV_BigEndian
    ui8p = basis;
    for ( i=0; i < FNV128size/2; ++i )
        {
        temp = *ui8p++;
        ctx->Hash[i] = ( temp<<8 ) + (*ui8p++);
        }
#else
    ui8p = basis + (FNV128size/2 - 1);
    for ( i=0; i < FNV128size/2; ++i )
        {
        temp = *ui8p--;
        ctx->Hash[i] = ( temp<<8 ) +  (*ui8p--);
        }
#endif
    ctx->Computed = FNVinited+FNV128state;
    return fnvSuccess;
}   /* end FNV128initBasis */

/* hash in a counted block  (32 bit)
 *****************************************************************/
int FNV128blockin ( FNV128context *ctx,
                    const void *vin,
                    long int length )
{
    const uint8_t *in = (const uint8_t*)vin;
    uint32_t   temp[FNV128size/2];
    uint32_t   temp2[3];
    int        i;

    if ( !ctx || !in )
        return fnvNull;

    if ( length < 0 )
        return fnvBadParam;
    switch ( ctx->Computed )
        {
        case FNVinited+FNV128state:
            ctx->Computed = FNVcomputed+FNV128state;
        case FNVcomputed+FNV128state:
            break;
        default:
            return fnvStateError;
        }
    for ( i=0; i<FNV128size/2; ++i )
         temp[i] = ctx->Hash[i];
    for ( ; length > 0; length-- )
        {
        /* temp = FNV128prime * ( temp ^ *in++ ); */
        temp[7] ^= *in++;
        temp2[2] = temp[7] << FNV128shift;
        temp2[1] = temp[6] << FNV128shift;
        temp2[0] = temp[5] << FNV128shift;
        for ( i=0; i<8; ++i )
            temp[i] *= FNV128primeX;
        temp[2] += temp2[2];
        temp[1] += temp2[1];
        temp[0] += temp2[0];
        for ( i=7; i>0; --i )
            {
            temp[i-1] += temp[i] >> 16;
            temp[i] &= 0xFFFF;
            }
        }
    for ( i=0; i<FNV128size/2; ++i )
        ctx->Hash[i] = temp[i];
    return fnvSuccess;
}   /* end FNV128blockin */

/* hash in a string  (32 bit)
 ******************************************************************/
int FNV128stringin ( FNV128context *ctx,
                     const char *in )
{
    uint32_t   temp[FNV128size/2];
    uint32_t   temp2[3];
    int        i;
    uint8_t    ch;

    if ( !ctx || !in )
        return fnvNull;

    switch ( ctx->Computed )
        {
        case FNVinited+FNV128state:
            ctx->Computed = FNVcomputed+FNV128state;
        case FNVcomputed+FNV128state:
            break;
        default:
             return fnvStateError;
         }
    for ( i=0; i<FNV128size/2; ++i )
         temp[i] = ctx->Hash[i];
    while ( (ch = (uint8_t)*in++) )
        {
        /* temp = FNV128prime * ( temp ^ *in++ ); */
        temp[7] ^= ch;
        temp2[2] = temp[7] << FNV128shift;
        temp2[1] = temp[6] << FNV128shift;
        temp2[0] = temp[5] << FNV128shift;
        for ( i=0; i<8; ++i )
            temp[i] *= FNV128primeX;
        temp[2] += temp2[2];
        temp[1] += temp2[1];
        temp[0] += temp2[0];
        for ( i=7; i>0; --i )
            {
            temp[i-1] += temp[i] >> 16;
            temp[i] &= 0xFFFF;
            }
         }
    for ( i=0; i<FNV128size/2; ++i )
        ctx->Hash[i] = temp[i];
    return fnvSuccess;
}   /* end FNV128stringin */

/* return hash  (32 bit)
 ******************************************************************/
int FNV128result ( FNV128context *ctx,
                   uint8_t out[FNV128size] )
{
    int    i;

    if ( !ctx || !out )
        return fnvNull;

    if ( ctx->Computed != FNVcomputed+FNV128state )
        return fnvStateError;
    for ( i=0; i<FNV128size/2; ++i )
        {
#ifdef FNV_BigEndian
        out[15-2*i] = ctx->Hash[i];
        out[14-2*i] = ctx->Hash[i] >> 8;
#else
        out[2*i] = ctx->Hash[i];
        out[2*i+1] = ctx->Hash[i] >> 8;
#endif
        ctx -> Hash[i] = 0;
        }
    ctx->Computed = FNVemptied+FNV128state;
    return fnvSuccess;
}   /* end FNV128result */

#endif    /*  FNV_64bitIntegers */
/*******************************************************************
 *        END VERSION FOR WHEN YOU ONLY HAVE 32-BIT ARITHMETIC     *
 *******************************************************************/

#endif    /* _FNV128_C_ */
]]></sourcecode>

</section>

<section>
  <name>FNV256 Code</name>

<t>The header and C source for 256-bit FNV-1a returning a byte
vector.</t>

<sourcecode type="C" markers="true" name="FNV256.h">
<![CDATA[
/************************** FNV256.h ************************/
/*************** See RFC NNNN for details. ******************/
/*
 * Copyright (c) 201, 20236 IETF Trust and the persons
 * identified as authors of the code.  All rights reserved.
 * See fnv-private.h for terms of use and redistribution.
 */

#ifndef _FNV256_H_
#define _FNV256_H_

/*
 *  Description:
 *      This file provides headers for the 256-bit version of
 *      the FNV-1a non-cryptographic hash algorithm.
 */

#include "FNVconfig.h"

#include <stdint.h>
#define FNV256size (256/8)

/* If you do not have the ISO standard stdint.h header file, then
 * you must typedef the following types:
 *
 *    type              meaning
 *  uint64_t    unsigned 64 bit integer (ifdef FNV_64bitIntegers)
 *  uint32_t    unsigned 32 bit integer
 *  uint16_t    unsigned 16 bit integer
 *  uint8_t     unsigned 8 bit integer (i.e., unsigned char)
 */

#include "FNVErrorCodes.h"

/*
 *  This structure holds context information for an FNV256 hash
 */
#ifdef FNV_64bitIntegers
    /* version if 64 bit integers supported */
typedef struct FNV256context_s {
        int Computed;  /* state */
        uint32_t Hash[FNV256size/4];
} FNV256context;

#else
    /* version if 64 bit integers NOT supported */

typedef struct FNV256context_s {
        int Computed;  /* state */
        uint16_t Hash[FNV256size/2];
} FNV256context;

#endif /* FNV_64bitIntegers */

/*
 *  Function Prototypes
 *    FNV256string: hash a zero terminated string not including
 *                  the zero
 *    FNV256block: FNV256 hash a specified length byte vector
 *    FNV256init: initializes an FNV256 context
 *    FNV256initBasis:  initializes an FNV256 context with a
 *                     provided basis
 *    FNV256blockin: hash in a specified length byte vector
 *    FNV256stringin: hash in a zero terminated string not
 *                    including the zero
 *    FNV256result: returns the hash value
 *
 *    Hash is returned as an array of 8-bit unsigned integers
 */

#ifdef __cplusplus
extern "C" {
#endif

/* FNV256 */
extern int FNV256string ( const char *in,
                          uint8_t out[FNV256size] );
extern int FNV256block ( const void *in,
                         long int length,
                         uint8_t out[FNV256size] );
extern int FNV256init ( FNV256context *);
extern int FNV256initBasis ( FNV256context * const,
                             const uint8_t basis[FNV256size] );
extern int FNV256blockin ( FNV256context * const,
                           const void *in,
                           long int length );
extern int FNV256stringin ( FNV256context * const,
                            const char *in );
extern int FNV256result ( FNV256context * const,
                          uint8_t out[FNV256size] );

#ifdef __cplusplus
}
#endif

#endif /* _FNV256_H_ */
]]></sourcecode>


<sourcecode type="C" markers="true" name="FNV256.c">
<![CDATA[
/***************************** FNV256.c ***************************/
/******************** See RFC NNNN for details ********************/
/* Copyright (c) 2016, 2023 IETF Trust and the persons identified as
 * authors of the code.  All rights
 * See fnv-private.h for terms of use and redistribution.
 */

/* This file implements the FNV (Fowler, Noll, Vo) non-cryptographic
 * hash function FNV-1a for 256-bit hashes.
 */

#ifndef _FNV256_C_
#define _FNV256_C_

#include "fnv-private.h"
#include "FNV256.h"

/* common code for 64 and 32 bit modes */

/* FNV256 hash a null terminated string  (64/32 bit)
 ******************************************************************/
int FNV256string ( const char *in, uint8_t out[FNV256size] )
{
    FNV256context    ctx;
    int              err;

    if ( (err = FNV256init ( &ctx )) != fnvSuccess )
        return err;
    if ( (err = FNV256stringin ( &ctx, in )) != fnvSuccess )
        return err;
    return FNV256result ( &ctx, out );
}   /* end FNV256string */

/* FNV256 hash a counted block  (64/32 bit)
 ******************************************************************/
int FNV256block ( const void *in,
                  long int length,
                  uint8_t out[FNV256size] )
{
    FNV256context    ctx;
    int              err;

    if ( (err = FNV256init ( &ctx )) != fnvSuccess )
        return err;
    if ( (err = FNV256blockin ( &ctx, in, length)) != fnvSuccess )
        return err;
    return FNV256result ( &ctx, out );
}   /* end FNV256block */


/******************************************************************
 *        START VERSION FOR WHEN YOU HAVE 64 BIT ARITHMETIC       *
 ******************************************************************/
#ifdef FNV_64bitIntegers

/* 256 bit FNV_prime = 2^168 + 2^8 + 0x63 */
/* 0x0000000000000000 0000010000000000
     0000000000000000 0000000000000163 */
#define FNV256primeX 0x0163
#define FNV256shift 8

/* 0xDD268DBCAAC55036 2D98C384C4E576CC
     C8B1536847B6BBB3 1023B4C8CAEE0535 */
uint32_t FNV256basis[FNV256size/4] = {
         0xDD268DBC, 0xAAC55036, 0x2D98C384, 0xC4E576CC,
         0xC8B15368, 0x47B6BBB3, 0x1023B4C8, 0xCAEE0535 };

/******************************************************************
 *         Set of init, input, and output functions below         *
 *         to incrementally compute FNV256                        *
 ******************************************************************/

/* initialize context  (64 bit)
 ******************************************************************/
int FNV256init ( FNV256context *ctx )
{
    int      i;

    if ( !ctx )
        return fnvNull;

    for ( i=0; i<FNV256size/4; ++i )
        ctx->Hash[i] = FNV256basis[i];
    ctx->Computed = FNVinited+FNV256state;
    return fnvSuccess;
}   /* end FNV256init */

/* initialize context with a provided basis  (64 bit)
 ******************************************************************/
int FNV256initBasis ( FNV256context* const ctx,
                      const uint8_t basis[FNV256size] )
{
    int      i;
    const uint8_t  *ui8p;
    uint32_t  temp;

    if ( !ctx )
        return fnvNull;

#ifdef FNV_BigEndian
    ui8p = basis;
    for ( i=0; i < FNV256size/4; ++i )
        {
            temp = (*ui8p++)<<8;
            temp = (temp + *ui8p++)<<8;
            temp = (temp + *ui8p++)<<8;
        ctx->Hash[i] = temp +  *ui8p;
        }
#else
    ui8p = basis + (FNV256size/4 - 1);
    for ( i=0; i < FNV256size/4; ++i )
        {
            temp = (*ui8p--)<<8;
            temp = (temp + *ui8p--)<<8;
            temp = (temp + *ui8p--)<<8;
        ctx->Hash[i] = temp +  *ui8p;
        }
#endif
    ctx->Computed = FNVinited+FNV256state;
    return fnvSuccess;
}   /* end FNV256initBasis */

/* hash in a counted block  (64 bit)
 ******************************************************************/
int FNV256blockin ( FNV256context *ctx,
                    const void *vin,
                    long int length )
{
    const uint8_t *in = (const uint8_t*)vin;
    uint64_t    temp[FNV256size/4];
    uint64_t    temp2[3];
    int         i;

    if ( !ctx || !in )
        return fnvNull;

    if ( length < 0 )
        return fnvBadParam;
    switch ( ctx->Computed )
        {
        case FNVinited+FNV256state:
            ctx->Computed = FNVcomputed+FNV256state;
        case FNVcomputed+FNV256state:
            break;
        default:
            return fnvStateError;
        }
    for ( i=0; i<FNV256size/4; ++i )
         temp[i] = ctx->Hash[i];
    for ( ; length > 0; length-- )
        {
        /* temp = FNV256prime * ( temp ^ *in++ ); */
        temp[7] ^= *in++;
        temp2[2] = temp[7] << FNV256shift;
        temp2[1] = temp[6] << FNV256shift;
        temp2[0] = temp[5] << FNV256shift;
        for ( i=0; i<FNV256size/4; ++i )
            temp[i] *= FNV256primeX;
        temp[2] += temp2[2];
        temp[1] += temp2[1];
        temp[0] += temp2[0];
        for ( i=FNV256size/4-1; i>0; --i )
            {
            temp[i-1] += temp[i] >> 16;
            temp[i] &= 0xFFFF;
            }
       }
    for ( i=0; i<FNV256size/4; ++i )
        ctx->Hash[i] = temp[i];
    return fnvSuccess;
}   /* end FNV256input */

/* hash in a string  (64 bit)
 ******************************************************************/
int FNV256stringin ( FNV256context *ctx, const char *in )
{
    uint64_t   temp[FNV256size/4];
    uint64_t   temp2[3];
    int        i;
    uint8_t    ch;

    if ( !ctx || !in )
        return fnvNull;

    switch ( ctx->Computed )
        {
        case FNVinited+FNV256state:
            ctx->Computed = FNVcomputed+FNV256state;
        case FNVcomputed+FNV256state:
            break;
        default:
             return fnvStateError;
         }
    for ( i=0; i<FNV256size/4; ++i )
         temp[i] = ctx->Hash[i];
    while ( (ch = (uint8_t)*in++) )
        {
        /* temp = FNV256prime * ( temp ^ ch ); */
        temp[7] ^= ch;
        temp2[2] = temp[7] << FNV256shift;
        temp2[1] = temp[6] << FNV256shift;
        temp2[0] = temp[5] << FNV256shift;
        for ( i=0; i<FNV256size/4; ++i )
            temp[i] *= FNV256primeX;
        temp[2] += temp2[2];
        temp[1] += temp2[1];
        temp[0] += temp2[0];
        for ( i=FNV256size/4-1; i>0; --i )
            {
            temp[i-1] += temp[i] >> 16;
            temp[i] &= 0xFFFF;
            }
       }
    for ( i=0; i<FNV256size/4; ++i )
        ctx->Hash[i] = temp[i];
    return fnvSuccess;
}   /* end FNV256stringin */

/* return hash  (64 bit)
 ******************************************************************/
int FNV256result ( FNV256context *ctx, uint8_t out[FNV256size] )
{
    int     i;

    if ( !ctx || !out )
        return fnvNull;

    if ( ctx->Computed != FNVcomputed+FNV256state )
        return fnvStateError;
    for ( i=0; i<FNV256size/4; ++i )
        {
#ifdef FNV_BigEndian
        out[31-4*i] = ctx->Hash[i];
        out[31-4*i] = ctx->Hash[i] >> 8;
        out[31-4*i] = ctx->Hash[i] >> 16;
        out[31-4*i] = ctx->Hash[i] >> 24;
#else
        out[4*i] = ctx->Hash[i];
        out[4*i+1] = ctx->Hash[i] >> 8;
        out[4*i+2] = ctx->Hash[i] >> 16;
        out[4*i+3] = ctx->Hash[i] >> 24;
#endif
        ctx -> Hash[i] = 0;
        }
    ctx->Computed = FNVemptied+FNV256state;
    return fnvSuccess;
}   /* end FNV256result */

/*****************************************************************
 *        END VERSION FOR WHEN YOU HAVE 64 BIT ARITHMETIC        *
 *****************************************************************/
#else    /*  FNV_64bitIntegers */
/*****************************************************************
 *      START VERSION FOR WHEN YOU ONLY HAVE 32-BIT ARITHMETIC   *
 *****************************************************************/

/* version for when you only have 32-bit arithmetic
 *****************************************************************/

/* 256 bit FNV_prime = 2^168 + 2^8 + 0x63 */
/* 0x00000000 00000000 00000100 00000000
     00000000 00000000 00000000 00000163 */
#define FNV256primeX 0x0163
#define FNV256shift 8

/* 0xDD268DBCAAC55036 2D98C384C4E576CC
     C8B1536847B6BBB3 1023B4C8CAEE0535 */
uint16_t FNV256basis[FNV256size/2] = {
         0xDD26, 0x8DBC, 0xAAC5, 0x5036,
         0x2D98, 0xC384, 0xC4E5, 0x76CC,
         0xC8B1, 0x5368, 0x47B6, 0xBBB3,
         0x1023, 0xB4C8, 0xCAEE, 0x0535 };

/*****************************************************************
 *         Set of init, input, and output functions below        *
 *         to incrementally compute FNV256                       *
 *****************************************************************/

/* initialize context  (32 bit)
 *****************************************************************/
int FNV256init ( FNV256context *ctx )
{
    int     i;

    if ( !ctx )
        return fnvNull;

    for ( i=0; i<FNV256size/2; ++i )
        ctx->Hash[i] = FNV256basis[i];
    ctx->Computed = FNVinited+FNV256state;
    return fnvSuccess;
}   /* end FNV256init */

/* initialize context with a provided basis  (32 bit)
 *****************************************************************/
int FNV256initBasis ( FNV256context *ctx,
                      const uint8_t basis[FNV256size] )
{
    int      i;
    const uint8_t  *ui8p;
    uint32_t temp;

    if ( !ctx )
        return fnvNull;

#ifdef FNV_BigEndian
    ui8p = basis;
    for ( i=0; i < FNV256size/2; ++i )
        {
        temp = *ui8p++;
        ctx->Hash[i] = ( temp<<8 ) + (*ui8p++);
        }
#else
    ui8p = basis + FNV256size/2 -1;
    for ( i=0; i < FNV256size/2; ++i )
        {
        temp = *ui8p--;
        ctx->Hash[i] = ( temp<<8 ) +  (*ui8p--);
        }
#endif
    ctx->Computed = FNVinited+FNV256state;
    return fnvSuccess;
}   /* end FNV256initBasis */

/* hash in a counted block  (32 bit)
 *****************************************************************/
int FNV256blockin ( FNV256context *ctx,
                    const void *vin,
                    long int length )
{
    const uint8_t *in = (const uint8_t*)vin;
    uint32_t   temp[FNV256size/2];
    uint32_t   temp2[6];
    int        i;

    if ( !ctx || !in )
        return fnvNull;

    if ( length < 0 )
        return fnvBadParam;
    switch ( ctx->Computed )
        {
        case FNVinited+FNV256state:
            ctx->Computed = FNVcomputed+FNV256state;
        case FNVcomputed+FNV256state:
            break;
        default:
            return fnvStateError;
        }
    for ( i=0; i<FNV256size/2; ++i )
        temp[i] = ctx->Hash[i];
    for ( ; length > 0; length-- )
        {
        /* temp = FNV256prime * ( temp ^ *in++ ); */
        temp[15] ^= *in++;
        for ( i=0; i<6; ++i )
            temp2[i] = temp[10+i] << FNV256shift;
        for ( i=0; i<FNV256size/2; ++i )
            temp[i] *= FNV256primeX;
        for ( i=0; i<6; ++i )
            temp[10+i] += temp2[i];
        for ( i=15; i>0; --i )
            {
            temp[i-1] += temp[i] >> 16;
            temp[i] &= 0xFFFF;
            }
        }
    for ( i=0; i<FNV256size/2; ++i )
        ctx->Hash[i] = temp[i];
    return fnvSuccess;
}   /* end FNV256blockin */

/* hash in a string  (32 bit)
 *****************************************************************/
int FNV256stringin ( FNV256context *ctx, const char *in )
{
    uint32_t   temp[FNV256size/2];
    uint32_t   temp2[6];
    int        i;
    uint8_t    ch;

    if ( !ctx || !in )
        return fnvNull;

    switch ( ctx->Computed )
        {
        case FNVinited+FNV256state:
            ctx->Computed = FNVcomputed+FNV256state;
        case FNVcomputed+FNV256state:
            break;
        default:
             return fnvStateError;
         }
    for ( i=0; i<FNV256size/2; ++i )
         temp[i] = ctx->Hash[i];
    while ( ( ch = (uint8_t)*in++ ) != 0)
        {
        /* temp = FNV256prime * ( temp ^ *in++ ); */
        temp[15] ^= ch;
        for ( i=0; i<6; ++i )
            temp2[i] = temp[10+i] << FNV256shift;
        for ( i=0; i<FNV256size/2; ++i )
            temp[i] *= FNV256primeX;
        for ( i=0; i<6; ++i )
            temp[10+i] += temp2[i];
        for ( i=15; i>0; --i )
            {
            temp[i-1] += temp[i] >> 16;
            temp[i] &= 0xFFFF;
            }
         }
    for ( i=0; i<FNV256size/2; ++i )
        ctx->Hash[i] = temp[i];
    return fnvSuccess;
}   /* end FNV256stringin */

/* return hash  (32 bit)
 *****************************************************************/
int FNV256result ( FNV256context *ctx, uint8_t out[FNV256size] )
{
    int    i;

    if ( !ctx || !out )
        return fnvNull;

    if ( ctx->Computed != FNVcomputed+FNV256state )
        return fnvStateError;
    for ( i=0; i<FNV256size/2; ++i )
        {
#ifdef FNV_BigEndian
        out[31-2*i] = ctx->Hash[i];
        out[30-2*i] = ctx->Hash[i] >> 8;
#else
        out[2*i] = ctx->Hash[i];
        out[2*i+1] = ctx->Hash[i] >> 8;
#endif
        ctx->Hash[i] = 0;
        }
    ctx->Computed = FNVemptied+FNV256state;
    return fnvSuccess;
}   /* end FNV256result */

#endif    /*  FNV_64bitIntegers */
/*****************************************************************
 *        END VERSION FOR WHEN YOU ONLY HAVE 32-BIT ARITHMETIC   *
 *****************************************************************/

#endif    /* _FNV256_C_ */
]]></sourcecode>

</section>

<section>
  <name>FNV512 Code</name>

<t>The header and C source for 512-bit FNV-1a returning a byte
vector.</t>


<sourcecode type="C" markers="true" name="FNV512.h">
<![CDATA[
/************************** FNV512.h ************************/
/*************** See RFC NNNN for details. ******************/
/*
 * Copyright (c) 2016, 2023 IETF Trust and the persons
 * identified as authors of the code.  All rights reserved.
 * See fnv-private.h for terms of use and redistribution.
 */

#ifndef _FNV512_H_
#define _FNV512_H_

/*
 *  Description:
 *      This file provides headers for the 512-bit version of
 *      the FNV-1a non-cryptographic hash algorithm.
 */

#include "FNVconfig.h"

#include <stdint.h>
#define FNV512size (512/8)

/* If you do not have the ISO standard stdint.h header file, then
 * you must typedef the following types:
 *
 *    type              meaning
 *  uint64_t    unsigned 64 bit integer (ifdef FNV_64bitIntegers)
 *  uint32_t    unsigned 32 bit integer
 *  uint16_t    unsigned 16 bit integer
 *  uint8_t     unsigned 8 bit integer (i.e., unsigned char)
 */

#include "FNVErrorCodes.h"

/*
 *  This structure holds context information for an FNV512 hash
 */
#ifdef FNV_64bitIntegers
    /* version if 64 bit integers supported */
typedef struct FNV512context_s {
        int Computed;  /* state */
        uint32_t Hash[FNV512size/4];
} FNV512context;

#else
    /* version if 64 bit integers NOT supported */

typedef struct FNV512context_s {
        int Computed;  /* state */
        uint16_t Hash[FNV512size/2];
} FNV512context;

#endif /* FNV_64bitIntegers */

/*
 *  Function Prototypes
 *    FNV512string: hash a zero terminated string not including
 *                  the terminating zero
 *    FNV512block: FNV512 hash a specified length byte vector
 *    FNV512init: initializes an FNV512 context
 *    FNV512initBasis: initializes an FNV512 context with a
 *                     provided basis
 *    FNV512blockin: hash in a specified length byte vector
 *    FNV512stringin: hash in a zero terminated string not
 *                    including the zero
 *    FNV512result: returns the hash value
 *
 *    Hash is returned as an array of 8-bit unsigned integers
 */

#ifdef __cplusplus
extern "C" {
#endif

/* FNV512 */
extern int FNV512string ( const char *in,
                          uint8_t out[FNV512size] );
extern int FNV512block ( const void *in,
                         long int length,
                         uint8_t out[FNV512size] );
extern int FNV512init ( FNV512context *);
extern int FNV512initBasis ( FNV512context * const,
                             const uint8_t basis[FNV512size] );
extern int FNV512blockin ( FNV512context *,
                           const void *in,
                           long int length );
extern int FNV512stringin ( FNV512context *,
                            const char *in );
extern int FNV512result ( FNV512context *,
                          uint8_t out[FNV512size] );

#ifdef __cplusplus
}
#endif

#endif /* _FNV512_H_ */
]]></sourcecode>


<sourcecode type="C" markers="true" name="FNV512.c">
<![CDATA[
/***************************** FNV512.c ***************************/
/******************** See RFC NNNN for details ********************/
/* Copyright (c) 2016, 2023 IETF Trust and the persons identified as
 * authors of the code.  All rights
 * See fnv-private.h for terms of use and redistribution.
 */

/* This file implements the FNV (Fowler, Noll, Vo) non-cryptographic
 * hash function FNV-1a for 512-bit hashes.
 */

#ifndef _FNV512_C_
#define _FNV512_C_

#include "fnv-private.h"
#include "FNV512.h"

/* common code for 64 and 32 bit modes */

/* FNV512 hash a null terminated string  (64/32 bit)
 ******************************************************************/
int FNV512string ( const char *in, uint8_t out[FNV512size] )
{
    FNV512context    ctx;
    int              err;

    if ( (err = FNV512init ( &ctx )) != fnvSuccess )
        return err;
    if ( (err = FNV512stringin ( &ctx, in )) != fnvSuccess )
        return err;
    return FNV512result ( &ctx, out );
}   /* end FNV512string */

/* FNV512 hash a counted block  (64/32 bit)
 ******************************************************************/
int FNV512block ( const void *in,
                  long int length,
                  uint8_t out[FNV512size] )
{
    FNV512context    ctx;
    int              err;

    if ( (err = FNV512init ( &ctx )) != fnvSuccess )
        return err;
    if ( (err = FNV512blockin ( &ctx, in, length)) != fnvSuccess )
        return err;
    return FNV512result ( &ctx, out );
}   /* end FNV512block */


/******************************************************************
 *        START VERSION FOR WHEN YOU HAVE 64 BIT ARITHMETIC       *
 ******************************************************************/
#ifdef FNV_64bitIntegers

/*
  512 bit FNV_prime = 2^344 + 2^8 + 0x57 =
   0x0000000000000000 0000000000000000
     0000000001000000 0000000000000000
     0000000000000000 0000000000000000
     0000000000000000 0000000000000157 */
#define FNV512primeX 0x0157
#define FNV512shift 24

/* 0xB86DB0B1171F4416 DCA1E50F309990AC
     AC87D059C9000000 0000000000000D21
     E948F68A34C192F6 2EA79BC942DBE7CE
     182036415F56E34B AC982AAC4AFE9FD9 */

uint32_t FNV512basis[FNV512size/4] = {
         0xB86DB0B1, 0x171F4416, 0xDCA1E50F, 0x209990AC,
         0xAC87D059, 0x9C000000, 0x00000000, 0x00000D21,
         0xE948F68A, 0x34C192F6, 0x2EA79BC9, 0x42DBE7CE,
         0x18203641, 0x5F56E34B, 0xAC982AAC, 0x4AFE9FD9 };

/******************************************************************
 *         Set of init, input, and output functions below         *
 *         to incrementally compute FNV512                        *
 ******************************************************************/

/* initialize context  (64 bit)
 ******************************************************************/
int FNV512init ( FNV512context *ctx )
{
    int i;

    if ( !ctx )
        return fnvNull;

    for ( i=0; i<FNV512size/4; ++i )
        ctx->Hash[i] = FNV512basis[i];
    ctx->Computed = FNVinited+FNV512state;
    return fnvSuccess;
}   /* end FNV512init */

/* initialize context with a provided basis  (64 bit)
 ******************************************************************/
int FNV512initBasis ( FNV512context* const ctx,
                      const uint8_t basis[FNV512size] )
{
    int      i;
    const uint8_t  *ui8p;
    uint32_t  temp;

    if ( !ctx )
        return fnvNull;

#ifdef FNV_BigEndian
    ui8p = basis;
    for ( i=0; i < FNV512size/4; ++i )
        {
            temp = (*ui8p++)<<8;
            temp = (temp + *ui8p++)<<8;
            temp = (temp + *ui8p++)<<8;
        ctx->Hash[i] = temp +  *ui8p;
        }
#else
    ui8p = basis + (FNV512size/4 - 1);
    for ( i=0; i < FNV512size/4; ++i )
        {
            temp = (*ui8p--)<<8;
            temp = (temp + *ui8p--)<<8;
            temp = (temp + *ui8p--)<<8;
        ctx->Hash[i] = temp +  *ui8p;
        }
#endif
    ctx->Computed = FNVinited+FNV512state;
    return fnvSuccess;
}   /* end FNV512initBasis */

/* hash in a counted block  (64 bit)
 ******************************************************************/
int FNV512blockin ( FNV512context *ctx,
                    const void *vin,
                    long int length )
{
    int         i;
    const uint8_t *in = (const uint8_t*)vin;
    uint64_t    temp[FNV512size/4];
    uint64_t    temp2[3];

    if ( !ctx || !in )
        return fnvNull;

    switch ( ctx->Computed )
        {
        case FNVinited+FNV512state:
            ctx->Computed = FNVcomputed+FNV128state;
        case FNVcomputed+FNV512state:
            break;
        default:
            return fnvStateError;
        }
    if ( length < 0 )
        return fnvBadParam;
    for ( i=0; i<FNV512size/4; ++i )
         temp[i] = ctx->Hash[i];
    for ( ; length > 0; length-- )
        {
        /* temp = FNV512prime * ( temp ^ *in++ ); */
        temp[7] ^= *in++;
        temp2[2] = temp[7] << FNV512shift;
        temp2[1] = temp[6] << FNV512shift;
        temp2[0] = temp[5] << FNV512shift;
        for ( i=0; i<FNV512size/4; ++i )
            temp[i] *= FNV512primeX;
        temp[2] += temp2[2];
        temp[1] += temp2[1];
        temp[0] += temp2[0];
        for ( i=FNV512size/4-1; i>0; --i )
            {
            temp[i-1] += temp[i] >> 16;
            temp[i] &= 0xFFFF;
            }
       }
    for ( i=0; i<FNV512size/4; ++i )
        ctx->Hash[i] = temp[i];
    return fnvSuccess;
}   /* end FNV512input */

/* hash in a string  (64 bit)
 ******************************************************************/
int FNV512stringin ( FNV512context *ctx, const char *in )
{
    uint64_t   temp[FNV512size/4];
    uint64_t   temp2[2];
    int        i;
    uint8_t    ch;

    if ( !ctx || !in )
        return fnvNull;

    switch ( ctx->Computed )
        {
        case FNVinited+FNV512state:
            ctx->Computed = FNVcomputed+FNV512state;
        case FNVcomputed+FNV512state:
            break;
        default:
             return fnvStateError;
         }
    for ( i=0; i<FNV512size/4; ++i )
         temp[i] = ctx->Hash[i];
    while ( (ch = (uint8_t)*in++) )
        {
        /* temp = FNV512prime * ( temp ^ ch ); */
        temp[7] ^= ch;
        temp2[2] = temp[7] << FNV128shift;
        temp2[1] = temp[6] << FNV128shift;
        temp2[0] = temp[5] << FNV128shift;
        for ( i=0; i<FNV512size/4; ++i )
            temp[i] *= FNV512prime;
        temp[2] += temp2[2];
        temp[1] += temp2[1];
        temp[0] += temp2[0];
        for ( i=FNVsize512/4-1; i>0; --i )
            {
            temp[i-1] += temp[i] >> 16;
            temp[i] &= 0xFFFF;
            }
       }
    for ( i=0; i<FNV512size/4; ++i )
        ctx->Hash[i] = temp[i];
    return fnvSuccess;
}   /* end FNV512stringin */

/* return hash  (64 bit)
 ******************************************************************/
int FNV512result ( FNV512context *ctx, uint8_t out[FNV512size] )
{
    int i;

    if ( !ctx || !out )
        return fnvNull;

    if ( ctx->Computed != FNVcomputed+FNV512state )
        return fnvStateError;
    for ( i=0; i<FNV512size/4; ++i )
        {
#ifdef FNV_BigEndian
        out[15-2*i] = ctx->Hash[i];
        out[14-2*i] = ctx->Hash[i] >> 8;
#else
        out[2*i] = ctx->Hash[i];
        out[2*i+1] = ctx->Hash[i] >> 8;
#endif
        ctx -> Hash[i] = 0;
        }
    ctx->Computed = FNVemptied+FNV512state;
    return fnvSuccess;
}   /* end FNV512result */

/******************************************************************
 *        END VERSION FOR WHEN YOU HAVE 64 BIT ARITHMETIC         *
 ******************************************************************/
#else    /*  FNV_64bitIntegers */
/******************************************************************
 *      START VERSION FOR WHEN YOU ONLY HAVE 32-BIT ARITHMETIC    *
 ******************************************************************/

/* version for when you only have 32-bit arithmetic
 ******************************************************************/

/*
  512 bit FNV_prime = 2^344 + 2^8 + 0x57 =
   0x0000000000000000 0000000000000000
     0000000001000000 0000000000000000
     0000000000000000 0000000000000000
     0000000000000000 0000000000000157 */
#define FNV512primeX 0x0157
#define FNV512shift 8

/* 0xB86DB0B1171F4416 DCA1E50F309990AC
     AC87D059C9000000 0000000000000D21
     E948F68A34C192F6 2EA79BC942DBE7CE
     182036415F56E34B AC982AAC4AFE9FD9 */

uint16_t FNV512basis[FNV512size/2] = {
     0xB86D, 0xB0B1, 0x171F, 0x4416, 0xDCA1, 0xE50F, 0x3099, 0x90AC,
     0xAC87, 0xD059, 0xC900, 0x0000, 0x0000, 0x0000, 0x0000, 0x0D21,
     0xE948, 0xF68A, 0x34C1, 0x92F6, 0x2EA7, 0x9BC9, 0x42DB, 0xE7CE,
     0x1820, 0x3641, 0x5F56, 0xE34B, 0xAC98, 0x2AAC, 0x4AFE, 0x9FD9
     };


/******************************************************************
 *         Set of init, input, and output functions below         *
 *         to incrementally compute FNV512                        *
 ******************************************************************/

/* initialize context  (32 bit)
 ******************************************************************/
int FNV512init ( FNV512context *ctx )
{
    int     i;

    if ( !ctx )
        return fnvNull;

    for ( i=0; i<FNV512size/2; ++i )
        ctx->Hash[i] = FNV512basis[i];
    ctx->Computed = FNVinited+FNV512state;
    return fnvSuccess;
}   /* end FNV512init */

/* initialize context with a provided basis  (32 bit)
 ******************************************************************/
int FNV512initBasis ( FNV512context *ctx,
                      const uint8_t basis[FNV512size] )
{
    int      i;
    const uint8_t  *ui8p;
    uint32_t temp;

    if ( !ctx )
        return fnvNull;

#ifdef FNV_BigEndian
    ui8p = basis;
    for ( i=0; i < FNV512size/2; ++i )
        {
        temp = *ui8p++;
        ctx->Hash[i] = ( temp<<8 ) + (*ui8p++);
        }
#else
    ui8p = basis + ( FNV512size/2 - 1 );
    for ( i=0; i < FNV512size/2; ++i )
        {
        temp = *ui8p--;
        ctx->Hash[i] = ( temp<<8 ) + (*ui8p--);
        }
#endif
    ctx->Computed = FNVinited+FNV512state;
    return fnvSuccess;
}   /* end FNV512initBasis */

/* hash in a counted block  (32 bit)
 ******************************************************************/
int FNV512blockin ( FNV512context *ctx,
                    const void *vin,
                    long int length )
{
    const uint8_t *in = (const uint8_t*)vin;
    uint32_t   temp[FNV512size/2];
    uint32_t   temp2[6];
    int        i;

    if ( !ctx || !in )
        return fnvNull;

    switch ( ctx->Computed )
        {
        case FNVinited+FNV512state:
            ctx->Computed = FNVcomputed+FNV512state;
        case FNVcomputed+FNV512state:
            break;
        default:
            return fnvStateError;
        }
    if ( length < 0 )
        return fnvBadParam;
    for ( i=0; i<FNV512size/2; ++i )
        temp[i] = ctx->Hash[i];
    for ( ; length > 0; length-- )
        {
        /* temp = FNV512prime * ( temp ^ *in++ ); */
        temp[15] ^= *in++;
        for ( i=0; i<6; ++i )
            temp2[i] = temp[10+i] << FNV512shift;
        for ( i=0; i<FNV512size/2; ++i )
            temp[i] *= FNV512primeX;
        for ( i=0; i<6; ++i )
            temp[10+i] += temp2[i];
        for ( i=15; i>0; --i )
            {
            temp[i-1] += temp[i] >> 16;
            temp[i] &= 0xFFFF;
            }
        }
    for ( i=0; i<FNV512size/2; ++i )
        ctx->Hash[i] = temp[i];
    return fnvSuccess;
}   /* end FNV512blockin */

/* hash in a string  (32 bit)
 ******************************************************************/
int FNV512stringin ( FNV512context *ctx, const char *in )
{
    uint32_t   temp[FNV512size/2];
    uint32_t   temp2[6];
    int        i;
    uint8_t    ch;

    if ( !ctx || !in )
        return fnvNull;

    switch ( ctx->Computed )
        {
        case FNVinited+FNV512state:
            ctx->Computed = FNVcomputed+FNV512state;
        case FNVcomputed+FNV512state:
            break;
        default:
             return fnvStateError;
         }
    for ( i=0; i<FNV512size/2; ++i )
         temp[i] = ctx->Hash[i];
    while ( (ch = (uint8_t)*in++) )
        {
        /* temp = FNV512prime * ( temp ^ *in++ ); */
        temp[15] ^= ch;
        for ( i=0; i<6; ++i )
            temp2[i] = temp[10+i] << FNV512shift;
        for ( i=0; i<FNV512size/2; ++i )
            temp[i] *= FNV512primeX;
        for ( i=0; i<6; ++i )
            temp[10+i] += temp2[i];
        for ( i=15; i>0; --i )
            {
            temp[i-1] += temp[i] >> 16;
            temp[i] &= 0xFFFF;
            }
         }
    for ( i=0; i<FNV512size/2; ++i )
        ctx->Hash[i] = temp[i];
    return fnvSuccess;
}   /* end FNV512stringin */

/* return hash  (32 bit)
 ******************************************************************/
int FNV512result ( FNV512context *ctx, unsigned char out[16] )
{
    int    i;

    if ( !ctx || !out )
        return fnvNull;

    if ( ctx->Computed != FNVcomputed+FNV512state )
        return fnvStateError;
    for ( i=0; i<FNV512size/2; ++i )
        {
#ifdef FNV_BigEndian
        out[31-2*i] = ctx->Hash[i];
        out[30-2*i] = ctx->Hash[i] >> 8;
#else
        out[2*i] = ctx->Hash[i];
        out[2*i+1] = ctx->Hash[i] >> 8;
#endif
        ctx->Hash[i] = 0;
        }
    ctx->Computed = FNVemptied+FNV512state;
    return fnvSuccess;
}   /* end FNV512result */

#endif    /*  FNV_64bitIntegers */
/******************************************************************
 *        END VERSION FOR WHEN YOU ONLY HAVE 32-BIT ARITHMETIC    *
 ******************************************************************/

#endif    /* _FNV512_C_ */
]]></sourcecode>

</section>

<section>
  <name>FNV1024 Code</name>

<t>The header and C source for 1024-bit FNV-1a returning a byte
vector.</t>


<sourcecode type="C" markers="true" name="FNV1024.h">
<![CDATA[
/************************* FNV1024.h ************************/
/*************** See RFC NNNN for details. ******************/
/*
 * Copyright (c) 2016, 2023 IETF Trust and the persons
 *  identified as authors of the code.  All rights reserved.
 * See fnv-private.h for terms of use and redistribution.
 */

#ifndef _FNV1024_H_
#define _FNV1024_H_

/*
 *  Description:
 *      This file provides headers for the 1024-bit version of
 *      the FNV-1a non-cryptographic hash algorithm.
 */

#include "FNVconfig.h"

#include <stdint.h>
#define FNV1024size (1024/8)

/* If you do not have the ISO standard stdint.h header file, then
 * you must typedef the following types:
 *
 *    type              meaning
 *  uint64_t    unsigned 64 bit integer (ifdef FNV_64bitIntegers)
 *  uint32_t    unsigned 32 bit integer
 *  uint16_t    unsigned 16 bit integer
 *  uint8_t     unsigned 8 bit integer (i.e., unsigned char)
 */

#include "FNVErrorCodes.h"

/*
 *  This structure holds context information for an FNV1024 hash
 */
#ifdef FNV_64bitIntegers
    /* version if 64 bit integers supported */
typedef struct FNV1024context_s {
        int Computed;  /* state */
        uint32_t Hash[FNV1024size/4];
} FNV1024context;

#else
    /* version if 64 bit integers NOT supported */

typedef struct FNV1024context_s {
        int Computed;  /* state */
        uint16_t Hash[FNV1024size/2];
} FNV1024context;

#endif /* FNV_64bitIntegers */

/*
 *  Function Prototypes
 *    FNV1024string: hash a zero terminated string not including
 *                   the terminating zero
 *    FNV1024block: FNV1024 hash a specified length byte vector
 *    FNV1024init: initializes an FNV1024 context
 *    FNV1024initBasis: initializes an FNV1024 context with a
 *                      provided basis
 *    FNV1024blockin: hash in a specified length byte vector
 *    FNV1024stringin: hash in a zero terminated string not
 *                     including the zero
 *    FNV1024result: returns the hash value
 *
 *    Hash is returned as an array of 8-bit unsigned integers
 */

#ifdef __cplusplus
extern "C" {
#endif

/* FNV1024 */
extern int FNV1024string ( const char *in,
                           unsigned char out[FNV1024size] );
extern int FNV1024block ( const void *in,
                          long int length,
                          unsigned char out[FNV1024size] );
extern int FNV1024init ( FNV1024context *);
extern int FNV1024initBasis ( FNV1024context * const,
                              const uint8_t basis[FNV1024size] );
extern int FNV1024blockin ( FNV1024context *,
                            const void *in,
                            long int length );
extern int FNV1024stringin ( FNV1024context *,
                            const char *in );
extern int FNV1024result ( FNV1024context *,
                           unsigned char out[FNV1024size] );

#ifdef __cplusplus
}
#endif

#endif /* _FNV1024_H_ */
]]></sourcecode>


<sourcecode type="C" markers="true" name="FNV1024.c">
<![CDATA[
/***************************** FNV1024.c **************************/
/******************** See RFC NNNN for details ********************/
/* Copyright (c) 2016, 2023 IETF Trust and the persons identified as
 * authors of the code.  All rights
 * See fnv-private.h for terms of use and redistribution.
 */

/* This file implements the FNV (Fowler, Noll, Vo) non-cryptographic
 * hash function FNV-1a for 1024-bit hashes.
 */

#ifndef _FNV1024_C_
#define _FNV1024_C_

#include "fnv-private.h"
#include "FNV1024.h"

/* common code for 64 and 32 bit modes */

/* FNV1024 hash a null terminated string  (64/32 bit)
 ******************************************************************/
int FNV1024string ( const char *in, uint8_t out[FNV1024size] )
{
    FNV1024context    ctx;
    int              err;

    if ( (err = FNV1024init ( &ctx )) != fnvSuccess)
        return err;
    if ( (err = FNV1024stringin ( &ctx, in )) != fnvSuccess)
        return err;
    return FNV1024result ( &ctx, out );
}   /* end FNV1024string */

/* FNV1024 hash a counted block  (64/32 bit)
 ******************************************************************/
int FNV1024block ( const void *in,
                  long int length,
                  uint8_t out[FNV1024size] )
{
    FNV1024context    ctx;
    int              err;

    if ( (err = FNV1024init ( &ctx )) != fnvSuccess)
        return err;
    if ( (err = FNV1024blockin ( &ctx, in, length)) != fnvSuccess)
        return err;
    return FNV1024result ( &ctx, out );
}   /* end FNV1024block */


/******************************************************************
 *        START VERSION FOR WHEN YOU HAVE 64 BIT ARITHMETIC       *
 ******************************************************************/
#ifdef FNV_64bitIntegers

/*
 1024 bit FNV_prime = 2^680 + 2^8 + 0x8d =
   0x0000000000000000 0000010000000000
     0000000000000000 0000000000000000
     0000000000000000 0000000000000000
     0000000000000000 0000000000000000
     0000000000000000 0000000000000000
     0000000000000000 000000000000018D */
#define FNV1024primeX 0x018D
#define FNV1024shift 24

/* 0x0000000000000000 005F7A76758ECC4D
     32E56D5A591028B7 4B29FC4223FDADA1
     6C3BF34EDA3674DA 9A21D90000000000
     0000000000000000 0000000000000000
     0000000000000000 0000000000000000
     0000000000000000 000000000004C6D7
     EB6E73802734510A 555F256CC005AE55
     6BDE8CC9C6A93B21 AFF4B16C71EE90B3 */

uint32_t FNV1024basis[FNV1024size/4] = {
  0x00000000, 0x00000000, 0x005F7A76, 0x758ECC4D,
  0x32E56D5A, 0x591028B7, 0x4B29FC42, 0x23FDADA1,
  0x6C3BF34E, 0xDA3674DA, 0x9A21D900, 0x00000000,
  0x00000000, 0x00000000, 0x00000000, 0x00000000,
  0x00000000, 0x00000000, 0x00000000, 0x00000000,
  0x00000000, 0x00000000, 0x00000000, 0x0004C6D7,
  0xEB6E7380, 0x2734510A, 0x555F256C, 0xC005AE55,
  0x6BDE8CC9, 0xC6A93B21, 0xAFF4B16C, 0x71EE90B3
 };

/******************************************************************
 *         Set of init, input, and output functions below         *
 *         to incrementally compute FNV1024                       *
 ******************************************************************/

/* initialize context  (64 bit)
 ******************************************************************/
int FNV1024init ( FNV1024context *ctx )
{
    int i;

    if ( !ctx )
        return fnvNull;

    for ( i=0; i<FNV1024size/4; ++i )
        ctx->Hash[i] = FNV1024basis[i];
    ctx->Computed = FNVinited+FNV1024state;
    return fnvSuccess;
}   /* end FNV1024init */

/* initialize context with a provided basis  (64 bit)
 ******************************************************************/
int FNV1024initBasis ( FNV1024context* const ctx,
                       const uint8_t basis[FNV1024size] )
{
    int            i;
    const uint8_t  *ui8p;
    uint32_t       temp;

    if ( !ctx )
        return fnvNull;

#ifdef FNV_BigEndian
    ui8p = basis;
    for ( i=0; i < FNV1024size/4; ++i )
        {
            temp = (*ui8p++)<<8;
            temp = (temp + *ui8p++)<<8;
            temp = (temp + *ui8p++)<<8;
        ctx->Hash[i] = temp +  *ui8p;
        }
#else
    ui8p = basis + (FNV1024size/4 - 1);
    for ( i=0; i < FNV1024size/4; ++i )
        {
            temp = (*ui8p--)<<8;
            temp = (temp + *ui8p--)<<8;
            temp = (temp + *ui8p--)<<8;
        ctx->Hash[i] = temp +  *ui8p;
        }
#endif
    ctx->Computed = FNVinited+FNV1024state;
    return fnvSuccess;
}   /* end FNV1024initBasis */

/* hash in a counted block  (64 bit)
 ******************************************************************/
int FNV1024blockin ( FNV1024context *ctx,
                    const void *vin,
                    long int length )
{
    int         i;
    const uint8_t *in = (const uint8_t*)vin;
    uint64_t    temp[FNV1024size/4];
    uint64_t    temp2[3];

    if ( !ctx || !in )
        return fnvNull;

    switch ( ctx->Computed )
        {
        case FNVinited+FNV1024state:
            ctx->Computed = FNVcomputed+FNV128state;
        case FNVcomputed+FNV1024state:
            break;
        default:
            return fnvStateError;
        }
    if ( length < 0 )
        return fnvBadParam;
    for ( i=0; i<FNV1024size/4; ++i )
         temp[i] = ctx->Hash[i];
    for ( ; length > 0; length-- )
        {
        /* temp = FNV1024prime * ( temp ^ *in++ ); */
        temp[7] ^= *in++;
        temp2[2] = temp[7] << FNV1024shift;
        temp2[1] = temp[6] << FNV1024shift;
        temp2[0] = temp[5] << FNV1024shift;
        for ( i=0; i<FNV1024size/4; ++i )
            temp[i] *= FNV1024primeX;
        temp[2] += temp2[2];
        temp[1] += temp2[1];
        temp[0] += temp2[0];
        for ( i=FNV1024size/4-1; i>0; --i )
            {
            temp[i-1] += temp[i] >> 16;
            temp[i] &= 0xFFFF;
            }
       }
    for ( i=0; i<FNV1024size/4; ++i )
        ctx->Hash[i] = temp[i];
    return fnvSuccess;
}   /* end FNV1024input */

/* hash in a string  (64 bit)
 ******************************************************************/
int FNV1024stringin ( FNV1024context *ctx, const char *in )
{
    uint64_t   temp[FNV1024size/4];
    uint64_t   temp2[2];
    int        i;
    uint8_t    ch;

    if ( !ctx || !in )
        return fnvNull;

    switch ( ctx->Computed )
        {
        case FNVinited+FNV1024state:
            ctx->Computed = FNVcomputed+FNV1024state;
        case FNVcomputed+FNV1024state:
            break;
        default:
             return fnvStateError;
         }
    for ( i=0; i<FNV1024size/4; ++i )
         temp[i] = ctx->Hash[i];
    while ( (ch = (uint8_t)*in++) )
        {
        /* temp = FNV1024prime * ( temp ^ ch ); */
        temp[7] ^= ch;
        temp2[2] = temp[7] << FNV128shift;
        temp2[1] = temp[6] << FNV128shift;
        temp2[0] = temp[5] << FNV128shift;
        for ( i=0; i<FNV1024size/4; ++i )
            temp[i] *= FNV1024prime;
        temp[2] += temp2[2];
        temp[1] += temp2[1];
        temp[0] += temp2[0];
        for ( i=FNVsize1024/4-1; i>0; --i )
            {
            temp[i-1] += temp[i] >> 16;
            temp[i] &= 0xFFFF;
            }
       }
    for ( i=0; i<FNV1024size/4; ++i )
        ctx->Hash[i] = temp[i];
    return fnvSuccess;
}   /* end FNV1024stringin */

/* return hash  (64 bit)
 ******************************************************************/
int FNV1024result ( FNV1024context *ctx, uint8_t out[FNV1024size] )
{
    int i;

    if ( !ctx || !out )
        return fnvNull;

    if ( ctx->Computed != FNVcomputed+FNV1024state )
        return fnvStateError;
    for ( i=0; i<FNV1024size/4; ++i )
        {
#ifdef FNV_BigEndian
        out[15-2*i] = ctx->Hash[i];
        out[14-2*i] = ctx->Hash[i] >> 8;
#else
        out[2*i] = ctx->Hash[i];
        out[2*i+1] = ctx->Hash[i] >> 8;
#endif
        ctx -> Hash[i] = 0;
        }
    ctx->Computed = FNVemptied+FNV1024state;
    return fnvSuccess;
}   /* end FNV1024result */

/******************************************************************
 *        END VERSION FOR WHEN YOU HAVE 64 BIT ARITHMETIC         *
 ******************************************************************/
#else    /*  FNV_64bitIntegers */
/******************************************************************
 *      START VERSION FOR WHEN YOU ONLY HAVE 32-BIT ARITHMETIC    *
 ******************************************************************/

/* version for when you only have 32-bit arithmetic
 ******************************************************************/

/*
 1024 bit FNV_prime = 2^680 + 2^8 + 0x8d =
   0x0000000000000000 0000010000000000
     0000000000000000 0000000000000000
     0000000000000000 0000000000000000
     0000000000000000 0000000000000000
     0000000000000000 0000000000000000
     0000000000000000 000000000000018D */
#define FNV1024primeX 0x018D
#define FNV1024shift 8

/* 0x0000000000000000 005F7A76758ECC4D
     32E56D5A591028B7 4B29FC4223FDADA1
     6C3BF34EDA3674DA 9A21D90000000000
     0000000000000000 0000000000000000
     0000000000000000 0000000000000000
     0000000000000000 000000000004C6D7
     EB6E73802734510A 555F256CC005AE55
     6BDE8CC9C6A93B21 AFF4B16C71EE90B3 */

uint16_t FNV1024basis[FNV1024size/2] = {
  0x0000, 0x0000, 0x0000, 0x0000, 0x005F, 0x7A76, 0x758E, 0xCC4D,
  0x32E5, 0x6D5A, 0x5910, 0x28B7, 0x4B29, 0xFC42, 0x23FD, 0xADA1,
  0x6C3B, 0xF34E, 0xDA36, 0x74DA, 0x9A21, 0xD900, 0x0000, 0x0000,
  0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000,
  0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000,
  0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0000, 0x0004, 0xC6D7,
  0xEB6E, 0x7380, 0x2734, 0x510A, 0x555F, 0x256C, 0xC005, 0xAE55,
  0x6BDE, 0x8CC9, 0xC6A9, 0x3B21, 0xAFF4, 0xB16C, 0x71EE, 0x90B3
};


/******************************************************************
 *         Set of init, input, and output functions below         *
 *         to incrementally compute FNV1024                       *
 ******************************************************************/

/* initialize context  (32 bit)
 ******************************************************************/
int FNV1024init ( FNV1024context *ctx )
{
    int     i;

    if ( !ctx )
        return fnvNull;

    for ( i=0; i<FNV1024size/2; ++i )
        ctx->Hash[i] = FNV1024basis[i];
    ctx->Computed = FNVinited+FNV1024state;
    return fnvSuccess;
}   /* end FNV1024init */

/* initialize context with a provided basis  (32 bit)
 ******************************************************************/
int FNV1024initBasis ( FNV1024context *ctx,
                       const uint8_t basis[FNV1024size] )
{
    int      i;
    const uint8_t  *ui8p;
    uint32_t temp;

    if ( !ctx )
        return fnvNull;

#ifdef FNV_BigEndian
    ui8p = basis;
    for ( i=0; i < FNV1024size/2; ++i )
        {
        temp = *ui8p++;
        ctx->Hash[i] = ( temp<<8 ) + (*ui8p++);
        }
#else
    ui8p = basis + ( FNV1024size/2 - 1 );
    for ( i=0; i < FNV1024size/2; ++i )
        {
        temp = *ui8p--;
        ctx->Hash[i] = ( temp<<8 ) + (*ui8p--);
        }
#endif
    ctx->Computed = FNVinited+FNV1024state;
    return fnvSuccess;
}   /* end FNV1024initBasis */

/* hash in a counted block  (32 bit)
 ******************************************************************/
int FNV1024blockin ( FNV1024context *ctx,
                    const void *vin,
                    long int length )
{
    const uint8_t *in = (const uint8_t*)vin;
    uint32_t   temp[FNV1024size/2];
    uint32_t   temp2[6];
    int        i;

    if ( !ctx || !in )
        return fnvNull;

    switch ( ctx->Computed )
        {
        case FNVinited+FNV1024state:
            ctx->Computed = FNVcomputed+FNV1024state;
        case FNVcomputed+FNV1024state:
            break;
        default:
            return fnvStateError;
        }
    if ( length < 0 )
        return fnvBadParam;
    for ( i=0; i<FNV1024size/2; ++i )
        temp[i] = ctx->Hash[i];
    for ( ; length > 0; length-- )
        {
        /* temp = FNV1024prime * ( temp ^ *in++ ); */
        temp[15] ^= *in++;
        for ( i=0; i<6; ++i )
            temp2[i] = temp[10+i] << FNV1024shift;
        for ( i=0; i<FNV1024size/2; ++i )
            temp[i] *= FNV1024primeX;
        for ( i=0; i<6; ++i )
            temp[10+i] += temp2[i];
        for ( i=15; i>0; --i )
            {
            temp[i-1] += temp[i] >> 16;
            temp[i] &= 0xFFFF;
            }
        }
    for ( i=0; i<FNV1024size/2; ++i )
        ctx->Hash[i] = temp[i];
    return fnvSuccess;
}   /* end FNV1024blockin */

/* hash in a string  (32 bit)
 ******************************************************************/
int FNV1024stringin ( FNV1024context *ctx, const char *in )
{
    uint32_t   temp[FNV1024size/2];
    uint32_t   temp2[6];
    int        i;
    uint8_t    ch;

    if ( !ctx || !in )
        return fnvNull;

    switch ( ctx->Computed )
        {
        case FNVinited+FNV1024state:
            ctx->Computed = FNVcomputed+FNV1024state;
        case FNVcomputed+FNV1024state:
            break;
        default:
             return fnvStateError;
         }
    for ( i=0; i<FNV1024size/2; ++i )
         temp[i] = ctx->Hash[i];
    while ( (ch = (uint8_t)*in++) )
        {
        /* temp = FNV1024prime * ( temp ^ *in++ ); */
        temp[15] ^= ch;
        for ( i=0; i<6; ++i )
            temp2[i] = temp[10+i] << FNV1024shift;
        for ( i=0; i<FNV1024size/2; ++i )
            temp[i] *= FNV1024primeX;
        for ( i=0; i<6; ++i )
            temp[10+i] += temp2[i];
        for ( i=15; i>0; --i )
            {
            temp[i-1] += temp[i] >> 16;
            temp[i] &= 0xFFFF;
            }
         }
    for ( i=0; i<FNV1024size/2; ++i )
        ctx->Hash[i] = temp[i];
    return fnvSuccess;
}   /* end FNV1024stringin */

/* return hash  (32 bit)
 ******************************************************************/
int FNV1024result ( FNV1024context *ctx, unsigned char out[16] )
{
    int    i;

    if ( !ctx || !out )
        return fnvNull;

    if ( ctx->Computed != FNVcomputed+FNV1024state )
        return fnvStateError;
    for ( i=0; i<FNV1024size/2; ++i )
        {
#ifdef FNV_BigEndian
        out[31-2*i] = ctx->Hash[i];
        out[30-2*i] = ctx->Hash[i] >> 8;
#else
        out[2*i] = ctx->Hash[i];
        out[2*i+1] = ctx->Hash[i] >> 8;
#endif
        ctx->Hash[i] = 0;
        }
    ctx->Computed = FNVemptied+FNV1024state;
    return fnvSuccess;
}   /* end FNV1024result */

#endif    /*  FNV_64bitIntegers */
/******************************************************************
 *        END VERSION FOR WHEN YOU ONLY HAVE 32-BIT ARITHMETIC    *
 ******************************************************************/

#endif    /* _FNV1024_C_ */
]]></sourcecode>

</section>

</section>

<section>
  <name>FNV Test Code</name>

<t>Here is a test driver:</t>


<sourcecode type="C" markers="true" name="main.c">
<![CDATA[
FNV/main.c
]]></sourcecode>

</section>

</section>

<section>  <!-- 7. -->
  <name>Security Considerations</name>

<t>This document is intended to provide convenient open source access
by the Internet community to the FNV non-cryptographic hash. No
assertion of suitability for cryptographic applications is made for
the FNV hash algorithms.</t>

<section>
  <name>Why is FNV Non-Cryptographic?</name>

<t>A full discussion of cryptographic hash requirements and strength
is beyond the scope of this document. However, here are three
characteristics of FNV that would generally be considered to make it
non-cryptographic:</t>

<ol>
  <li>Sticky State - A cryptographic hash should not have a state in
  which it can stick for a plausible input pattern. But, in the very
  unlikely event that the FNV hash variable becomes zero and the input
  is a sequence of zeros, the hash variable will remain at zero until
  there is a non-zero input byte and the final hash value will be
  unaffected by the length of that sequence of zero input bytes. Of
  course, for the common case of fixed length input, this would
  usually not be significant because the number of non-zero bytes
  would vary inversely with the number of zero bytes and for some
  types of input, runs of zeros do not occur. Furthermore, the use of
  a different offset_basis or the inclusion of even a little
  unpredictable input may be sufficient to stop an adversary from
  inducing a zero hash variable.</li>
  
  <li>Diffusion - Every output bit of a cryptographic hash should be
  an equally complex function of every input bit. But it is easy to
  see that the least significant bit of a direct FNV hash is the XOR
  of the least significant bits of every input byte and does not
  depend on any other input bit. While more complex, the second
  through seventh least significant bits of an FNV hash have a similar
  weakness; only the top bit of the bottom byte of output, and higher
  order bits, depend on all input bits. If these properties are
  considered a problem, they can be easily fixed by XOR folding (see
  Section 3).</li>

  <li>Work Factor - Depending on intended use, it is frequently
  desirable that a hash function should be computationally expensive
  for general purpose and graphics processors since these may be
  profusely available through elastic cloud services or botnets.  This
  is to slow down testing of possible inputs if the output is
  known. But FNV is designed to be inexpensive on a general-purpose
  processor. (See Appendix A.)</li>
</ol>

<t>Nevertheless, none of the above have proven to be a problem in
actual practice for the many applications of FNV.</t>

</section>

<section>
  <name>Inducing Collisions</name>

<t>While use of a cryptographic hash should be considered when active
adversaries are a factor, the following attack can be made much more
difficult with very minor changes in the use of FNV.</t>

<t>If FNV is being used in a known way for hash tables in a network
server or the like, for example some part of a web server, an
adversary could send requests calculated to cause hash table
collisions and induce substantial processing delays. As mentioned in
Section 2.2, use of an offset_basis not knownable by the adversary
will substantially eliminate this problem.</t>

</section>

</section>

<section>  <!-- 8. -->
  <name>IANA Considerations</name>

<t>This document requires no IANA Actions.</t>

</section>  <!-- 8. -->
        
</middle>


<!-- ____________________BACK_MATTER____________________ -->
<back>

<references>
  <name>Normative References</name>

<xi:include
    href="https://www.rfc-editor.org/refs/bibxml/reference.RFC.0020.xml"/>

</references>
 
<references> 
  <name>Informative References</name>

<reference anchor="BFDseq"
           target="draft-ietf-bfd-secure-sequence-numbers-09.txt">
  <front>
    <title>Secure BFD Sequence Numbers</title>
    <author fullname="Mahesh Jethanandani" initials="M."
            surname="Jethanandani">
      <organization>Kloud Services</organization>
      <address>
        <email>mjethanandani@gmail.com</email>
      </address>
    </author>
    <author fullname="Sonal Agarwal" initials="S."
            surname="Agarwal">
      <organization>Cisco Systems, Inc</organization>
      <address>
        <postal>
          <street>170 West Tasman Drive</street>
          <city>San Jose</city>
          <region>California</region>
          <code>95070</code>
          <country>USA</country>
        </postal>
        <email>agarwaso@cisco.com</email>
        <uri>www.cisco.com</uri>
      </address>
    </author>
    <author fullname="Ashesh Mishra" initials="A."
            surname="Mishra">
      <organization>O3b Networks</organization>
      <address>
        <email>mishra.ashesh@gmail.com</email>
      </address>
    </author>
    <author fullname="Ankur Saxena" initials="A."
            surname="Saxena">
      <organization>Ciena Corporation</organization>
      <address>
        <postal>
          <street>3939 North First Street</street>
          <city>San Jose</city>
          <region>California</region>
          <code>95134</code>
          <country>USA</country>
        </postal>
        <email>ankurpsaxena@gmail.com</email>
      </address>
    </author>
    <author fullname="Alan DeKok" initials="A."
            surname="DeKok">
      <organization>Network RADIUS SARL</organization>
      <address>
        <postal>
          <street>100 Centrepointe Drive #200</street>
          <city>Ottawa</city>
          <region>Ontario</region>
          <code>K2G 6B1</code>
          <country>Canada</country>
        </postal>
        <email>aland@freeradius.org</email>
      </address>
    </author>
    <date year="2022" month="March" day="22"/>
  </front>
</reference>

<reference anchor="FNV"
           target="http://www.isthe.com/chongo/tech/comp/fnv/index.html">
  <front>
    <title>FNV website</title>
    <author>
      <organization>Fowler-Noll-Vo</organization>
    </author>
  </front>
</reference>

<reference anchor="IEEE" target="http:www.ieee.org">
  <front>
    <title>IEEE website</title>
    <author>
      <organization>Institute for Electrical and Electronics
      Engineers</organization>
    </author>
  </front>
</reference>
  
<reference anchor="IEEE8021Qbp">
  <front>
    <title>Media Access Control (MAC) Bridges and Virtual Bridged
    Local Area Networks - Equal Cost Multiple Path (ECMP)</title>
    <author surname="IEEE 802.1"/>
    <date year="2014" month="April" day="7"/>
  </front>
  <seriesInfo name="IEEE Std" value="802.1Qbp-2014"/>
</reference>

<reference anchor="IPv6flow"
           target="https://researchspace.auckland.ac.nz/bitstream/handle/2292/13240/flowhashRep.pdf">
  <front>
    <title>Comparing Hash Function Algorithms for the IPv6 Flow
    Label</title>
    <author fullname="Lewis Anderson" initials="L."
            surname="Anderson">
      <organization>The University of Auckland</organization>
      <address>
        <email>land062@auckland.ac.nz</email>
      </address>
    </author>
    <author fullname="Nevil Brownlee" initials="N."
            surname="Brownlee">
      <organization>The University of Auckland</organization>
      <address>
        <email>n.brownlee@auckland.ac.nz</email>
      </address>
    </author>
    <author fullname="Brian E. Carpenter" initials="B."
            surname="Carpenter">
      <organization>The University of Auckland, Department of Computer
      Science</organization>
      <address>
        <email>brain@cs.auckland.ac.nz</email>
      </address>
    </author>
    <date year="2012" month="March"/>
  </front>
  <seriesInfo name="University of Auckland Department of Computer
    Science Technical Report" value="2012-002"/>
  <seriesInfo name="ISSN" value="1173-3500"/>
</reference>


<xi:include
    href="https://www.rfc-editor.org/refs/bibxml/reference.RFC.3174.xml"/>
<xi:include
    href="https://www.rfc-editor.org/refs/bibxml/reference.RFC.6194.xml"/>
<xi:include
    href="https://www.rfc-editor.org/refs/bibxml/reference.RFC.6234.xml"/>
<xi:include
    href="https://www.rfc-editor.org/refs/bibxml/reference.RFC.6437.xml"/>
<xi:include
    href="https://www.rfc-editor.org/refs/bibxml/reference.RFC.7357.xml"/>
<xi:include
    href="https://www.rfc-editor.org/refs/bibxml/reference.RFC.7873.xml"/>
<xi:include
    href="https://www.rfc-editor.org/refs/bibxml/reference.RFC.8200.xml"/>

</references>

<section>  <!-- Appendix A -->
  <name>Work Comparison with SHA-1</name>

<t>This section provides a simplistic rough comparison of the level of
effort required per input byte to compute FNV-1a and SHA-1
<xref target="RFC3174"/>.</t>

<t>Ignoring transfer of control and conditional tests and equating all
logical and arithmetic operations, FNV requires 2 operations per byte,
an XOR and a multiply.</t>

<t>SHA-1 is a relatively weak cryptographic hash producing a 160-bit
hash. It has been partially broken <xref target="RFC6194"/>. It is
actually designed to accept a bit vector input although almost all
computer uses apply it to an integer number of bytes. It processes
blocks of 512 bits (64 bytes) and we estimate the effort involved in
SHA-1 processing a full block. Ignoring SHA-1 initial set up, transfer
of control, and conditional tests, but counting all logical and
arithmetic operations, including counting indexing as an addition,
SHA-1 requires 1,744 operations per 64 bytes block or 27.25 operations
per byte. So by this rough measure, it is a little over 13 times the
effort of FNV for large amounts of data. However, FNV is commonly used
for small inputs. Using the above method, for inputs of N bytes, where
N is &lt;= 55 so SHA-1 will take one block (SHA-1 includes padding and
an 8-byte length at the end of the data in the last block), the ratio
of the effort for SHA-1 to the effort for FNV will be 872/N. For
example, with an 8 byte input, SHA-1 will take 109 times as much
effort as FNV.</t>

<t>Stronger cryptographic functions than SHA-1 generally have an even
higher work factor.</t>

</section> <!-- Appendix A -->

<section>  <!-- Appendix B -->
  <name>Previous IETF Reference to FNV</name>

<t>FNV-1a was referenced in draft-ietf-tls-cached-info-08.txt that has
since expired. It was later decided that it would be better to use a
cryptographic hash for that application.</t>

<t>Below is the Java code for FNV64 from that TLS draft included by the
kind permission of the author:</t>

<sourcecode type="java" markers="true">
 /*
 * Java code sample, implementing 64 bit FNV-1a
 * By Stefan Santesson
 */

import java.math.BigInteger;

public class FNV {

   static public BigInteger getFNV1aToByte(byte[] inp) {

       BigInteger m = new BigInteger("2").pow(64);
       BigInteger fnvPrime = new BigInteger("1099511628211");
       BigInteger fnvOffsetBasis =
               new BigInteger("14695981039346656037");

       BigInteger digest = fnvOffsetBasis;

       for (byte b : inp) {
           digest = digest.xor(BigInteger.valueOf((int) b &amp; 255));
           digest = digest.multiply(fnvPrime).mod(m);
       }
       return digest;

   }
}
</sourcecode>

</section>

<section>  <!-- Appendix C -->
  <name>A Few Test Vectors</name>

<t>Below are a few test vectors in the form of ASCII strings and their
FNV32 and FNV64 hashes using the FNV-1a algorithm.</t>

<table>
  <name>Strings without null (zero byte) termination</name>
  <thead>
<tr><th>String</th><th align="center">FNV32</th><th
align="center">FNV64</th></tr>
  </thead>
  <tbody>
<tr><td>""</td><td>0x811c9dc5</td><td>0xcbf29ce4 84222325</td></tr>
<tr><td>"a"</td><td>0xe40c292c</td><td>0xaf63dc4c 8601ec8c</td></tr>
<tr><td>"foobar"</td><td>0xbf9cf968</td><td>0x85944171f 73967e8</td></tr>
  </tbody>
</table>

<t></t>

<table>
  <name>Strings including null (zero byte) termination</name>
  <thead>
<tr><th>String</th><th align="center">FNV32</th><th
align="center">FNV64</th></tr>
  </thead>
  <tbody>
<tr><td>""</td><td>0x050c5d1f</td><td>0xaf63bd4c 8601b7df</td></tr>
<tr><td>"a"</td><td>0x2b24d044</td><td>0x089be207 b544f1e4</td></tr>
<tr><td>"foobar"</td><td>0x0c1c9eb8</td><td>0x34531ca7 168b8f38</td></tr>
  </tbody>
</table>

</section>

<section>
  <name>Change History</name>

<t>RFC Editor Note: Please delete this appendix on publication.</t>

<section>
  <name>From -00 to -01</name>

<ol>
  <li>Add Security Considerations section on why FNV is
non-cryptographic.</li>
  <li>Add Appendix A on a work factor comparison with SHA-1.</li>
  <li>Add Appendix B concerning previous IETF draft referenced to
  FNV.</li>
  <li>Minor editorial changes.</li>
</ol>

</section>

<section>
  <name>From -01 to -02</name>

<ol>
  <li>Correct FNV_Prime determination criteria and add note as to why
  s &lt; 5 and s &gt; 10 are not considered.</li>
  <li>Add acknowledgements list.</li>
  <li>Add a couple of references.</li>
  <li>Minor editorial changes.</li>
</ol>
  
</section>

<section>
  <name>From -02 to -03</name>

  <t>Minor editing changes.</t>
  
</section>

<section>
  <name>From -03 to -04</name>

  <t>Minor addition to Section 6, point 3. Update dates and version
  number. Minor editing changes.</t>
  
</section>

<section>
  <name>From -04 to -05</name>

<ol>
  <li>Add Twitter as a use example and IPv6 flow hash study
  reference.</li>
  <li>Update dates and version number.</li>
</ol>
  
</section>

<section>
  <name>From -05 to -06</name>

<ol>
  <li>Add code subsections.</li>
  <li>Update dates and version number.</li>
</ol>
  
</section>

<section>
  <name>From -06 to -07 to -08</name>

<ol>
  <li>Update Author info.</li>
  <li>Minor edits.</li>
</ol>
  
</section>

<section>
  <name>From -08 to -09</name>

<ol>
  <li>Change reference for ASCII to <xref target="RFC0020"/>.</li>
  <li>Add more details on history of the string used to compute
  offset_basis.</li>
  <li>Re-write "Work Factor" part of Section 6 to be more
  precise.</li>
  <li>Minor editorial changes.</li>
</ol>
  
</section>

<section>
  <name>From -09 to -10</name>

<ol>
  <li>Inclusion of initial partial version of code and some
  documentation about the code, Section 6.</li>
  <li>Insertion of new Section 4 on hashing values.</li>
</ol>
  
</section>

<section>
  <name>From -10 to -11</name>

  <t>Changes based on code improvements primarily from Tony Hansen who
  has been added as an author. Changes based on comments from Mukund
  Sivaraman and Roman Donchenko.</t>
  
</section>

<section>
  <name>From -11 to -12</name>

  <t>Keep alive update.</t>
  
</section>

<section>
  <name>From -12 to -13</name>

  <t>Fixed bug in pseudocode in Section 2.3.</t>

  <t>Change code to eliminate the BigEndian flag and so there are
  separate byte vector output routines for FNV32 and FNV64, equivalent
  to the other routines, and integer output routines for cases where
  Endianness consistency is not required.</t>

</section>

<section>
  <name>From -13 to -14 to -15 to -16 to -17</name>

  <t>Keep alive updates.</t>

  <t>Update an author address.  Update reference.  Update author
  affiliation.</t>
  
</section>

<section>
  <name>From -17 to -18</name>

  <t>Add reference to draft-ietf-bfd-secure-sequence-numbers. Update author
  info. Minor editorial changes.</t>
  
</section>

<section>
  <name>From -18 to -19</name>

  <t>Add references to the following, each of which uses FNV: RFC 7357, RFC
  7873, and IEEE Std. 802.1Qbp-2014.</t>
  
</section>

<section>
  <name>From -19 to -20</name>

  <t>Convert to XML v3. Fix code for loger FNV hashes.</t>
  
</section>

</section>

<section anchor="Acknowledgements" numbered="false">
  <name>Acknowledgements</name>

  <t>The contributions of the following are gratefully
  acknowledged:</t>

  <t>Roman Donchenko, Frank Ellermann, Tony Finch, Bob Moskowitz,
  Gayle Noble, Stefan Santesson, and Mukund Sivaraman.</t>
    
</section>

</back>

</rfc>
