Ir para conteúdo

POWERED BY:

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 violin101
      Caros amigos, saudações.
       
      Por favor, poderiam me ajudar.

      Estou com a seguinte dúvida:
      --> como faço para para implementar o input código do produto, para quando o usuário digitar o ID o sistema espera de 1s a 2s, sem ter que pressionar a tecla ENTER.

      exemplo:
      código   ----   descrição
           1       -----   produto_A
       
      Grato,
       
      Cesar
    • Por violin101
      Caros amigos, saudações.
       
      Humildemente peço desculpa por postar uma dúvida que tenho.

      Preciso salvar no MySql, os seguinte Registro:

      1 - Principal
      ====> minha dúvida começa aqui
      ==========> como faço para o Sistema Contar Automaticamente o que estiver despois do 1.____?
      1.01 - Matriz
      1.01.0001 - Estoque
      1.01.0002 - Oficina
      etc

      2 - Secundário
      2.01 - Loja_1
      2.01.0001 - Caixa
      2.01.0002 - Recepção
      etc
       
      Resumindo seria como se fosse um Cadastro de PLANO de CONTAS CONTÁBEIL.

      Grato,


      Cesar









       
    • Por violin101
      Caros amigos, saudações.

      Por favor, me perdoa em recorrer a orientação dos amigos.

      Preciso fazer um Relatório onde o usuário pode Gerar uma Lista com prazo para vencimento de: 15 / 20/ 30 dias da data atual.

      Tem como montar uma SQL para o sistema fazer uma busca no MySql por período ou dias próximo ao vencimento ?

      Tentei fazer assim, mas o SQL me traz tudo:
      $query = "SELECT faturamento.*, DATE_ADD(faturamento.dataVencimento, INTERVAL 30 DAY), fornecedor.* FROM faturamento INNER JOIN fornecedor ON fornecedor.idfornecedor = faturamento.id_fornecedor WHERE faturamento.statusFatur = 1 ORDER BY faturamento.idFaturamento $ordenar ";  
      Grato,
       
      Cesar
       
       
       
       
    • Por violin101
      Caros amigos, saudações
       
      Por favor, me perdoa em recorrer a orientação dos amigos, tenho uma dúvida.
       
      Gostaria de uma rotina onde o Sistema possa acusar para o usuário antes dos 30 dias, grifar na Tabela o aviso de vencimento próximo, por exemplo:
       
      Data Atual: 15/11/2024
                                           Vencimento
      Fornecedor.....................Data.....................Valor
      Fornecedor_1...........01/12/2024..........R$ 120,00 <== grifar a linha de Laranja
      Fornecedor_1...........01/01/2025..........R$ 130,00
      Fornecedor_2...........15/12/2024..........R$ 200,00 <== grifar a linha de Amarelo
      Fornecedor_2...........15/01/2025..........R$ 230,00
      Fornecedor_3...........20/12/2024..........R$ 150,00
       
      Alguém tem alguma dica ou leitura sobre este assunto ?

      Grato,
       
      Cesar
    • Por violin101
      Caros amigos, saudações.

      Por favor, me perdoa em recorrer a ajuda dos amigos, mas preciso entender uma processo que não estou conseguindo sucesso.

      Como mencionado no Título estou escrevendo um Sistema Web para Gerenciamento de Empresa.
       
      Minha dúvida, que preciso muito entender:
      - preciso agora escrever a Rotina para Emissão de NFe e essa parte não estou conseguindo.
       
      tenho assistido alguns vídeos e leituras, mas não estou conseguindo sucesso, já fiz toda as importações das LIB da NFePhp conforme orientação.

      Preciso de ajuda.

      Algum dos amigos tem conhecimento de algum passo-a-passo explicando a criação dessa rotina ?

      tenho visto alguns vídeos com LARAVEL, mas quando tento utilizar e converter para PHP+Codeiginter, dá uma fila de erros que não entendo, mesmo informando as lib necessárias.

      Alguns do amigo tem algum vídeo, leitura explicando essa parte ?

      Grato,

      Cesar.
×

Informação importante

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