1/*+============================================================================
  2|
  3| LEGAL
  4|
  5|     PN Version 1.0 - A Program for generating pseudonoise sequences.
  6|     Copyright (C) 1986-2025 by Sean Erik O'Connor.  All Rights Reserved.
  7|
  8|     This program is free software: you can redistribute it and/or modify
  9|     it under the terms of the GNU General Public License as published by
 10|     the Free Software Foundation, either version 3 of the License, or
 11|     (at your option) any later version.
 12|
 13|     This program is distributed in the hope that it will be useful,
 14|     but WITHOUT ANY WARRANTY; without even the implied warranty of
 15|     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 16|     GNU General Public License for more details.
 17|
 18|     You should have received a copy of the GNU General Public License
 19|     along with this program.  If not, see <http://www.gnu.org/licenses/>.
 20|     
 21|     The author's address is seanerikoconnor!AT!gmail!DOT!com
 22|     with !DOT! replaced by . and the !AT! replaced by @
 23|
 24+============================================================================*/
 25
 26#include <stdio.h>
 27
 28char * legalNotice =
 29{
 30    "\n"
 31	"PN Version 1.0 - A Program for generating pseudonoise sequences.\n"
 32    "Copyright (C) 1986-2025 by Sean Erik O'Connor.  All Rights Reserved.\n"
 33    "\n"
 34    "PN comes with ABSOLUTELY NO WARRANTY; for details see the\n"
 35    "GNU General Public License.  This is free software, and you are welcome\n"
 36    "to redistribute it under certain conditions; see the GNU General Public License\n"
 37    "for details.\n\n"
 38}  ;
 39
 40int pn( void ) ;
 41
 42/* Print out values of a sample binary PN sequence. */
 43
 44int main( int argc, char * argv[] )
 45{
 46    int i ;
 47
 48    /*  Show the legal notice first.  */
 49    printf(  "%s", legalNotice )  ;
 50
 51    /* Print out PN sequence values. */
 52    for (i = 1 ;  i <= 15 ;  ++i)
 53    {
 54        printf( "pn %3d = %3d\n", i, pn() ) ;
 55    }
 56
 57    return 0 ;
 58}
 59
 60
 61/* Compute the bits of a binary PN sequence. */
 62
 63int pn( void )
 64{
 65    static unsigned int shiftRegister = 1 ;   /* Initial state of shift register, */
 66                                              /* a  = 1, a  = ... = a  = 0        */
 67                                              /*  0       1          3            */
 68
 69    unsigned int m        = 4 ;               /* Period of PN sequence = 2^m - 1 = 15   */
 70
 71    unsigned int primPoly = 0x13 ;            /* Primitive polynomial coefficients,     */
 72                                              /* x^4 + x + 1 = 10011 (binary) = 13 hex  */
 73
 74    unsigned int mask     = (1 << (m-1)) ;    /* Selects the mth bit.                   */
 75
 76    unsigned int multiplier =
 77                 (primPoly ^ (1 << m)) ;      /* Feedback multiplier, corresponding to   */
 78                                              /* the taps f ... f on the shift register, */
 79                                              /*           3     0                       */
 80                                              /* equal to the primitive polynomial       */
 81                                              /* coefficients with highest degree term   */
 82                                              /* omitted.                                */
 83
 84    unsigned int feedback ; /* Feedback bit.                  */
 85    unsigned int bit ;      /* Bit 0 or 1 of the PN sequence. */
 86    unsigned int product ;  /* Bitwise product.               */
 87    unsigned int i ;
 88
 89    product = shiftRegister & multiplier ;   /* Modulo 2 multiplication of feedback */
 90                                             /* taps times shift register contents  */
 91                                             /* in parallel.                        */
 92
 93    /* Sum up the product bits modulo 2 to get the feedback bit. */
 94    for (i = 1, feedback = 0 ;  i <= m ;  ++i)
 95    {
 96        feedback ^= (1 & product) ;         /* Add first bit of the product into  */
 97                                            /* the running sum of the feedback.   */
 98
 99        product >>= 1 ;                     /* Shift to expose next bit of the product. */
100    }
101
102    bit = shiftRegister & 1 ;               /* Save the zeroth (PN sequence) bit of */
103                                            /* of the shift register.               */
104
105    shiftRegister >>= 1 ;                   /* Shift the shift register contents.   */
106
107    if (feedback)                           /* Add feedback (if any) to the shift register */
108    {                                       /* contents.                                   */
109        shiftRegister ^= mask ;
110    }
111
112    return( bit ) ;                         /* Return the PN sequence bit. */
113}