Generating truly random numbers on a microcontroller is a known difficult task.
One way is to run two different clocks, and measure the clock drift between them.
The PIC16F1783 which I'm using does have two clocks. One using an external crystal and the LFINTOSC which powers the watchdog.
Is there any way to get at the LFINTOSC, while running from the external crystal clock?
One idea I had was to let the watchdog expire, and check the value of TMR2, but it doesn't work because TMR2 is reset by watchdog resets.
One could write a counter to EEPROM in a tight loop and let the watchdog expire, and then inspect the contents of EEPROM, but this seems very inelegant. Edit: It's much better to write to RAM, as suggested in one of the comments.
Is there a better way? Without using external circuitry, and without relying on ADC-noise (which might not always be present)?
Datasheet: http://ww1.microchip.com/downloads/en/DeviceDoc/40001579E.pdf
Edit: I want the random number to generate a GUID to be able to tell apart multiple devices sharing the same RS485 bus. My idea is to use something similar to 1-wire device discovery. But to do this I need to have different GUID on different devices, and I'd like to avoid the hassle of having to program them with unique ID:s.
Best Answer
Okay, I've found a way to do exactly what I asked in the question, without involving the WDT.
It's a bit of a hack, to say the least, and sacrifices two pins (EDIT: only one pin is sacrificed) (but requires no external components, so if you have two unused pins it will be "for free").
The idea is to use the PSMC (Programmable Switch Mode Control) of the PIC16F1783.
This can be clocked from the INTOSC, by way of connecting the HFINTOSC to the 4x PLL, yielding a 64MHz frequency to clock the PWM output.
The PWM output can then be routed to another pin on the PCB.
Now, using the external crystal to clock the CPU, the PWM signal can be read in a tight loop. Since the two clocks are not synchronized, there should jitter between the two clocks, and the PWM input should contain some unpredictable jitter.
On way to use this jitter to build random values could be to have a 16 byte checksum array. The TMR1 could then be configured to run as fast as possible, and every time the PWM signal changes, the TMR1 value could be written to the start of the array. Then the MD5 sum of of the array could be taken, and written back to the same array.
By iterating this procedure a few thousand times, a 16 byte MD5 hash could be built, which should be entirely random.
However, a MD5 checksumming algorithm barely fits on the PIC16F1783, so this is more of use on slightly more powerful chips. The same idea could be used though, by simply incrementing a byte by the TMR1 value, and let it wrap around a few thousand times before considering it "random enough".
The only way this could fail is if the internal 500kHz source would somehow sync up to the crystal oscillator. I have no idea if that is possible.
Update:
The following code seems to work in practice in my lab:
Update 2:
Only one pin is sacrificed, since it is possible to read the output PWM pin, you don't need to route it to an input and read that.