Portál AbcLinuxu, 10. května 2025 10:54

Dotaz: ovládáni zařízení na COM pomocí C

22.6.2007 11:32 bukowski | skóre: 11
ovládáni zařízení na COM pomocí C
Přečteno: 335×
Odpovědět | Admin
Zdravim

mám na sériovém portu připojené zařízení a potřebuju číst stav DCD (Data Carrier Detect), a zároveň posílat každé 3 sec, že žiji. Potřeboval bych nakopnout jak na to. Když použiji TIOCMIWAIT tak čeká dokud se nezmění bit na DCD (nemá timeout) ale já potřebuji posílat každé 3 sec heartbeat a když použiji select() tak ten mi zase nereaguje na DCD. Vlákno použít je asi blbost nebo fork ()? Rád se přiučím předem dík za odpověď
Nástroje: Začni sledovat (3) ?Zašle upozornění na váš email při vložení nového komentáře.

Odpovědi

22.6.2007 13:08 radikn
Rozbalit Rozbalit vše Re: ovládáni zařízení na COM pomocí C
Odpovědět | | Sbalit | Link | Blokovat | Admin
Kdysi jsem delal program, ktery z COM portu prepisoval do textoveho souboru vypis uskutecnenych odchozich hovoru z telefonni ustredny. Urcite by se to dalo udelat podle toho. Klidne ti to poslu, na jaky e-mail?
22.6.2007 13:53 Jan Martinek | skóre: 43 | blog: johny | Brno
Rozbalit Rozbalit vše Re: ovládáni zařízení na COM pomocí C
Odpovědět | | Sbalit | Link | Blokovat | Admin
Myslím, že vlákna by se použít dala. Třeba takhle:
#!/usr/bin/env python
import sys, os, threading, tty, fcntl, struct, time

class CD_thread(threading.Thread):
    def __init__(self):
        threading.Thread.__init__(self)
        self.fd = fd

    def run(self):
        while True:
            fcntl.ioctl(fd, tty.TIOCMIWAIT, tty.TIOCM_CD)
            lock.acquire()
            print "DCD changed"
            lock.release()

fd = os.open('/dev/ttyS0', os.O_RDWR | os.O_SYNC)
lock = threading.Lock()
cd_thread = CD_thread()
cd_thread.start()

try:
    while True:
        lock.acquire()
        print "heartbeat"
        os.write(fd, "\xff"*100)
        lock.release()
        time.sleep(3)
except:
    cd_thread._Thread__stop()
Hlavní vlákno každé tři sekundy něco pošle na sériový port, přičemž jiné vlákno hlídá změnu stavové linky DCD. Zámek hlídá, aby si vlákna moc nelezla do zelí (ale to se stejně trochu děje). Zdá se, že to funguje, ze sériového portu skutečně něco leze a na DCD to reaguje, ale fakt nevím, jestli nemůže nastat nějaká problémová situace.
22.6.2007 18:24 macrek | skóre: 12
Rozbalit Rozbalit vše Re: ovládáni zařízení na COM pomocí C
Odpovědět | | Sbalit | Link | Blokovat | Admin
podla mna vlakno na to kludne pouzi, da sa aj fork(), ale vlakno je na to IMHO v pohode.
An eye for an eye makes the whole world blind.
Josef Kufner avatar 22.6.2007 18:43 Josef Kufner | skóre: 70
Rozbalit Rozbalit vše Re: ovládáni zařízení na COM pomocí C
Odpovědět | | Sbalit | Link | Blokovat | Admin
man 2 signal
man 2 alarm
Je to jednoduché a mělo by to fungovat bez problémů. Spočívá to v tom, že si nastavíš, za jak dlouho ti má jádro poslat signál SIGALRM. A jakýkoliv příchozí signál by měl přerušit ten syscall, který se během něho prováděl. Takže do obsluhy signálu bych dal jen nastavení nějakého flagu a pořešil to v hlavní smyčce (nebudeš mít problémy se synchronizací).
Hello world ! Segmentation fault (core dumped)
25.6.2007 12:01 bukowski | skóre: 11
Rozbalit Rozbalit vše Re: ovládáni zařízení na COM pomocí C
tak SIGALRM ten syscall neprerusi jen provede urcenou funkci tak jsem to udelal takhle

 void sig_alrm(int signo)
{
  printf ("posilam heartbeat\n") ;
  alarm(3);  //znovu za 3 sec
}

