Półbajt ang. nibble - prośba o pomoc

0

Witam!

Chciałbym Was bardzo prosić o wytłumaczenie o co tu chodzi. Od kilku dni "studiuję" temat i o ile w teorii wydaje mi się zrozumiały, to w praktyce nie bardzo.

Jest urządzenie (multimetr), które transmituje poprzez port szeregowy odczyt, wybrany zakres itd. Transmisja jest bitowa (tak mi się wydaje) i jakoś "zakodowana".
Podsłuchałem port i wygląda to tak:

 00 D0 E2 F1 18 2E 3B 4E 5B 6A 7D 8C 97 A0 B0 C0   .Đâń..;N[j}Ś— °Ŕ
 D0 E2 F1 18 2E 3B 4E 5B 6A 7D 8C 97 A0 B0 C0 D0   Đâń..;N[j}Ś— °ŔĐ
 E2 F1 18 2E 3B 4E 5B 6A 7D 8C 97 A0 B0 C0 D0 E2   âń..;N[j}Ś— °ŔĐâ
 F1 18 2E 3B 4E 5B 6A 7D 8C 97 A0 B0 C0 D0 E2 F1   ń..;N[j}Ś— °ŔĐâń
 18 2E 3B 4E 5B 6A 7D 8C 97 A0 B0 C0 D0 E2 F1 18   ..;N[j}Ś— °ŔĐâń.
 2E 3B 4E 5B 6A 7D 8C 97 A0 B0 C0 D0 E2 F1 18 2E   .;N[j}Ś— °ŔĐâń..
 3B 4E 5B 6A 7D 8C 97 A0 B0 C0 D0 E2 F1 18 2E 3B   ;N[j}Ś— °ŔĐâń..;

itd.

Mam od źródłowy aplikacji, którą kiedyś lekko zmodyfikowałem i robiło co trzeba - zrzut do pliku ASCII w formie czytelnej dla ludzi. Urządzenie jednak pada i zakupione zostało nowe - niby identyczne (taki sam symbol), ale coś zmienili w protokole i już aplikacja nie działa. Zresztą do obu dołączona jest mala aplikacja Windows, która pokazuje odczyt oraz ekran LCD multimetru i ta stara również nie współpracuje z nowym urządzeniem.

W aplikacje, którą kiedyś znalazłem i lekko zmodyfikowałem jest taki opis:

    /**
     * display value is encoded by maping the elements of the 7-segment
     * to the lower nibbles of byte groups [a,b] = {[1,2],[3,4],[5,6],[7,8]}:
     * 	     0a:00	
     * 0a:01       0b:00
     *       0b:01	
     * 0a:02       0b:02
     *       0b:03
     *       
     * by combining the nibbles to a byte we got a uniqe identifier 
     * for every symbol shown on the display.
     * 
     * for byte 1 bit 3 represents a leading minus (negative value)
     * for bytes 3,5,7 bit 3 indicates a leading decimal point
     * (means the encoded value is the first part of the fraction)
     */

Napisałem na szybko taką funkcję:


 string swapNibbles(string _x) {
            int _in, _out = 0;
            string wynik = "";

            Int32.TryParse(_x, out _in);            
            _out = (_in & 0x0F) << 4 | (_in & 0xF0) >> 4;
            wynik = _out.ToString();

            return wynik;
        }

ale dalej nie wiem, czy to ma sens i co dalej z tym robić?

Bardzo proszę o pomoc!

0

No a jak wygląda transmisja na nowym urządzeniu?

0

W zasadzie tak samo, ale coś zmieniło się w "środku".

Zacząłem pisać do producenta, ale znalazłem dwie aplikację pod Linuxa napisane w C. Studiuję...

0
Piotr66 napisał(a):

W zasadzie tak samo, ale coś zmieniło się w "środku".

Może pokaż dump transmisji z nowego urządzenia oraz starego, oraz to co jest na wyświetlaczu (na nowym i starym) w momencie transmisji? Skoro mówisz, że zmiana jest nieduża to będzie się dało ją "zgadnąć".

0

