Ir para conteúdo
Fernando Rafael

Leitura de porta serial (COM ou LPT) com PHP

Recommended Posts

Estou tendo problemas com a leitura de porta serial com PHP, consigo facilmente escrever na mesma, mas sempre que no comando de abertura da porta eu coloco que haverá permissão de leitura o PHP trava e fica carregando infinitamente, e a porta fica bloqueada e só libera quando reiniciado o computador.

 

Segue o cod:

exec('MODE COM1:9600,n,8,1'); //seta configuração na porta COM

$fp=fopen("COM1","r+b");

if(!$fp) {
    echo("Erro ao abrir a porta com1");
    exit;
} else {    
    echo("COM1 aberta <br>");
}

fwrite($fp,Chr(5));  //escrevendo valor na porta

sleep(1); //pausa para retorno dos dados

echo (fgets($fp));

fclose($fp);

já tentei  colocar:

fopen("COM1", "r+b");

fopen("COM1", "r+t");

fopen("COM1", "r");

fopen("COM1", "w+");

fopen("COM1", "r+");

 

para escrever consigo normal, com o fopen("COM1", "w");.  mas já para ler.....

 

Notei que na internet tem muito sobre esse problema, inclusive falaram em permissões de usuário, sendo que nos casos em que conseguiram, estavam usando o Linux.

Eu estou usando o W7 e W10 com PHP 5.2, se for o caso de permissões, como resolvo isso? preciso configurar mais alguma coisa no php.ini?

 

Também já tentei usar uma classe: phpSerial

 

Mas retorna o seguinte erro:

Warning: Reading serial port is not implemented for Windows in C:\Program Files (x86)\Apache Software Foundation\Apache2.2\htdocs\balanca\php_serial.class.php on line 503

 

 

Compartilhar este post


Link para o post
Compartilhar em outros sites

Testando um pouco mais, reparei que o PHP não trava quando abrimos a porta com permissão de leitura (w+, r, r+ e etc...), mas sim quando tentamos exibir seu retorno, nos seguintes comandos:

 

fread($fp);

ou  

fgets($fp);

 

