Noise and silence-resistant

This commit is contained in:
Daniele Verducci (Slimpenguin) 2022-06-05 07:44:54 +02:00
parent ddf94e1ae1
commit a99a1b9763
2 changed files with 26 additions and 14 deletions

View File

@ -34,8 +34,12 @@ DESCRIPTION = 'Decodes a file using the manchester encoding'
FRAME_DELIMITER = 126 # (01111110) FRAME_DELIMITER = 126 # (01111110)
FRAME_DELIMITER_EVERY_BYTES = 64 FRAME_DELIMITER_EVERY_BYTES = 64
PREAMBLE_DURATION = 128 PREAMBLE_DURATION = 512
ZERO_POINT = 0 # The 0 value: values less than this are considered 0, more than this 1 # Values nearest to zero than this are not considered: set to more than noise, less than signal
AUDIO_MIN_VOLUME = 12288
# The 0 value: values less than this are considered 0, more than this 1. This should be
# auto-adjusting (to leave out the DC component, in hw one would use a transformer)
ZERO_POINT = 0
class Main: class Main:
@ -69,6 +73,7 @@ class Main:
while True: while True:
(cycles, raising) = self.goToNextZeroCrossing(True) (cycles, raising) = self.goToNextZeroCrossing(True)
analyzedCycles = analyzedCycles + cycles analyzedCycles = analyzedCycles + cycles
self._log.debug("Found zero crossing after {}, raising: {}".format(cycles, raising))
if analyzedCycles > PREAMBLE_DURATION * self.clockDuration / 4: if analyzedCycles > PREAMBLE_DURATION * self.clockDuration / 4:
# At this point we should have an idea of the clock duration, move on # At this point we should have an idea of the clock duration, move on
@ -103,7 +108,7 @@ class Main:
decodedByte = self.decodeByte(True) decodedByte = self.decodeByte(True)
if decodedByte != FRAME_DELIMITER: if decodedByte != FRAME_DELIMITER:
raise ValueError('Expecting a frame delimiter, found {} at position {}'.format(decodedByte, position)) raise ValueError('Expecting a frame delimiter, found {} at position {}'.format(decodedByte, position))
self._log.info('Found frame delimiter') self._log.debug('Found frame delimiter')
decodedByte = self.decodeByte(False) decodedByte = self.decodeByte(False)
try: try:
@ -144,7 +149,7 @@ class Main:
return decodedByte return decodedByte
def decodeBit(self): def decodeBit(self, allowSilence = False):
# Decodes a bit. Searches for the phase invertion at 75% to 125% of the clock cycle # Decodes a bit. Searches for the phase invertion at 75% to 125% of the clock cycle
bitDuration = 0 bitDuration = 0
while True: while True:
@ -194,16 +199,20 @@ class Main:
if not frame: if not frame:
raise ValueError('No more data to read') raise ValueError('No more data to read')
v = int(struct.unpack('<h', frame)[0]) > ZERO_POINT lvl = int(struct.unpack('<h', frame)[0])
if prev == None: if lvl > AUDIO_MIN_VOLUME or lvl < -AUDIO_MIN_VOLUME:
prev = v v = lvl > AUDIO_MIN_VOLUME
if v != prev: if prev == None:
# Zero-point crossing! prev = v
if adjustClockDuration: if v != prev:
self.clockDuration = (self.clockDuration + cyclesSinceLastInversion) / 2 # Zero-point crossing!
return (cyclesSinceLastInversion, v) if adjustClockDuration:
self.clockDuration = (self.clockDuration + cyclesSinceLastInversion) / 2
return (cyclesSinceLastInversion, v)
cyclesSinceLastInversion = cyclesSinceLastInversion + 1 # Count only cycles after first valid signal
if prev != None:
cyclesSinceLastInversion = cyclesSinceLastInversion + 1
@ -222,10 +231,13 @@ if __name__ == '__main__':
parser.add_argument('inputFile', help="audio file to decode (in wav format)") parser.add_argument('inputFile', help="audio file to decode (in wav format)")
parser.add_argument('outputFile', help="decoded file to write") parser.add_argument('outputFile', help="decoded file to write")
parser.add_argument('-v', '--verbose', action='store_true', help="verbose output") parser.add_argument('-v', '--verbose', action='store_true', help="verbose output")
parser.add_argument('-d', '--debug', action='store_true', help="even more verbose output, for debug")
args = parser.parse_args() args = parser.parse_args()
if args.verbose: if args.verbose:
logging.basicConfig(level=logging.INFO) logging.basicConfig(level=logging.INFO)
elif args.debug:
logging.basicConfig(level=logging.DEBUG)
else: else:
logging.basicConfig() logging.basicConfig()

View File

@ -34,7 +34,7 @@ DESCRIPTION = 'Encodes a file using the manchester encoding and outputs it as au
FRAME_DELIMITER = 126 # (01111110) FRAME_DELIMITER = 126 # (01111110)
FRAME_DELIMITER_EVERY_BYTES = 64 FRAME_DELIMITER_EVERY_BYTES = 64
PREAMBLE_DURATION = 128 PREAMBLE_DURATION = 512
AUDIO_VOLUME = 16384 # 0 to 32767 AUDIO_VOLUME = 16384 # 0 to 32767
AUDIO_BITRATE = 44100 AUDIO_BITRATE = 44100