Po pierwsze RTFM z urządzeniem na pewno przyszła dokumentacja lub jest ona dostępna na sieci jako tak zwany datasheet.
Po drugie sprawdź szybkość transmisji na porcie szeregowym może był upgrade i pcha po 56000 a nie po 9600.
Po trzecie ja ten opis 7-seg zrozumiałem tak, masz bajt = 8 bitów. Pierwsze 3 bity mówią które segmenty oznaczone jako 0a należy podświetlić, nie ma segmentu 0a:03 ale piszą że ten bit jest wykorzystywany do wyświetlenia minusa dla pierwszej cyfry lub kropki dla pozostałych. 4 ostatnie bity mówią które segmenty 0b należy podświetlić. Ma to sens, opis jest faktycznie zagmatwany.

Ja bym się skoncentrował na dołączonej aplikacji, jeżeli jest to coś co działa to masz już połowę pracy za sobą.

0
0xmarcin napisał(a):

Po pierwsze RTFM z urządzeniem na pewno przyszła dokumentacja lub jest ona dostępna na sieci jako tak zwany datasheet.

Nie ma. Mam oryginale opakowanie. Na stronie producenta nie ma,chyba do niego napiszę prośbę.

Po drugie sprawdź szybkość transmisji na porcie szeregowym może był upgrade i pcha po 56000 a nie po 9600.

Wg. "książeczki" od urządzenia, transmisja jest na 2400

Po trzecie ja ten opis 7-seg zrozumiałem tak, masz bajt = 8 bitów. Pierwsze 3 bity mówią które segmenty oznaczone jako 0a należy podświetlić, nie ma segmentu 0a:03 ale piszą że
ten bit jest wykorzystywany do wyświetlenia minusa dla pierwszej cyfry lub kropki dla pozostałych. 4 ostatnie bity mówią które segmenty 0b należy podświetlić. Ma to sens, opis jest
faktycznie zagmatwany.

Ale jaki jest cel, takich kombinacji?

Znalazłem dzisiaj dwa programiki pod Linux'a. Tzn. ich kod do kompilacji. Postudiuję i równocześnie napiszę prośbę do producenta.

Ja bym się skoncentrował na dołączonej aplikacji, jeżeli jest to coś co działa to masz już połowę pracy za sobą.

0
CosmoFire napisał(a):
Piotr66 napisał(a):

W zasadzie tak samo, ale coś zmieniło się w "środku".

Może pokaż dump transmisji z nowego urządzenia oraz starego, oraz to co jest na wyświetlaczu (na nowym i starym) w momencie transmisji? Skoro mówisz, że zmiana jest nieduża to będzie się dało ją "zgadnąć".