int main() 
{ 
  int fd,er, rts = TIOCM_RTS, dtr = TIOCM_DTR, talk;
  if ((fd=open("/dev/ttyS0", O_RDWR | O_NOCTTY | O_NONBLOCK)) < 0) 
	{
		perror("open_port: Unable to open /dev/ttyS0 - ");
		return 1;
	}
  //nastav bity potrebuju volty na diodu
  ioctl(fd, TIOCMBIS, &rts);
  ioctl(fd, TIOCMBIC, &dtr);
  
  if (signal(SIGALRM, sig_alrm) == SIG_ERR)
        perror("signal(SIGALRM) error");
  

  while(1){
	alarm(3);
	er = ioctl(fd, TIOCMIWAIT, TIOCM_CAR);
  	if (er < 0)
  	{
    		if (er != EINTR /*ERESTARTSYS*/)
      		perror("wait_DCD");
	}
  	else {
		alarm(0);
		if (ioctl(fd, TIOCMGET, &talk) < 0)
    		perror("get_DCD");
		if (talk & TIOCM_CAR)
			printf ("1\n");
		else
			printf ("0\n");
	}
}
close (fd);
} 
25.6.2007 13:16 petris
Rozbalit Rozbalit vše Re: ovládáni zařízení na COM pomocí C
preruseni si musite vyzadat, viz man siginterrupt ;-)
25.6.2007 15:32 bukowski | skóre: 11
Rozbalit Rozbalit vše Re: ovládáni zařízení na COM pomocí C
diiik super beha to


static int s, alarm_flag = 0 ;
static struct sockaddr_in server_address;

void send_data (int data)
{
  if (sendto(s, &data, 1, 0, (struct sockaddr*)&server_address, sizeof(server_address))==-1)
  	{perror("Error sending datagram: "); close(s); exit(-1); }
}


void sig_alrm(int signo)
{
  alarm_flag++;
}

int main() 
{ 
  int fd, rts = TIOCM_RTS, dtr = TIOCM_DTR, talk, mask = TIOCM_CAR, result, DCD,er;

  
  s = socket(PF_INET, SOCK_DGRAM, 0);

  if (s == -1) {
    perror("Server: Error Opening socket \n");
    exit (-1);
  }

  // pripravime adresu serveru 

  server_address.sin_family=AF_INET;
  server_address.sin_port=htons(32000);
  server_address.sin_addr.s_addr=inet_addr("127.0.0.1");

  printf("Sending datagram to server\n");

  // posleme datagram na pripravenou adresu serveru

  //open the device

  if ((fd=open("/dev/ttyS0", O_RDWR | O_NOCTTY | O_NONBLOCK)) < 0) 
	{
		perror("open_port: Unable to open /dev/ttyS0 - ");
		return 1;
	}
  
  //nastav bity potrebuju volty na diodu
  ioctl(fd, TIOCMBIS, &rts);
  ioctl(fd, TIOCMBIC, &dtr);

  siginterrupt (SIGALRM,1);
  if (signal(SIGALRM, sig_alrm) == SIG_ERR)
        perror("signal(SIGALRM) error");
  

  while(1){
	alarm(3);
  	if ( ioctl(fd, TIOCMIWAIT, TIOCM_CAR)< 0) if (errno != EINTR /*ERESTARTSYS*/) perror("wait_DCD");
	if (alarm_flag) {
		printf ("posilam heartbeat\n") ;
		send_data (1);
		alarm_flag = 0;
		}
  	else {
		alarm(0);
		if (ioctl(fd, TIOCMGET, &talk) < 0)
    		perror("get_DCD");
		if (talk & TIOCM_CAR)
			{printf ("1\n");send_data (10);}
		else
			{printf ("0\n");send_data (0);}
	}
}
  close(fd);	
  close(s);
}
25.6.2007 16:47 Michal Kubeček | skóre: 72 | Luštěnice
Rozbalit Rozbalit vše Re: ovládáni zařízení na COM pomocí C
Také je dobré zkontrolovat hodnotu příznaku SA_RESTART, viz sigaction(2).

Založit nové vláknoNahoru

Tiskni Sdílej: Linkuj Jaggni to Vybrali.sme.sk Google Del.icio.us Facebook

ISSN 1214-1267, (c) 1999-2007 Stickfish s.r.o.