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

Półbajt ang. nibble - prośba o pomoc
P6
  • Rejestracja:ponad 4 lata
  • Ostatnio:ponad 4 lata
  • Postów:6
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:

Kopiuj
 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ę:

Kopiuj

 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!

Azarien
  • Rejestracja:ponad 21 lat
  • Ostatnio:około 18 godzin
0

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

P6
  • Rejestracja:ponad 4 lata
  • Ostatnio:ponad 4 lata
  • Postów:6
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ę...

edytowany 1x, ostatnio: Piotr66
CF
  • Rejestracja:ponad 4 lata
  • Ostatnio:około 2 lata
  • Postów:27
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ąć".

99xmarcin
  • Rejestracja:około 5 lat
  • Ostatnio:5 miesięcy
  • Postów:2420
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ą.


Holy sh*t, with every month serenityos.org gets better & better...
obscurity
w ogóle to dla mnie jakieś dziwne że protokół zakłada przesyłanie zaświeconych segmentów wyświetlacza 7-segmentowego zamiast gołych odczytów. To chyba odbiorca powinien decydować jakiego rodzaju ma wyświetlacz i samemu sobie poradzić z konwersją liczb na wejście wyświetlacza...
AL
@obscurity: może to i dziwne, ale mam taką teorię: przetwornik jest "multimetrowy", tzn. wprost zapalający tylko odp. segmenty, sterowanie też wprost do niego, a transmisja dołączona "obok" i procek nią sterujący de facto nie widzi nastaw.
P6
  • Rejestracja:ponad 4 lata
  • Ostatnio:ponad 4 lata
  • Postów:6
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ą.

P6
  • Rejestracja:ponad 4 lata
  • Ostatnio:ponad 4 lata
  • Postów:6
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..

Kopiuj
 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

CF
ale to podaj jeszcze jakie wskazania były na tych dwóch multimetrach, że były różne o dwa stopnie to inna kwestia i nie mówi o tym co się mogło zmienić w formacie.
P6
  • Rejestracja:ponad 4 lata
  • Ostatnio:ponad 4 lata
  • Postów:6
0

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

P6
  • Rejestracja:ponad 4 lata
  • Ostatnio:ponad 4 lata
  • Postów:6
0

Kod z aplikacji dla Linux'a:

Kopiuj
#
# 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

Kopiuj
# 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?

edytowany 1x, ostatnio: Piotr66

Zarejestruj się i dołącz do największej społeczności programistów w Polsce.

Otrzymaj wsparcie, dziel się wiedzą i rozwijaj swoje umiejętności z najlepszymi.