Mała różnica niestety zawsze będzie.Obydwa multimetry wskazują różną temperaturę, prawie o 2 stopnie. Dziwne, ale..

 13 27 3D 45 5B 63 7E 8B 9F A0 B0 C0 D1 E2 13 27   .'=E[c~‹ź °ŔŃâ.'
 3D 45 5B 63 7E 8B 9F A0 B0 C0 D1 E2 13 27 3D 45   =E[c~‹ź °ŔŃâ.'=E
 5B 63 7E 8B 9F A0 B0 C0 D1 E2 13 27 3D 45 5B 63   [c~‹ź °ŔŃâ.'=E[c
 7E 8B 9F A0 B0 C0 D1 E2 13 27 3D 45 5B 63 7E 8B   ~‹ź °ŔŃâ.'=E[c~‹
 9F A0 B0 C0 D1 E2 13 27 3D 45 5B 63 7E 8B 9F A0   ź °ŔŃâ.'=E[c~‹ź 
 B0 C0 D1 E2 13 27 3D 45 5B 63 7E 8B 9F A0 B0 C0   °ŔŃâ.'=E[c~‹ź °Ŕ
 D1 E2 13 27 3D 45 5B 63 7E 8B 9F A0 B0 C0 D1 E2   Ńâ.'=E[c~‹ź °ŔŃâ
 13 27 3D 45 5B 63 7E 8B 9F A0 B0 C0 D1 E2 13 27   .'=E[c~‹ź °ŔŃâ.'
 3D 45 5B 63 7E 8B 9F A0 B0 C0 D1 E2 13 27 3D 45   =E[c~‹ź °ŔŃâ.'=E
 5B 63 7E 8B 9F A0 B0 C0 D1 E2 13 27 3D 45 5B 63   [c~‹ź °ŔŃâ.'=E[c
 7E 8B 9F A0 B0 C0 D1 E2 13 27 3D 45 5B 63 7E 8B   ~‹ź °ŔŃâ.'=E[c~‹
 9F A0 B0 C0 D1 E2 13 27 3D 45 5B 63 7E 8B 9F A0   ź °ŔŃâ.'=E[c~‹ź 
 B0 C0 D1 E2 13 27 3D 45 5B 63 7E 8B 9F A0 B0 C0   °ŔŃâ.'=E[c~‹ź °Ŕ
 D1 E2 13 27 3D 45 5B 63 7E 8B 9F A0 B0 C0 D1 E2   Ńâ.'=E[c~‹ź °ŔŃâ
 13 27 3D 45 5B 63 7E 8B 9F A0 B0 C0 D1 E2 13 27   .'=E[c~‹ź °ŔŃâ.'
 3D 45 5B 63 7E 8B 9F A0 B0 C0 D1 E2 13 27 3D 45   =E[c~‹ź °ŔŃâ.'=E
 5B 63 7E 8B 9F A0 B0 C0 D1 E2 13 27 3D 45 5B 63   [c~‹ź °ŔŃâ.'=E[c

0

Stare pokazywało 25,9 stopni, nowe 24.

0

Kod z aplikacji dla Linux'a:

#
# Decode the measured data from the binary string
#
sub decode_bin_str {
	my ($AC, $DC, $auto, $unknown1,
	    $minus, $digi1, $dot1, $digi2, $dot2, $digi3, $dot3, $digi4,
	    $micro, $unknown2, $kilo, $diode_test,
	    $milli, $percent, $mega, $cont_check,
	    $unknown3, $ohm, $rel, $hold,
	    $amp, $volt, $hz, $unknown4,
	    $min, $unknown5, $celsius, $max) = shift =~
    	   /^(.)(.)(.)(.)(.)(.{7})(.)(.{7})(.)(.{7})(.)(.{7})
	    (.)(.)(.)(.)(.)(.)(.)(.)(.)(.)(.)(.)(.)(.)(.)(.)(.)(.)(.)(.) *$/x;
	
	my %digi = (
		"1111101" => 0,
		"0000101" => 1,
		"1011011" => 2,
		"0011111" => 3,
		"0100111" => 4,
		"0111110" => 5,
		"1111110" => 6,
		"0010101" => 7,
		"1111111" => 8,
		"0111111" => 9,
	);

	my $val = ($minus ? "-" : "") . $digi{$digi1} . ($dot1 ? "." : "") .
					$digi{$digi2} . ($dot2 ? "." : "") .
					$digi{$digi3} . ($dot3 ? "." : "") .
					$digi{$digi4};
	
	my $flags = join(" ", $AC         ? "AC"         : (),
			      $DC         ? "DC"         : (),
			      $auto       ? "auto"       : (),
			      $diode_test ? "diode_test" : (),
			      $cont_check ? "cont_check" : (),
			      $rel        ? "rel"        : (),
			      $hold       ? "hold"       : (),
			      $min        ? "min"        : (),
			      $max        ? "max"        : ());
	
	my $unit = ($micro   ? "u"   : "") .
		   ($kilo    ? "k"   : "") .
		   ($milli   ? "m"   : "") .
		   ($mega    ? "M"   : "") .
		   ($percent ? "%"   : "") .
		   ($ohm     ? "Ohm" : "") .
		   ($amp     ? "A"   : "") .
		   ($volt    ? "V"   : "") .
		   ($hz      ? "Hz"  : "") .
		   ($celsius ? "C"   : "");

	$val, $flags, $unit;
}

oraz

# Read next (complete) binary string from the optocoupler
#
sub next_bin_str {
	my $bin_str;
	while (!eof(USB_TTY) and length($bin_str) != 14 * 4) {
		my $byte = getc(USB_TTY);
		my $nibble_number = ord($byte) >> 4 & 0xf;
		
		$bin_str .= substr(unpack("B*", $byte), 4);
		length($bin_str) == $nibble_number * 4 or $bin_str = "";
	}
	$bin_str;
}

Więcej chyba nie potrzeba?

1 użytkowników online, w tym zalogowanych: 0, gości: 1