Também tentei limitar o tamanho em bytes da leitura, mas sem sucesso...   : (

 

fread($fp, 1024);

ou  

fgets($fp, 1024);

 

Vi um cara dizendo que conseguiu usando comandos CMD via "exec" do PHP, mas não citou nenhuma fonte ou código....

 

Compartilhar este post


Link para o post
Compartilhar em outros sites

Oi boa tarde EdCesar.

testei a classe indicada por você, mas sem sucesso...

 

Segue o código completo:

<?php

include 'PhpSerial.php';
// Let's start the class
$serial = new PhpSerial();
// First we must specify the device. This works on both linux and windows (if
// your linux serial device is /dev/ttyS0 for COM1, etc)
$serial->deviceSet("COM6");
// We can change the baud rate, parity, length, stop bits, flow control
$serial->confBaudRate(9600);
$serial->confParity("none");
$serial->confCharacterLength(8);
$serial->confStopBits(1);
$serial->confFlowControl("none");
// Then we need to open it
$serial->deviceOpen();
// To write into
$serial->sendMessage("Hello !");
// Or to read from
$read = $serial->readPort();
// If you want to change the configuration, the device must be closed
$serial->deviceClose();
// We can change the baud rate
$serial->confBaudRate(9600);

?>

 

 

Na linha:  $serial->deviceOpen(); ele mostra o seguinte erro no navegador:

 

Warning: Unable to open the device in C:\Program Files (x86)\Apache Software Foundation\Apache2.2\htdocs\balanca\PhpSerial.php on line 149

 

A classe que testei antes foi essa: https://www.phpclasses.org/browse/file/17926.html

na qual retorna o seguinte erro, também quando se tenta ler algo (Linha: $read=$serial->readPort();):


 

require "php_serial.class.php";
$serial = new phpSerial;
$serial->deviceSet("COM6");
$serial->confBaudRate(9600);

$serial->deviceOpen();
$serial->sendMessage(chr(5));
sleep(7);
$read=$serial->readPort(); //tabém da erro ao executar a leitura
$serial->deviceClose();

Retorno no navegador:

Warning: Reading serial port is not implemented for Windows in C:\Program Files (x86)\Apache Software Foundation\Apache2.2\htdocs\balanca\php_serial.class.php on line 503

 

A porta funciona, pois consigo comunicar normal com outras aplicações, incluindo simuladores de balança, Serial Port Monitor e etc...

 

 

Spoiler


 

Compartilhar este post


Link para o post
Compartilhar em outros sites

Boa tarde, continuando as buscas...

 

achei um comando para setar um timeOut no PHP (Colocar "stream_set_timeout($fp, 10);"  depois de abrir a porta), mas não funcionou, continua carregando infinitamente...

 

Achei um equivalente em powerShell, que sem o comando de timeOut fica rodando infinitamente...

segue o cod:

 

$port= new-Object System.IO.Ports.SerialPort COM6,9600,None,8,One

$port.ReadTimeout = 10000

$port.Open()

while($myinput = $port.ReadLine())
{
$myinput
}

$port.Close()

 

Mas não me retornou nada depois que entra em timeOut...

 

Bem, estou usando um simulador de balança (filizola/toledo) juntamente com um simulador de porta serial, testo a conexão com um programa de monitoramento de porta ou com o PC Scale da Filizola, e neles eu tenho retorno....

 

OBS: o simulador de balança tem um botão que envia dados para a porta, sem a necessidade de escrever na porta para que a mesma retorne algo.

 

Compartilhar este post


Link para o post
Compartilhar em outros sites

Vendo em muitos Fóruns, notei que o PHP tem bloqueios para leitura de portas seriais no Windows, sendo necessário configurar ou usar DLL (não encontrei tutoriais do tipo).

 

Bem...  Resolvi da forma mais aceitável diante da situação (ambiente Windows):

Fiz um programa em JAVA (exportado em um arquivo .jar) que recebe os parâmetros da porta serial e efetua a leitura da mesma. Mas fiz de uma forma mais diretamente "ligada" ao PHP. Sabendo que o arquivo .jar pode ser executado pelo CMD passando até mesmo parâmetros por linha de comando, fiz com que o PHP acionasse esse arquivo pelo comando "exec" ou "passthru" e depois tratei o retorno no próprio PHP, acredito que dessa forma é bem melhor do que ficar criando arquivos .txt para ficar lendo depois...

Então, sem mais delongas segue os fontes (JAVA e PHP):

PHP:

$comando_cmd="java -jar ler_serial.jar 9600 8 1 0 COM3 ENQ"; //Arquivo seguido dos parâmetros que serão capturados pelo arquivo .jar (separados por espaços).  OBS: só coloquei parâmetros pois não quis criar um arquivo .jar para cada tipo de porta ou configurações e comandos (ENQ por exemplo serve apenas para saber qual comando será usando no arquivo .jar, deve haver alguma forma de executar os comandos no JAVA que estejam armazenados em uma variável, eliminando assim os "if" e "else" para cada instrução dentro do arquivo .jar).

echo passthru($comando_cmd); //exibe o retorno.

 

JAVA:

import jssc.SerialPort; //Para funcionar é preciso adicionar no projeto uma biblioteca (usei essa: jSSC-2.7.0-Release.zip), vale pesquisar como adicionar, é fácil
import jssc.SerialPortException; //Para funcionar é preciso adicionar no projeto uma biblioteca (usei essa: jSSC-2.7.0-Release.zip), vale pesquisar como adicionar, é fácil

//Fonte sobre essa biblioteca: http://nakalabs.herokuapp.com/articles/arduinoAndJava.html

public class Main {
    
    public static void main(String[] args) throws InterruptedException {
   

         //Parametros estáticos, caso não usem os valores passados por linha de comando
        /*
        int BAUD_RATE =  9600;
        int DATA_BITS = 8;
        int STOP_BITS = 1;
        int PARITY    = 0;
        String SERIAL_PORT = "COM2";
        */
        
        //Armazena os parâmetros nas variáveis
        int BAUD_RATE =  Integer.parseInt(args[0]); //9600
        int DATA_BITS = Integer.parseInt(args[1]); //8
        int STOP_BITS = Integer.parseInt(args[2]); //1
        int PARITY    = Integer.parseInt(args[3]); //0
        String SERIAL_PORT = args[4]; //COM3
        
        String COMANDO ="ENQ"; //Comando inicial, caso nada seja passado como parametro, vai executar um ENQ (muito comum em comunicação serial)
        
        if(args[5].equals("ENQ")){ //Serve apenas para comparar o parâmetro passado e  executar algum comando previamente programado
            
            COMANDO =  ""+(char)5; //Coloquei ""+ (aspas dupla concatenada) só para aceitar como String, pois nem todo comando é do tipo (char) no meu caso
        }
        
        else if(args[5].equals("BEL")){ //Outro comando, e assim vai...
            
            COMANDO =  ""+(char)7+"p";  //Salientando que esses comandos são das minhas necessidades, isso vai depender do comando que o dispositivo esteja aguardando para retornar algo...
        }    
    
        SerialPort serialPort = new SerialPort(SERIAL_PORT);
        
        try {
            //Os comandos "exec" e "passthru" capturam esses retornos, basta exibilos ao seu favor...
            System.out.println("Porta aberta: " + serialPort.openPort());
            System.out.println("Parametros configurados: " + serialPort.setParams(BAUD_RATE, DATA_BITS, STOP_BITS, PARITY));
            System.out.println("Enviado mensagem: " + serialPort.writeString(""+COMANDO+"")); //Aqui ele escreve o comando na porta
            Thread.sleep(1000); //Aguarda 1 segundo para ler a porta
            //System.out.println("Retorno: " + serialPort.readString()); //Retorno da porta em String
            System.out.println("Retorno: " + serialPort.readHexString("")); //Retorno da porta em Hexadecimal, Com "" (aspas dupla) no parâmetro, significa que não haverá espaços entre os valores retornados ou colocando qualquer caractere, servirá como separador.
            System.out.println("Porta fechada: " + serialPort.closePort());
        
        }
        catch (SerialPortException ex){
            System.out.println(ex);
        }
    }
}

Postei essa mesma solução no ACBR, Imaster e Devmedia. nos que suportarem anexos, colocarei também a biblioteca do JAVA zipada.

 

Não sou bom em programação Java, caso alguém tenha melhorias para o código, ficarei muito grato... Vale visitar meu site: www.sisnorg.com (ainda em desenvolvimento, mas funcional para um possível contato comigo).

Abraços a todos que me ajudaram nos mais diversos fóruns... 

Compartilhar este post


Link para o post
Compartilhar em outros sites

Rafael, é possível ler essas portas sem bloqueio desenvolvemos uma aplicação no trablho que envia comando hexadecimais para para um equipamento com porta serial, no caso usamos um cabo conversor de usb para serial.

 

usamos a seguinte codificação:

//Passa os paramentros de acesso para a porta COM, aqui no caso a porta com
exec("mode com5: BAUD=9600 PARITY=n DATA=8 STOP=1 to=off dtr=off rts=off");

//Faz a abertura em modo de escrita
$fp =fopen("com5", "w");

após a abertura você pode enviar o comando:

fwrite($fp, "mensagem a ser enviada"); //Hexa para camera 1
fclose($fp);

 

Compartilhar este post


Link para o post
Compartilhar em outros sites

Oi Icarof, o problema com PHP não era a escrita, mas sim a leitura (em ambiente Windows), o PHP sempre travava na Leitura, minha solução foi deixar o PHP com a escrita, e para os casos em que se faz necessário a leitura o Java entra em ação.

Compartilhar este post


Link para o post
Compartilhar em outros sites

entendo, no caso não faziamos a leitura, os dados eram enviado direto a maquina que recebia os camandos

Compartilhar este post


Link para o post
Compartilhar em outros sites

Crie uma conta ou entre para comentar

Você precisar ser um membro para fazer um comentário

Criar uma conta

Crie uma nova conta em nossa comunidade. É fácil!

Crie uma nova conta

Entrar

Já tem uma conta? Faça o login.

Entrar Agora

  • Conteúdo Similar

    • Por ILR master
      Fala galera.
      Espero que todos estejam bem.
      Seguinte: Tenho um arquivo xml onde alguns campos estão com : (dois pontos), como o exemplo abaixo:
       
      <item>
      <title>
      d sa dsad sad sadasdas
      </title>
      <link>
      dsadas dsa sad asd as dsada
      </link>
      <pubDate>sadasdasdsa as</pubDate>
      <dc:creator>
      d sad sad sa ad as das
      </dc:creator>
      </item>
       
      Meu código:
       
      $link = "noticias.xml"; 
      $xml = simplexml_load_file($link); 
      foreach($xml -> channel as $ite) {     
           $titulo = $ite -> item->title;
           $urltitulo = $ite -> item->link;
           print $urltitulo = $ite -> item->dc:creator;
      } //fim do foreach
      ?>
       
      Esse campo dc:creator eu não consigo ler. Como faço?
       
      Agradeço quem puder me ajudar.
       
      Abs
       
       
    • Por First
      Olá a todos!
       
      Eu estou criando um sistema do zero mas estou encontnrando algumas dificuldades e não estou sabendo resolver, então vim recorrer ajuda de vocês.
      Aqui está todo o meu código: https://github.com/PauloJagata/aprendizado/
       
      Eu fiz um sistema de rotas mas só mostra o conteúdo da '/' não sei porque, quando eu tento acessar o register nada muda.
      E eu também quero que se não estiver liberado na rota mostra o erro de 404, mas quando eu tento acessar um link inválido, nada acontece.
      Alguém pode me ajudar com isso? E se tiver algumas sugestão para melhoria do código também estou aceitando.
       
       
      Desde já, obrigado.
    • Por landerbadi
      Olá pessoal, boa tarde
       
      Tenho uma tabela chamada "produtos" com os seguintes campos (id, produto) e outra tabela chamada "itens" com os seguintes campos (id, prod_01, prod_02, prod_03, prod_04).
       
      Na tabela produtos eu tenho cadastrado os seguintes produtos: laranja, maçã, uva, goiaba, arroz, feijão, macarrão, etc.
       
      Na tabela itens eu tenho cadastrado os itens da seguinte maneira:
       
      1, laranja, uva, arroz, feijão;
      2, maçã, macarrão, goiaba, uva;
      3, arroz, feijão, maçã, azeite
       
      Meu problema é o seguinte: 
      Eu escolho um produto da tabela "produtos", por exemplo "uva".  Preciso fazer uma consulta na tabela "itens" para ser listado todos os registros que contenham o produto "uva" e que todos os demais produtos estejam cadastrados na tabela "produtos".
       
      No exemplo acima seria listado apenas dois registros, pois o terceiro registro não contém o produto "uva". 
       
      Alguém pode me ajudar? Pois estou quebrando a cabeça a vários dias e não consigo achar uma solução.
    • Por landerbadi
      Boa tarde pessoal. Estou tentado fazer uma consulta no banco de dados porém estou tendo dificuldades. Tenho uma tabela chamada "itens" com os seguintes campos: id, item, plural, ativo. Nela tem cadastrado vários itens e seu respectivo plural. No campo ativo eu coloco a letra "S" para informar que esta palavra está ativa no sistema. Por exemplo: 1, casa, casas, S 2, mesa, mesas, S 3, cama, camas, S 4, moto, motos, S 5, rádio, rádios O quinto registro "radio" não está ativo no sistema pois não tem um "S" no campo ativo. E outra tabela chamada "variações" com os seguintes campos (id, item1, item2, item3) com os seguintes registros: 1, casa, camas, moto 2, mesas, casas, radio 3, rádio, cama, mesa Eu preciso fazer uma busca na tabela variações da seguinte maneira: Eu escolho um registro na tabela "itens", por exemplo "casa". Preciso fazer com que o php me liste todos os registros da tabela "variações" que contenham a palavra "casa". Porém se tiver algum registro com a palavra "casas" também tem que ser listado. Neste caso ele irá encontrar dois registros. Agora eu preciso que o php verifique os demais itens e faça a listagem apenas dos item que estão ativos (que contenham um "S" no campo ativo. Neste caso ele irá encontrar apenas um registro, pois o segundo registro contém a palavra "rádio". E "rádio" não está ativo na tabela itens. Como faço isso?
    • Por First
      Olá a todos!
       
      Quando eu tento fazer o login me mostra esse erro "Could not log you in."; Alguém sabe me ajudar a resolver esse problema no meu código?
      <?php require_once("core/init.php"); if (Input::exists()) { if (Token::check(Input::get("token"))) { $validate = new Validate(); $validation = $validate->check($_POST, array( "username" => array("required" => true), "password" => array("required" => true) )); if ($validation->passed()) { $user = new User(); $remember = (Input::get("remember")) === "on" ? true : false; $login = $user->login(Input::get("username"), Input::get("password"), $remember); if ($login) { Session::flash("home", "Welcome back!"); Redirect::to("index.php"); } else { echo "Could not log you in."; } } else { foreach ($validation->errors() as $error) { echo $error."<BR>"; } } } } ?> <form action="" method="POST"> <div class="field"> <label for="username">Username</label> <input type="text" name="username" id="username"> </div> <div class="field"> <label for="password">Password</label> <input type="password" name="password" id="password"> </div> <div class="field"> <label for="remember"> <input type="checkbox" name="remember" id="remember"> Remember me </label> </div> <input type="hidden" name="token" value="<?php echo Token::generate(); ?>"> <input type="submit" value="Log in"> </form>  
       
      Desde já obrigado.
×

Informação importante

Ao usar o fórum, você concorda com nossos Termos e condições.