

Instituto Federal Catarinense Bacharelado em Engenharia de Controle e Automação  $Campus\ Luzerna$ 

João Peterson Scheffer

Desenvolvimento de um método e sistema para compilação e simulação de redes de petri para utilização em controladores lógicos industriais

Luzerna Dezembro de 2022

# João Peterson Scheffer

Desenvolvimento de um método e sistema para compilação e simulação de redes de petri para utilização em controladores lógicos industriais

Trabalho de conclusão apresentado à banca examinadora do curso de Bacharelado em Engenharia de Controle e Automação, do Instituto Federal Catarinense, Campus Luzerna, como requisito para a obtenção do título de Bacharel em Engenharia de Controle e Automação, em cumprimento às exigências de sua componente curricular. Orientador: Prof. Msc. Tiago Javorani Prati

Luzerna Dezembro de 2022

#### João Peterson Scheffer

Desenvolvimento de um método e sistema para compilação e simulação de redes de petri para utilização em controladores lógicos industriais

Trabalho de conclusão apresentado à banca examinadora do curso de Bacharelado em Engenharia de Controle e Automação, do Instituto Federal Catarinense, Campus Luzerna, como requisito para a obtenção do título de Bacharel em Engenharia de Controle e Automação, em cumprimento às exigências de sua componente curricular.

Luzerna (SC), 08 de dezembro de 2022:

Prof. Msc. Tiago Javorani Prati Instituto Federal Catarinense

BANCA EXAMINADORA

**Prof. Msc. XXXXXXX**Instituto Federal Catarinense

Prof. Dr. XXXXXX

Instituto Federal Catarinense

#### **AGRADECIMENTOS**

Deixo meus agradecimentos primeiramente aos meus pais e familia, que me incentivaram, apoiaram e me custodiaram durante minha formação, espero que estejam orgulhosos
de mim. Agradecimento aos meus amigos, colegas e corpo do IFC Luzerna, que me deram
durante minha formação um ambiente não só de educação e desenvolvimento intelectual
mas também de amadurecimento, onde pude me desenvolver como pesoa e ser quem sou
hoje, resguardo assim um carinho imenso por esta época da minha vida e das pessoas que
me acompanharam.

Agradeço ao professor Thiago Javorani Prati, meu orientador, que em aula me apresentou aos conceitos principais tratados neste trabalho, quais tomei gosto e agora culmina-se em uma nova obra.

Agradeço também aos professores Ricardo Kerschbaumer, Marcelo Cendron e Ricardo Antonello, cada qual de maneiras diferentes inspiraram minha paixão pelo mundo da programação, que é meu foco atual, seja na minha profissão ou mesmo neste trabalho.

# **RESUMO**

O texto do resumo em língua portuguesa deve ser digitado em um único bloco, sem espaço de parágrafo. O resumo deve ser significativo, composto de uma sequência de frases concisas, afirmativas e não de uma enumeração de tópicos. Não deve conter citações. O espaçamento entre linhas é 1,5 e o tamanho da fonte é 12. Abaixo do resumo deve-se informar as palavras-chave (palavras ou expressões significativas retiradas do texto) ou, termos retirados de thesaurus da área. De 150 a 500 palavras.

Palavras-chave: redes de petri; automação industrial; software. .

# **ABSTRACT**

O texto do resumo em língua inglesa deve ser digitado em um único bloco, sem espaço de parágrafo. O resumo deve ser significativo, composto de uma sequência de frases concisas, afirmativas e não de uma enumeração de tópicos. Não deve conter citações diretas. Abaixo do resumo deve-se informar as palavras-chave (palavras ou expressões significativas retiradas do texto) ou, termos retirados de thesaurus da área. Até 500 palavras.

**Keywords**: petri nets; industrial automation; *software*. .

# LISTA DE ILUSTRAÇÕES

| $Figura\ 1\ -$ | Rede de petri com dois lugares e uma transição      | 28 |
|----------------|-----------------------------------------------------|----|
| Figura 2 –     | Rede de petri com transição temporizada             | 30 |
| Figura 3 -     | Configuração comedor de fichas                      | 30 |
| Figura 4 -     | Arco de reset                                       | 31 |
| Figura 5 -     | Arco negado, ou inibidor                            | 31 |
| Figura 6 –     | Lista ligada com dois nós                           | 35 |
| Figura 7 –     | Dinâmica da lista priorizada para pacotes           | 36 |
| Figura 8 –     | Processamento concorrente usando mutex              | 37 |
| Figura 9 –     | Exemplo tradução de lista de instrução e Ladder     | 43 |
| Figura 10 –    | Visão do arquivo pnet um uma analisador hexadecimal | 65 |
| Figura 11 –    | Rede de petri exemplo                               | 66 |

# LISTA DE TABELAS

# LISTA DE CÓDIGOS

| Código 1 – Inicialização de uma rede de petri de forma matricial                                                                                            | 33 |
|-------------------------------------------------------------------------------------------------------------------------------------------------------------|----|
| Código 2 — Implementação usual de callbaks em C                                                                                                             | 38 |
| Código 3 – Temporizador de 1 s em C                                                                                                                         | 36 |
| Código 4 — Seção de entrada e saída em lista de instrução                                                                                                   | 41 |
| Código 5 — Seção de entrada com multiplas saída em lista de instrução $\dots$                                                                               | 42 |
| Código 6 – Estrutura C da matriz                                                                                                                            | 45 |
| Código 7 — Codificação dos eventos de entrada                                                                                                               | 47 |
| Código 8 — Estrutura C da rede de petri $\dots \dots \dots$ | 48 |
| Código 9 — Sensibilização das transições                                                                                                                    | 52 |
| Código 10 — Processamento dos eventos de borda de entrada                                                                                                   | 53 |
| Código 11 – Sensibilização de transições pelos evet<br>nos de entrada                                                                                       | 54 |
| Código 12 – Disparo das transições                                                                                                                          | 55 |
| Código 13 – Movimentação das fichas                                                                                                                         | 57 |
| Código 14 — Elemento de transição temporizado                                                                                                               | 59 |
| Código 15 – Inserção de elemento na lista priorizada por tempo                                                                                              | 59 |
| Código 16 – Retirada de elemento na lista priorizada por tempo                                                                                              | 59 |
| Código 17 – Rotina de execução do $\it thread$ temporizador                                                                                                 | 6( |
| Código 18 – Cabeçalho da serialização de matriz $\ \ldots \ \ldots \ \ldots \ \ldots \ \ldots$                                                              | 61 |
| Código 19 — Cabeçalho de arquivo da rede de petri $\ \ldots \ \ldots \ \ldots \ \ldots \ \ldots \ \ldots$                                                   | 62 |
| Código 20 — Definição da função de compilação para lista de instrução $\ \ldots \ \ldots \ \ldots$                                                          | 66 |
| Código<br>21 – Exemplo de lista de instrução - Marcação inicial<br>$\dots \dots \dots$                                                                      | 66 |
| Código 22 — Compilação da marcação dos lugares iniciais                                                                                                     | 67 |
| Código 23 – Exemplo de lista de instrução - Sensibilização                                                                                                  | 67 |
| Código 24 — Exemplo de lista de instrução - Sensibilização com temporizador $$                                                                              | 67 |
| Código 25 — Compilação da sensibilização das entradas                                                                                                       | 68 |
| Código 26 – Exemplo de lista de instrução - Disparo das transições                                                                                          | 70 |
| Código 27 — Compilação dos arcos condicionais e movimentação das fichas                                                                                     | 70 |
| Código 28 — Exemplo de lista de instrução - Saídas                                                                                                          | 72 |
| Código 29 – Compilação para as codições de saída                                                                                                            | 72 |

#### LISTA DE ABREVIATURAS E SIGLAS

PLC Programmable logic controller.

CLP Controlador lógico programável.

WYSIWYG What you see is what you get.

Vscode Visual studio code.

build Processo de compilação em lote de arquivos fonte para criação de um

programa executável ou biblioteca.

IEEE Instituto de Engenheiros Eletricistas e Eletrônicos.

POSIX Conjunto de padrões definidos pela IEEE que especifica uma interface

comum para sistemas operacionais Unix-like, visando garantir a porta-

bilidade de aplicativos entre diferentes plataformas.

Desktop Computadores de uso pessoal como computadores de mesa e Notebo-

oks's.

Bindings Definições de funções, tipos e outras estruturas programáticas que ma-

peiam diretamente com suas contrapartidas em outro ambiente/sistemas/linguagem de programação. Efetivamente é uma definição de intemperabilidade para que linguagens possam reutilizar código escrito

em outras linguagens de programação, ou ainda outros ambientes de

execução de código.

Framework Conjunto de ferramentas base que auxiliam no desenvolvimento de

aplicações.

Plugin Conjunto de códigos que estendem a funcionalidade de um programa,

como uma biblioteca.

Repositório Neste contexto se trata do armazenamento de um projeto de software

com seus arquivos fonte de forma versionada

callback Função programática que é executada automaticamente após uma ação

ser concluída sem que o usuário chame a manualmente.

array Do inglês, uma lista indexada de valores.

thread Do inglês, uma linha, que representa uma linha de execução de código

paralela/concorrente a execução normal de um programa.

thread safe — Representa código que é capaz de ser executado de forma paralela sem que hajam condições de acesso concorrente e possibilidade de corrupção ou alteração indevida de de memória.

mutex Conceito que representa a capacidade de uma instância de executar código paralelo e é requisitado e liberado para diversas instancias concorrentes, quem possui o *mutex*, pode executar, quem não possui não executa, criando um jogo de execução em turnos.

lista ligada Do inglês, lista conectada. Uma lista onde elementos não possuem posição numérica, mas são ligados uns aos outros via um ponteiro de referência para o próximo elemento da lista.

buffer Espaço utilizado para manipulação intermediária de algo entre o inicio e fim de u processo. Um armazenamento temporário.

# SUMÁRIO

| 1       | INTRODUÇÃO                                                                 | 21         |
|---------|----------------------------------------------------------------------------|------------|
| 1.1     | JUSTIFICATIVA                                                              | 22         |
| 1.2     | OBJETIVOS                                                                  | 23         |
| 1.2.1   | Objetivo geral                                                             | 23         |
| 1.2.2   | Objetivos específicos                                                      | 23         |
| 2       | METODOLOGIA                                                                | <b>2</b> 5 |
| 2.1     | A BASE DO PROJETO                                                          | 25         |
| 2.2     | REDE DE PETRI                                                              | 26         |
| 2.3     | COMPILADOR DE LISTA DE INSTRUÇÃO                                           | 26         |
| 2.4     | PUBLICAÇÃO                                                                 | 26         |
| 3       | FUNDAMENTAÇÃO TEÓRICA                                                      | 27         |
| 3.1     | REDES DE PETRI                                                             | 27         |
| 3.1.1   | Tipos de Redes de petri                                                    | 28         |
| 3.1.2   | Delimitação de uma rede para uso industrial                                | 29         |
| 3.1.3   | Implementação de redes de petri em software                                | 32         |
| 3.1.4   | Limitações e comportamento                                                 | 33         |
| 3.1.4.1 | Disparos simultâneos                                                       | 33         |
| 3.1.4.2 | Entradas                                                                   | 34         |
| 3.2     | ESTRUTURAS DE DADOS E EXECUÇÃO DE CÓDIGO                                   | 34         |
| 3.2.1   | Listas ligadas                                                             | 35         |
| 3.2.2   | Filas priorizadas                                                          | 35         |
| 3.2.3   | $Threads \ e \ mutex \ locks$                                              | 36         |
| 3.2.4   | $\it Threads \ e \ callbacks \ \ldots \ \ldots \ \ldots \ \ldots \ \ldots$ | 37         |
| 3.2.5   | Temporização                                                               | 39         |
| 3.3     | PROGRAMAÇÃO DE PLC'S                                                       | 40         |
| 3.3.1   | IEC 61131-3                                                                | 40         |
| 3.3.2   | Lista de instrução                                                         | 40         |
| 3.3.3   | Lista de instrução e Ladder                                                | 42         |
| 3.3.4   | Referência WEG TPW04                                                       | 43         |
| 4       | DESENVOLVIMENTO                                                            | <b>4</b> 5 |
| 4.1     | BIBLIOTECA C                                                               | 45         |

| 4.1.1 | Representação matricial                           | 45         |
|-------|---------------------------------------------------|------------|
| 4.1.2 | Checagem                                          | 49         |
| 4.1.3 | Sensibilização                                    | <b>52</b>  |
| 4.1.4 | Disparo                                           | 53         |
| 4.1.5 | Temporização e assincronia                        | 58         |
| 4.1.6 | Serialização de matrizes                          | 61         |
| 4.1.7 | Serialização da Rede de petri                     | 62         |
| 4.2   | ALGORITMO DE COMPILAÇÃO PARA LISTA DE INSTRUÇÃO . | 64         |
| 5     | CONCLUSÃO                                         | <b>7</b> 5 |
|       | REFERÊNCIAS                                       | 77         |

# 1 INTRODUÇÃO

Quando se trata de sistemas dinâmicos como um todo há sempre um interesse de criar, entender, prever e controlar estes sistemas, e no âmbito de sistemas discretos orientados a eventos <sup>1</sup>, como alguns sistemas industriais de automação de interesse neste trabalho, há várias metodologias, práticas, tecnologias e linguagens de programação capazes de trabalhar estes sistemas de forma a alcançar resultados esperados, cada qual possuí vantagens e desvantagens.

No âmbito industrial, para realizar o controle e automação de processos, comumente são utilizados controladores lógicos programáveis, PLC's, e estes podem ser programados para realizar as funções desejadas e para isso empregam a utilização de linguagens de programação como:

- Ladder, que representa um processo linear de forma visual inspirada em lógica de contatos <sup>2</sup>.
- Lista de instrução, que é uma linguagem escrita, não visual, e que representa um fluxo de operações com base em comandos de texto.
- Grafcet, uma linguagem visual que representa um processo em forma de fluxo com base em passos de um nó para outro nó do processo, onde cada nó é uma instrução de ação e cada passo é dado conforme uma transição atrelada a um evento.

Ainda há outras tecnologias que possuem outras formas abstratas e abordagens diferentes, sendo a linguagem *Ladder* um exemplo de aproveitamento de conhecimento e simplicidade, porém há casos onde certas ferramentas não apresentam melhor desempenho e eficiência dado certos tipos de especificações, em especial no caso do *Ladder*, onde controle de estado <sup>3</sup> e paralelismo <sup>4</sup> são conceitos difíceis de serem implementados.

Para resolver algum destes problemas, condições e situações como as mencionadas, emprega se o uso de linguagens de modelagem lógica capazes de definir um sistema desejado, representa lo como um sistema discreto orientado a eventos, onde podem se empregar abstrações lógicas adequadas a situação ou ainda prover ferramentas para realizar análise como alcançabilidade, significando se existem casos passíveis de serem realizados apesar da intenção contrária do projeto, se o sistema é reinicializável, significando a possibilidade do sistema de retornar a um estado inicial sempre, dentre outras análises complexas.

Neste contexto há varias linguagens de modelagens lógicas, como autômatos, máquinas de *Moore*, dentre outras linguagens lógicas, que podem ser usadas para abstrair alguns

Sistemas que trabalham com valores discretos como ligado e desligado por exemplo e possuem mudança de estado conforme eventos também discretos.

<sup>&</sup>lt;sup>2</sup> Uma forma simples de programação baseada na representação visual de lógica de contato, como vista em diagramas com relés e contatoras.

<sup>&</sup>lt;sup>3</sup> Armazenamento e lógica do estado/situação atual de um processo

<sup>&</sup>lt;sup>4</sup> Capacidade de unir dois fluxos de trabalho com razões de trabalho diferentes em um ponto definido.

conceitos como os citados anteriormente e nesse contexto apresenta se então a rede de Petri (IZHIKEVICH, 2011), que é capaz de reproduzir conceitos abstratos como paralelismo, sincronia, mutualidade exclusiva, etc., de forma simples bem como vários outros conceitos comuns de lógica e aritmética e conceitos ainda exclusivos, intrínsecos a si própria.

As redes de Petri são assim um instrumento de maior formalidade e podem representar conceitos abstratos, fazendo da mesma uma ótima ferramenta para automação e controle de sistemas a eventos discretos, e que cada vez mais vem sendo estudada em meio acadêmico quanto a sua utilização como mecanismo de modelagem de processos, modelagem de sistemas criticos (LEVESON; STOLZY, 1987) e (GHEZZI et al., 1991), e prevenção de deadlocks <sup>5</sup> (KAID et al., 2015).

Visto o um grau de interesse acadêmico, aplicações em situações bem definidas, significando se tratar de uma ferramenta adequada, e ainda a utilidade industrial, dado sua ótima representação de processos e liberdade de abstração, pode-se dizer que há interesse prático em redes de petri, porém o emprego real deste tipo de tecnologia é mínimo, existem metodologias (MOREIRA; BASILIO, 2014) que por exemplo, propõem a funcionalidade de interpretação de redes de petri de alto nível para código Ladder capaz de ser então usado em PLC's para uso industrial, função implementada em projetos como PetriLab (SOUZA, 2015), porém não trazem integração real, uma boa experiência de programação ou mesmo boas práticas de design de software, características que são valorizadas e por vezes indispensáveis para o processo de desenvolvimento, integração, teste, manutenção e melhoria contínua de sistemas e controle de sistemas em meio industrial, em outras palavras, há necessidade de ferramentaria adequada e atrativa as empresas e desenvolvedores destes sistemas industriais bem como difusão da utilização de redes de petri como ferramenta de uso geral e específico.

#### 1.1 JUSTIFICATIVA

Sendo apresentado o estado atual de disseminação de utilização de redes de petri, tanto geral como industrial, justifica se a criação de ferramentaria necessária ao desenvolvimento de aplicações. Aplicações tais que implementam necessariamente formas visuais de representação e edição, bem como simulação e observação do funcionamento destas redes e ainda a funcionalidade de transformar estas redes em programas prontos para utilização em meio industrial.

A capacidade de transformar tais redes em programas industriais, compreende se como o processo de compilação, onde a rede de petri é traduzida através de um algoritmo proposto para o resultado de saída, sendo esta saída código puro, arquivos digitais ou programas prontos pra utilização com PLC's industriais.

Singularidade em processos onde o sistema entra em um ponto de parada de forma imprevista e não possuí forma de autocorreção, permanece ou parado em falha ou em repetição contínua de uma única instrução.

# 1.2 OBJETIVOS

# 1.2.1 Objetivo geral

Desenvolvimento de um ambiente de definição, simulação e compilação e armazenamento de rede de petri voltadas para utilização em meio industrial, PLC's.

# 1.2.2 Objetivos específicos

- Desenvolvimento de uma biblioteca implementada em linguagem C que deve implementar os seguintes pontos:
  - Estrutura de dados.
  - serialização de dados para armazenamento.
  - Capacidade de checagem e validação.
  - $-\,$  Capacidade de execução normal e temporizada de forma assíncrona.
- Desenvolvimento de algoritmos de compilação de redes de petri para os seguintes alvos:
  - Lista de instrução, em formato de texto para a referência PLC WEG TPW04.

#### 2 METODOLOGIA

Este trabalho será de grande parte um processo de desenvolvimento por parte do autor de maneira autônoma, dado que os tópicos de programação e alguns detalhes de implementação são por natureza de livre implementação, tornando este trabalho por grande parte um trabalho exploratório.

#### 2.1 A BASE DO PROJETO

Para implementação do sistema proposto, é necessário entender que deseja-se que os resultados do sistema, as redes de petri, sejam acessíveis em várias plataformas, visando uma generalização. Entende se assim que deve se ter um motor capaz de executar estas redes e seu funcionamento em suas plataformas, em computadores de uso geral, e em sistemas embarcados torna se atrativo uma implementação na forma de uma biblioteca em linguagem C, pois a maioria dos sistemas embarcados funcionam com toolchains¹ em C, e também pois abre as portas para que outros ambientes e linguagens de programação utilizem esta biblioteca, isto é possível pelo fato de C ser de baixo nível e seguir padrões estabelecidos de arquivos e execução de código no geral, tornando possível que linguagens de mais alto nível possam fazer bindings² para com a biblioteca.

Mais ainda, outro motor importante é a capacidade de executar tais redes em plataformas distintas, onde a portabilidade da biblioteca em C torna se difícil, como os PLC's
comentados anteriormente, onde a lista de instrução é mais comum do que a linguagem
C, podendo a lista de instrução, tanto pela difusão quanto pelo baixo nível de abstração,
um candidato preferido para alvo de compilação, assim a rede pode ser editada, simulada
e testada em um computador de uso geral por exemplo e então compilada para lista de
instrução, que deve ser gerada de forma a garantir funcionalidade igual a da biblioteca
em C.

O baixo nível de abstração da lista de instrução ainda possibilita que esta seja usada como fonte para compilação posterior pelas ferramentas de terceiros para suas plataformas alvo, abrindo mais funcionalidade para esse tipo de sistema de trabalho, por exemplo, lista de instrução é comumente mapeada para *Ladder* de forma intercambiáveis, ou seja, bastante útil.

Para implementação da representação das redes, da dinâmica e da compilação será utilizada a linguagem C bem como um sistema de compilação para a biblioteca usando ferramentas básicas no padrão POSIX, em especial do projeto GNU (GNU, 2023) sendo elas o compilador *gcc*, e para *build* o *make*.

Conjunto de ferramentas que possibilitam a programação e manuseio de programas para uma plataforma e/ou arquitetura de computador específica.

<sup>&</sup>lt;sup>2</sup> Sistema onde se mapeia de forma diretas funções, variáveis e definições de código entre sistemas/ambientes de programação diferentes, possibilitando interoperabilidade entre elas

#### 2.2 REDE DE PETRI

Quanto a definição do tipo de rede de petri adotada neste trabalho, serão adotadas redes de petri com extensões específicas e utilidade geral e industrial, garantindo maior flexibilidade no design. A definição destas extensões e os detalhes de implementação serão discutidos e embasados conforme trabalhos anteriores bem como a experiência empírica do autor.

### 2.3 COMPILADOR DE LISTA DE INSTRUÇÃO

Lista de instrução é um tipo de programação relativamente difundida e portanto bem generalizada, mas ainda assim há diferenças entre fabricantes. Em vista disto a implementação do compilador de rede de petri para lista de instrução irá utilizar a referência de lista de instrução da fabricante WEG para o PLC TPW04 (WEG, 2015), sendo este um modelo amplamente utilizado em meio industrial e também de fácil acesso em educacional e acadêmico. Futuros trabalhos podem partir da mesma referência para implementação de compiladores para mais arquiteturas de PLC's, dada que as diferenças de implementação são pequenas entre tipos de plataformas e fabricantes diferentes devido a norma IEC 61161-3 (International Electrotechnical Commission, 2013).

# 2.4 PUBLICAÇÃO

Todo o trabalho desenvolvido nesta obra será versionado e disponibilizado como o repositório "pnet" (PETERSON, 2023) via Github (GITHUB, 2023), sob a licença domínio público MIT (MIT, 2023).

# 3 FUNDAMENTAÇÃO TEÓRICA

Nesta introdução teórica, será abordado o funcionamento básico das redes de petri e alguns dos principais tipos existentes. É importante compreender esses conceitos fundamentais antes de prosseguir para a implementações de *software*, pois definirá a ideia de funcionamento e implementação, onde poderá se observar a utilidade das redes, possíveis abstrações e comportamentos desejados, como os de interesse abordados na introdução.

Também conceitos relacionados a implementação de *software* e outros conceitos adjacentes necessários ao entendimento do desenvolvimento das partes integrantes do trabalho.

#### 3.1 REDES DE PETRI

As redes de petri são uma poderosa ferramenta para a modelagem e análise de sistemas concorrentes e paralelos. Introduzidas por Carl Adam Petri em 1962 (PETRI, 1962), as redes de petri fornecem uma representação gráfica e formal para descrever a dinâmica de sistemas complexos, permitindo a análise de propriedades importantes, como comportamento temporal, concorrência e paralelismo.

Uma rede de petri é composta por lugares, transições, fichas e arcos. Cada componente tem um papel fundamental na representação e modelagem do sistema.

- Lugares: Os lugares representam estados ou condições do sistema. Eles são representados por círculos em um diagrama de rede de petri. Os lugares podem conterfichas, que são unidades discretas que refletem o estado atual do sistema.
- Transições: As transições representam eventos ou ações que podem ocorrer no sistema. Elas são representadas por retângulos em um diagrama de rede de petri. Para que uma transição seja disparada, é necessário que todos os lugares de entrada estejam marcados com pelo menos um ficha.
- Fichas: As fichas são unidades discretas que se movem entre os lugares em resposta à ocorrência das transições. Elas refletem o estado atual do sistema e representam o fluxo de controle. Quando uma transição é disparada, ela consome as fichas dos lugares de entrada e produz novas fichas nos lugares de saída.
- Arcos: Os arcos conectam os lugares às transições e vice-versa. Eles indicam as relações de dependência entre os elementos do sistema. Existem dois tipos de arcos: arcos de entrada, que conectam lugares a transições, indicando que os lugares são precondições para a ocorrência da transição, e arcos de saída, que conectam transições a lugares, indicando que as transições produzem fichas nos lugares de saída. Estes arcos podem ainda conter pesos, desta forma, um arco de entrada pode retirar mais de uma ficha de um lugar, ou um arco de saída pode inserir várias fichas em outro lugar.

Figura 1 – Rede de petri com dois lugares e uma transição



O funcionamento de uma rede de petri ocorre através de um processo chamado de disparo. Quando todas as condições de precondições de uma transição são satisfeitas, a transição é disparada, consumindo as fichas dos lugares de entrada e produzindo fichas nos lugares de saída correspondentes. Esse processo de disparo ocorre de forma determinística, onde as transições só podem ser disparadas quando todas as condições são atendidas.

Em mais detalhes, as fichas só se movem de um lugar ao outro caso uma transição aconteça, e ela só irá acontecer se o arco for atendido, por exemplo, o arco de peso de saída só irá permitir que a transição ocorra caso a quantidade de fichas no lugar de origem for maior ou igual ao peso, efetivamente colocando uma condição de disparo. Quando todas as condições atreladas a uma transição são atendidas, tal transição diz se sensibilizada.

### 3.1.1 Tipos de Redes de petri

Existem vários tipos de redes de petri, cada um com suas características e aplicabilidades específicas. Alguns dos principais tipos são:

- Redes de petri coloridas (JENSEN, 1997): Nesse tipo de rede, além das fichas, são utilizadas cores para representar diferentes propriedades ou atributos dos elementos do sistema. Isso permite uma modelagem mais expressiva, onde as cores das fichas podem influenciar o comportamento das transições.
- Redes de petri temporizadas (ZHOU; HRúZ, 2007): Nesse tipo de rede, são adicionadas informações de tempo às transições e arcos. Essas informações podem incluir atrasos, tempo de execução ou intervalos de tempo específicos para a ocorrência de eventos. Isso possibilita a análise de propriedades temporais e a simulação de sistemas baseados em tempo.

- Redes de petri estocásticas (HILLSTON, 2009): Nas redes de petri estocásticas, são incorporadas probabilidades às transições e arcos, permitindo a modelagem de sistemas com comportamento probabilístico. Essas redes são úteis para a análise de sistemas em que a ocorrência de eventos é incerta ou aleatória.
- Redes de petri priorizadas (ZHOU; HRúZ, 2007): Redes onde as transições tem a
  propriedade de serem priorizadas para disparo em relação ao outras, úteis em caso
  de disparo simultâneo de uma ou mais transições.
- Redes de petri de alto nível (ISO, 2000): As redes de petri de alto nível são uma extensão das redes de petri tradicionais, que permitem uma representação mais abstrata e simplificada dos sistemas. Elas são úteis para modelar sistemas complexos e lidar com a explosão combinatória que pode ocorrer em redes de petri tradicionais. Abstrações como lugares compartilhados, macro transições e hierarquia de sub-redes são utilizadas para simplificar a representação e análise desses sistemas.

Esses são apenas alguns dos principais tipos de redes de petri, e cada um deles oferece recursos e abordagens específicas para a modelagem e análise de sistemas. O uso correto e adequado desses tipos de redes de petri depende das características do sistema a ser modelado e das propriedades que se deseja analisar.

#### 3.1.2 Delimitação de uma rede para uso industrial

As redes de petri são particularmente interessantes pois apresentam um bom grau de flexibilidade e capacidade de abstração, podendo comportar lógicas completas, simples e robustas, então é de interesse delimitar o tipo de rede que deverá ser trabalhada de forma que possamos ter as funcionalidades e características para modelagem de sistemas, como preferência para sistemas de automação industrial.

O software Petrilab (SOUZA, 2015) abre o caminho com uma delimitação interessante e prática para uso industrial, onde se propõe a utilização de métodos de entrada e saída para rede de petri, condições lógicas que permitem o acionamento de transições e os tipos de arcos desejados, sendo eles os de peso e arcos negados. Baseado neste trabalho podemos construir e reavaliar algumas decisões sobre a definição da rede de petri.

Particularmente no uso industrial a utilização de temporização é indispensável para qualquer tipo de processo, então esse é um pré-requisito inegociável. Em nome da implementação multi plataforma e considerando também precisão, será definido que as transições temporizadas trabalhem com milissegundos em vez de microssegundos. Um exemplo de transição temporizada pode ser visto na figura 2.

Os arcos com peso são úteis porém é fácil encontrar situações onde algumas abstrações se fazem convenientes, como é o caso de realizar a retirada de todos as fichas de um lugar de uma só vez, um comedor de fichas, figura 3, cujo trabalho é exatamente este. Conforme a lógica desejada, tal comportamento pode ser também atrelado a um outro tipo de arco,

Figura 2 – Rede de petri com transição temporizada



o arco de *reset* (reinicialização) (DUFOURD; FINKEL; SCHNOEBELEN, 1998), não presente na implementação do Petrilab, onde após o disparo de uma transição o mesmo irá remover todos as fichas do lugar atrelado, sendo essa relação não direcional, como nos arcos de peso. Um exemplo de arco de *reset* pode ser visto na figura 4.

Figura 3 – Configuração comedor de fichas



Ainda há mais um tipo de arco de interesse, por vezes pode ser necessário a criação de lógicas de exclusão mútua, ou de verificação simples, podendo vir a utilizar um arranjo lógico de lugares e transições dedicado, tendo a função de detectar quando não há fichas em lugar, comportamento que pode ser alcançado com um único arco, o arco de negado, também chamado de arco inibidor, que somente permite o disparo de uma transição

Figura 4 – Arco de reset



quando não há fichas em um determinado lugar.

Figura 5 – Arco negado, ou inibidor



Tal arco permite abstrações mais simples mas também possibilita que nossa rede de petri seja computacionalmente completa (ZAITSEV, 2014), ou seja, teoricamente ela pode vir a computar qualquer problema, propriedade útil quando desejamos que nossa rede de petri seja capaz de modelar qualquer tipo de sistema visando a automação industrial, e também que isso seja feito de forma conveniente.

Ainda para o uso industrial devemos nos perguntar como podemos interagir com a rede de petri em *software* com o processo/sistema atrelado, para isso necessitamos de entradas e saídas. As entradas devem ser atreladas as transições, pois são estas que movem o

estado da rede.

A forma como as entradas são atreladas é baseada no seguinte fato, se uma rede de petri é disparada continuamente, em tempo real, toda transição é sempre executada quando possível, e se atrelarmos o estado binário de uma entrada à capacidade de disparo de uma transição, a mesma pode ser levada ao disparo várias vezes por ciclo, podendo ocasionar um comportamento indesejado, portanto se propõe a utilização de eventos instantâneos para ativação. No caso de entradas discretas, estes eventos são as bordas, de subida ou descida, que acontecem quando a entrada muda de estado e duram por apenas um ciclo de execução, garantindo disparos únicos de transições, sendo na subida, descida, ou ambos, da entrada.

No Petrilab são dadas como entradas também, de certa forma, condições lógicas não baseadas em eventos. Na implementação aqui proposta, iremos descartar essas condições em nome da simplicidade, pois esse tipo de funcionalidade pode ser alcançado por uma abstração com alguns lugares e transições extras, ficando a critério do autor da rede.

As saídas podem se referir ao estado interno da rede, ou seja, as fichas nos lugares, então é de livre escolha a forma como podemos retirar a informação da rede de como acionar as saídas. Propõe-se a utilização de comparações do tipo maior ou igual entre um número de fichas e a quantidade de fichas em determinado lugar, tal condição é binária e será atrelada a uma saída externa.

Nada impede também a utilização da própria quantidade de fichas seja a saída, caso deseje se uma saída numérica, que pode ser acessada diretamente do lugar da rede de petri.

### 3.1.3 Implementação de redes de petri em software

Não nos atrelando a definição formal matemática das redes de petri, é necessário abordar a representação dessas redes, de maneira computacional na forma de *software*, e existem dois jeitos clássicos de realizar essa representação, de forma relacional ou matricial.

A forma relacional trata lugares e transições como vértices de um gráfico direcionado, e em código há uma lista de lugares e transições e uma lista de relações, onde cada relação é atrelada a um lugar e transição e possui metainformação sobre o tipo de relação, como o próprio tipo, que pode ser um arco direcionado de peso, um arco negado ou de reset. Esse tipo de representação é mais compreensível e intuitiva, porém pode sofrer impactos de desempenho, pois para realização de verificações e disparos da redes é necessário percorrer estas listas para cada tipo de operação distinta.

A outra forma, matricial, declara-se uma matriz de arcos, onde as linhas são os lugares e as colunas as transições, e o valor em determinada posição determina a quantidade de fichas a serem movidas, como no caso de arcos de peso. Este tipo de implementação é mais simples e rápida, pois não há necessidade de percorrer as relações, que é uma operação

de tempo linear, porém a representação matricial cresce quadraticamente conforme mais lugares e transições são atrelados, aumentando consumo de memória e armazenamento.

Dado que o tamanho de redes de petri não são usualmente grandes, e que há prioridade de processamento sobre memória, será escolhida a representação matricial para implementação em *software* da rede de petri proposta. Tal representação acomoda tranquilamente os tipos de arcos propostos, bem como temporização e de entrada e saída.

Código 1 – Inicialização de uma rede de petri de forma matricial

```
1 pnet_t *pnet = pnet_new(
2
       pnet_arcs_map_new(1,2,
            -1,
3
 4
       ),
5
       pnet_arcs_map_new(1,2,
 6
7
                 0,
8
9
       ),
       NULL,
10
       NULL,
11
       pnet_places_init_new(2,
12
            1, 0
13
       ),
14
       NULL,
15
       NULL,
16
       NULL,
17
       NULL,
18
       NULL
19
20);
```

Fonte: Do autor.

#### 3.1.4 Limitações e comportamento

Seja qual for o paradigma de implementação escolhido, motor de execução a ser implementado, há algumas situações a serem abordadas, detalhes de implementação que irão afetar o comportamento geral de execução da rede e que devem ser delimitados e abordados de forma a garantir equivalência das implementações.

#### 3.1.4.1 Disparos simultâneos

Quando duas transições estão sensibilizadas simultaneamente, no momento do disparo qual deve ser o comportamento? Há algumas formas de pensar sobre tal dilema, uma possível solução seria a utilização de rede de petri priorizadas em que uma transição é priorizada sobre outra, garantindo assim que apenas umas delas dispare por vez.

Outra maneira é garantir a propriedade de uma transição por disparo, parando o

processamento após o primeiro disparo, que será realizado na transição sensível de menor índice, o que pode por vezes causar ambiguidade sobre qual transição será disparada antes.

Para esta implementação será utilizada a segunda opção devido a simplicidade e ao fato das redes serem temporizadas, de entrada/saída e serem executadas em tempo real, diminuindo naturalmente a probabilidade que haja colisão de sensibilidade entre uma ou mais transições ao mesmo tempo, devido as entradas virem de um sistema físico real em tempo real. Delimita se assim um aviso de ambiguidade para este tipo de implementação e a necessidade de se garantir apenas uma transição por disparo na rede.

Tal discussão também se estende as transições temporizadas, que ao serem executadas de forma assíncrona podem vir a se sobrepor sobre outras transições sensibilizadas, novamente nosso comprometimento irá garantir que apenas uma aconteça por vez.

#### 3.1.4.2 Entradas

Como discutido anteriormente, as entradas são dadas como eventos atrelados a transições, e com a mesma preocupação de disparos simultâneos existem dois casos a serem considerados, transições com múltiplos eventos de entrada, e várias transições que utilizam a mesma entrada.

Para fins de simplificação será reforçado que cada transição seja atrelada unicamente a uma entrada e vice-versa, garantindo que disparos simultâneos causados pela mesma entrada nunca ocorram. Caso a mesma entrada seja atrelada a várias transições pode haver ambiguidade, pelo fato das transições dependerem do mesmo evento no mesmo ciclo de execução.

Casos onde deseja-se tal comportamento podem ser facilmente e seguramente implementados utilizando se múltiplas transições e lugares auxiliares, deixando assim essa abstração de responsabilidade do autor da rede de petri.

# 3.2 ESTRUTURAS DE DADOS E EXECUÇÃO DE CÓDIGO

Quando trata-se de programação de bibliotecas e artifícios¹ de código, ou seja, código que será usado por terceiros, é necessário sempre se ter em mente que a implementação seja além de funcional, bem documentada e que seja robusta a erros, conceito que envolve mas não é limitado a, tratamento de erros e exceções, visibilidade parcial da implementação, que esconde parte da funcionalidade evitando uso indesejado por parte do usuário, ou ainda a utilização de testes unitários, que testam partes individuais da biblioteca de forma automatizada. Então é necessário entender alguns conceitos de organização e execução de código por que garantirão nossas expectativas de funcionamento bem como irão garantir a robustez desejada a biblioteca.

Pequena parte, fragmento, de um todo

## 3.2.1 Listas ligadas

Listas ligadas, ou listas encadeadas são estruturas de dados parecidas com listas normais, que são conjuntos de dados de mesmo tipo dispostos de forma sequencial por um índice numérico, já as listas ligadas não são. Elas ganham a denominação de lista pois cada elemento guarda dentro de si mesmo uma referência para o próximo elemento da lista, de forma encadeada, daí o nome desse tipo de lista. Suas vantagens em relação as listas normais são que elas não tem espaço de memória predefinido e alinhado, elas podem crescer indefinidamente e de forma desorganizada com relação ao alinhamento de memória, são flexíveis, mas com o custo de terem de ser percorridas toda vez para se encontrar o elemento em dada posição, operação que tem um tempo de execução proporcional ao tamanho da lista, enquanto que a lista normal faz isso de forma indexada, por isso é de tempo constante.

Figura 6 – Lista ligada com dois nós



Para implementações em C a lista é basicamente um conjunto de nós, que são estruturas que guardam dentro de si um valor de tipo desejado, e um ponteiro de memória para outro nó na lista.

# 3.2.2 Filas priorizadas

Filas priorizadas são como filas normais, listas onde o elemento a ser inserido é colocado a frente e retirado ao fim, ou seja, quem entra primeiro sai primeiro. A lista priorizada implementa a funcionalidade de inserir elementos baseado em uma medida de privilégio que pode ser um número arbitrário ou uma medida como tempo de vida, uma escala de erro e etc. Da mesma forma que na fila normal, elementos são retirados ao fim da fila.

Podem ser implementados a partir de listas indexadas ou listas ligadas.

Um exemplo de fila priorizada seria uma onde processamos pacotes de entrega, sejam eles pacotes de protocolos de comunicação ou pacotes postais, onde temos uma lista ligada com dois pacotes de prioridade baixa, zero por exemplo, agora suponha que desejase processar um pacote com maior urgência, para tanto, se insere o pacote com seu valor de prioridade igual a um, assim a fila passa ter como último elemento o novo pacote, de prioridade maior.

int prioridade = 0;
int value;
node \*next;

int prioridade = 1;
int value;
node \*next;

Figura 7 – Dinâmica da lista priorizada para pacotes

#### 3.2.3 Threads e mutex locks

O processamento paralelo e concorrente são abordagens utilizadas para melhorar a eficiência e desempenho dos sistemas computacionais, permitindo a execução simultânea de múltiplas tarefas. No processamento paralelo, várias tarefas são executadas ao mesmo tempo, utilizando recursos computacionais simultaneamente. Isso é comumente aplicado em sistemas com múltiplos núcleos de processamento, onde diferentes tarefas podem ser executadas em paralelo, aumentando a capacidade de processamento total do sistema.

Uma das formas de alcançar o processamento paralelo é por meio do uso de threads. As threads são unidades básicas de execução em um programa, representando fluxos independentes de controle que podem executar tarefas em paralelo. Uma aplicação pode conter várias threads, permitindo a execução simultânea de diferentes partes do código. As threads compartilham o mesmo espaço de memória, o que facilita a comunicação e a troca de dados entre elas.

A execução concorrente de *threads* também pode trazer desafios. Um problema comum é a ocorrência de *race conditions*, que acontecem quando duas ou mais *threads* acessam ou modificam uma região de memória ou variável compartilhada simultaneamente, resultando em comportamento imprevisível do programa. Isso ocorre quando as operações

das threads não são adequadamente sincronizadas, resultando em um conflito entre as operações concorrentes. Para evitar race conditions, são utilizados mecanismos de sincronização, como mutex locks, mutex locks são mecanismos de exclusão mútua, onde uma thread obtém um lock para bloquear o acesso de outras threads a um recurso compartilhado até que seja liberado. Isso garante que apenas uma thread possa acessar o recurso por vez, evitando race conditions. Os mutex locks são usados para proteger regiões críticas do código onde ocorrem operações compartilhadas, garantindo a consistência e prevenindo conflitos entre as threads.



Figura 8 – Processamento concorrente usando mutex

## 3.2.4 Threads e callbacks

Em utilização normal, quando um thread precisa acessar um recurso protegido, ela solicita o bloqueio do mutex, aguardando até que ele esteja disponível. Isso impede que outras threads acessem o recurso simultaneamente, evitando race conditions. Uma vez obtido o bloqueio do mutex, a thread pode acessar o recurso de forma segura, realizar as operações necessárias e, ao finalizar, deve desbloquear o mutex para permitir que outras threads possam acessar o recurso. Dessa forma, o uso adequado de mutexes permite controlar a execução e garantir a sincronização correta entre as threads, evitando conflitos e garantindo a integridade dos dados compartilhados.

A utilização de callbacks é uma forma comum de comunicação entre dois threads em programação concorrente. Um callback é uma função ou bloco de código que é passado como argumento para outra função. Ao utilizar callbacks para comunicação entre threads, um thread pode registrar um callback que será acionado quando uma determinada condição for atendida ou quando algum evento específico ocorrer. Essa abordagem é particularmente útil em situações em que um thread precisa notificar outro thread sobre algo

que aconteceu. Por exemplo, considere um cenário em que um thread A está aguardando a conclusão de um processamento realizado pelo thread B. Em vez de ficar bloqueado a execução, esperando o thread B terminar, o thread A pode registrar um callback que será executado pelo thread B assim que o processamento for concluído. Dessa forma, o thread A pode continuar executando outras tarefas sem ficar preso esperando a finalização do thread B.

A utilização de *callbacks* nesse contexto permite uma abordagem assíncrona de comunicação, na qual os *threads* podem operar de forma independente e notificar uns aos outros quando necessário. Além disso, a flexibilidade dos *callbacks* permite que o código de um *thread* possa ser modificado ou estendido sem afetar diretamente o código do outro *thread*, tornando o sistema mais modular e adaptável. No entanto, é importante considerar questões de sincronização e concorrência ao utilizar *callbacks* entre *threads*. Mecanismos adequados, como *mutex locks* ou semáforos², devem ser empregados para garantir a correta sincronização e evitar problemas de *race conditions* ou acesso simultâneo a recursos compartilhados.

Tratando de implementações, mais específico em C, callback são funções e tem formatos distintos, e geralmente são passados como ponteiros de função. Normalmente callbacks são dados como funções genéricas que não retornam nada e recebem informações genérica usado void\*.

Código 2 – Implementação usual de callbaks em C

```
1 typedef struct{
    int value;
    char *name;
4 } custom_data_t;
6 void callback(void *args){
    custom_data_t *data = (custom_data_t*)args;
    printf("Name: %s, value: %d\n", data->name, data->value);
8
9 }
10
11 int main(){
    custom_data_t data = {
12
      .value = 42,
13
      .name = "The answer"
14
    };
15
16
    register_callback(callback, &data);
17
18 }
```

Fonte: Do autor.

Mecanismo de sincronização análogo a um semáforo de trânsito, onde só se executa código quando houver um "sinal verde", que é dado pelo outro ator de execução

A implementação desses conceitos em C pode ser alcançada através da biblioteca pthreads (LINUX FOUNDATION AND MICHAEL KERRISK, 2021), biblioteca de paralelismo/concorrência padrão em sistema que suportam POSIX e a mais difundida das implementações, geralmente presente em sistemas operacionais por padrão.

## 3.2.5 Temporização

Temporizadores são um tópico interessante em si, pois em sistemas embarcados geralmente têm-se acesso a temporizadores de hardware, que são precisos e fáceis de usar, porém para sistemas operacionais que não são de tempo real e não possuem temporizadores de hardware disponíveis a nível de usuário, acabam se tornando um leve problema. A maneira como pode-se implementar um temporizador seria na criação de uma execução paralela/concorrente de código onde pode-se contar o tempo através de chamadas de função que retornam o tempo passado desde um ponto arbitrário, como a execução do sistema ou da própria aplicação, valor que pode ser impreciso em cima do fato de ainda estar sendo contado continuamente em uma rotina concorrente, que não é garantida de ser executada em tempo real. O resultado é que podemos alcançar timers com boa acurácia, na casa dos milissegundos, porém com baixa precisão, com bastante variação em torno do valor original de contagem.

A utilização de execução concorrente de código pelo uso de *threads* recaí sobre a capacidades da maioria dos sistemas, sejam computadores pessoais ou sistemas embarcados, portanto trata-se de uma boa escolha para implementação genérica. Há ainda a possibilidade de se utilizar temporizadores de *hardware* de forma condicional conforme a arquitetura envolvida, o que adiciona um grau de complexidade ao sistema de compilação e deve ser alterado pontualmente para cada sistema a ser suportado, portanto um boa escolha para projetos mais especializados e menos genéricos.

Para realizar a medição utiliza se a função clock da biblioteca padrão C. Essa função retorna o tempo de CPU decorrido desde o início do programa em ciclos de *clock*, que podem ser convertidos em unidades de tempo, como milissegundos. A função clock oferece uma forma padronizada de obter-se a temporização, independentemente do sistema operacional, garantindo maior portabilidade e precisão, no entanto, é importante ressaltar que a função clock mede o tempo de CPU decorrido e não segue de fato o tempo real.

A utilização normalmente recai sobre o paradigma de obter se o tempo inicial e calcular o tempo decorrido subtraindo chamadas seguintes a função até que a diferença entre elas seja maior ou igual ao tempo desejado.

Código 3 – Temporizador de 1 s em C

```
1 #include <time.h>
2
3 #define CLOCK_TO_MS(x) ((int)((x) * 1000 / CLOCKS_PER_SEC))
4
```

```
5 int now = CLOCK_TO_MS(clock());
6
7 if((now - CLOCK_TO_MS(clock())) >= 1000){
8  printf("1 second passed");
9 }
```

# 3.3 PROGRAMAÇÃO DE PLC'S

## 3.3.1 IEC 61131-3

A norma IEC 61131-3 (International Electrotechnical Commission, 2013) é um padrão internacional amplamente utilizado para a programação de controladores lógicos programáveis (PLCs) e sistemas de automação industrial. Ela foi desenvolvida pela Comissão Eletrotécnica Internacional (International Electrotechnical Commission - IEC) e define uma linguagem comum e um ambiente de programação para PLCs.

Uma das principais características da norma IEC 61131-3 é a definição de cinco linguagens de programação padronizadas para PLCs. Essas linguagens são: diagrama de contatos (LD - Ladder Diagram), lista de instruções (IL - Instruction List), texto estruturado (ST - Structured Text), diagrama de blocos de função (FBD - Function Block Diagram) e gráfico sequencial (SFC - Sequential Function Chart). Cada linguagem tem sua própria sintaxe e semântica, permitindo aos programadores escolher a linguagem mais adequada para a tarefa em questão.

Além das linguagens de programação, a norma IEC 61131-3 também estabelece diretrizes para a organização e estruturação de programas PLC, o uso de variáveis, a manipulação de tempo, a comunicação com dispositivos externos e a interface com o *hardware* do PLC. Essas diretrizes garantem a interoperabilidade entre diferentes PLCs e facilitam a portabilidade de programas entre diferentes sistemas.

Outro aspecto importante da norma IEC 61131-3 é a definição de blocos de função reutilizáveis. Esses blocos são unidades de *software* que encapsulam uma funcionalidade específica e podem ser usados em diferentes programas e projetos. Eles promovem a modularidade, a reutilização de código e a manutenção simplificada, permitindo um desenvolvimento mais eficiente e flexível de sistemas de automação.

A norma também aborda aspectos relacionados à segurança funcional, especificando requisitos para garantir a integridade e confiabilidade dos sistemas de automação. Ela define categorias de segurança e diretrizes para a programação de funções de segurança, como monitoramento de entradas, lógica de falha segura e supervisão de sistemas.

#### 3.3.2 Lista de instrução

Uma das linguagem padronizadas para programação de PLC's é a lista de instrução que consiste em uma sequência de instruções que especificam as operações a serem executadas

pelo PLC, afim de controlar o comportamento do sistema desejado. Dentre os comandos mais comuns presentes destacam-se o LD (Load), OUT (Output), SET, AND e os relacionados a bordas de subida e descida, temporizadores e aos comandos de MPS (Master Program Select), MRD (Master Reset Display) e MPP (Master Program Present).

O comando LD (Load) é utilizado para carregar um valor ou estado lógico em uma variável interna ou memória do PLC. Essa instrução é fundamental para a atribuição de valores a variáveis e para a leitura de entradas do sistema.

O comando out (Output) é responsável por escrever um valor lógico em uma saída específica do PLC. Ele controla a ativação ou desativação de dispositivos externos conectados ao sistema automatizado.

O comando set é utilizado para definir um estado lógico específico em variáveis internas do PLC. Esse comando é útil para inicializar variáveis ou para acionar estados específicos em situações de controle.

O comando AND realiza a operação lógica "e"entre dois ou mais valores. Ele permite realizar condições de controle mais complexas ao combinar múltiplas entradas.

Além desses comandos, as bordas de subida, LDP, e descida LDF, são utilizadas para controlar a ocorrência de eventos específicos no sistema. As bordas de subida são acionadas quando um sinal lógico muda de zero para um, enquanto as bordas de descida são acionadas quando o sinal lógico muda de um para zero. Essas bordas são fundamentais para a detecção de transições e acionamento de ações em tempo real.

Já os temporizadores são componentes essenciais na programação de controladores lógicos programáveis. Eles permitem a criação de atrasos programados e temporizações precisas, sendo utilizados para controlar o tempo de espera entre eventos e para executar ações em momentos específicos. Os temporizadores podem ser configurados para atrasos fixos ou variáveis, dependendo das necessidades do sistema.

Por fim, os comandos MPS (Memory push), inserir em memória, MRD (Memory read), ler da memória, e MPP (Memory pop), retirar da memória, são utilizados em sistemas com múltiplos saídas. O comando funciona basicamente como uma pilha, onde o resultado de uma operação pode ser inserido com (MPS), re lido quantas vezes forem necessárias com (MRD) e por fim lido a última vez enquanto ao mesmo tempo se remove da pilha, (MPP). Esses comandos são importantes para a coordenação e controle de programas em sistemas complexos, pois possibilitam atrelar várias saídas a um mesmo conjunto de condições.

Estes comandos basicamente dão ordem de execução a um programa, toda seção de execução é feita primeiramente carregando um valor (LD), usando o para alguma lógica, e or fim direcionando o resultado para uma saída, como por exemplo OUT, RST ou comandos como MOV e ADD.

Um exemplo de seção seria abrir uma variável e escrever seu valor em uma saída.

Código 4 – Seção de entrada e saída em lista de instrução

```
1 LD XO
2 OUT YO
```

Ou em várias usando MPS, MRD e MPP.

Código 5 – Seção de entrada com multiplas saída em lista de instrução

```
1 LD XO
2 MPS
3 OUT YO
4 MRD
5 SET D2O1
6 MPP
7 MOV K1 D2O4
```

Fonte: Do autor.

## 3.3.3 Lista de instrução e Ladder

Uma das características da programação de PLC's é que pelo padrão IEC 61161-3, plataforma de programação de desses PLC's implementam um conjunto dessas linguagens, sendo mais usuais a linguagem Ladder e de lista de instrução, e normalmente também fabricantes dão a habilidade aos programadores de alterar em tempo real entre as linguagens, isso é possível pois todas são relativamente simples e normalmente podem ser traduzidas a partir da lista de instrução.

Assim, contanto que se tenha um método de tradução de uma dada linguagem para lista de instrução e vice-versa, há possibilidade do fabricante implementar essas tradução em tempo real.

A linguagem Ladder em especial é um ótimo candidato pois as operações mais básicas são facilmente mapeadas. No exemplo da figura 9, temos um simples programa de PLC, onde pode-se observar um exemplo de implementação da relação entre as linguagem Ladder e lista de instrução. Observa-se que para as ramificações Ladder de entrada, utilizamos somente operações lógicas, uma operação lógica "ou"e outra "e, negada", assim, para entradas, a maior parte dos casos será uma expressão lógica em termos das entradas.

Para saída iremos utilizar as instruções de memória, inserindo o valor após a última operação lógica e depois lendo até a última entrada, onde o valor é retirado e a declaração do programa termina.

Podem haver outras implementações e variações desta apresentada, porém serve de ilustração para a capacidade de se converter entre as linguagens, provando em torno também que quando se tem um programa em lista de instrução, podemos facilmente alcançar outros paradigmas e linguagens, se tornando um ótimo alvo base para compilação

Lista de instrução

LD XO
OR X1
ANDI X3
MPS
OUT Q0
MRD
OUT Q1
MPP
OUT Q2

Figura 9 – Exemplo tradução de lista de instrução e Ladder

de programas.

#### 3.3.4 Referência WEG TPW04

A definição de lista de instrução é padronizada, porém, fabricantes optam por realizar implementações que abrangem a norma em grande parte mas que por vezes apresentam diferenças e particularidades. Da empresa WEG, o modelo de PLC TPW04 (WEG, 2015),pode apresentar algumas diferenças em relação à definição da norma IEC 61131-3. Essas diferenças podem ocorrer devido a recursos específicos do CLP como otimizações de desempenho ou necessidades específicas do ambiente de automação em que o TPW04 é utilizado. É importante ressaltar que as diferenças podem variar dependendo da versão do firmware do TPW04. Aqui estão algumas possíveis diferenças.

- Conjunto de Instruções: O TPW04 pode ter um conjunto de instruções específico que vai além das instruções definidas na norma IEC 61131-3. A WEG pode adicionar instruções adicionais para atender a requisitos específicos do CLP ou para fornecer recursos extras aos programadores.
- Sintaxe e Semântica: Embora o TPW04 siga os conceitos e princípios gerais da norma IEC 61131-3, pode haver diferenças sutis na sintaxe e nomenclatura das instruções. Essas diferenças podem ser introduzidas para simplificar a programação ou otimizar o desempenho do CLP.
- Recursos Específicos: O TPW04 pode oferecer recursos específicos que não estão incluídos na norma IEC 61131-3. Isso pode incluir recursos avançados de temporização, contadores específicos, funções de comunicação específicas ou suporte para dispositivos periféricos exclusivos da WEG.

Como este projeto visa a implementação prática para com o modelo TPW04, está será a referência a ser seguida para o desenvolvimento do compilador, respeitando as instruções específicas WEG, porém tentando utilizar as instruções mais triviais e gerais de forma a facilitar futuras adaptações para outras referências de outros modelos e fabricantes.

## 4 DESENVOLVIMENTO

Começando pela base, é desenvolvido primeiramente a biblioteca em C, onde serão definidos a estrutura, checagem, serialização e armazenamento e dinâmica/simulação da rede, que irão compor a biblioteca e permitir que ela seja de uso genérico conforme a especificação.

Ao fim será desenvolvido o algoritmo que se utilizará da estrutura de dados da rede petri para realizar a compilação para lista de instrução, referência WEG TPW04 (WEG, 2015).

#### 4.1 BIBLIOTECA C

No desenvolvimento da biblioteca em C, como qualquer outro tipo de programa, uma das primeiras coisa a serem feitas é a definição de uma estrutura de dados, como discutido previamente, iremos implementar a rede de petri em código utilizando a representação matricial. Implementada a funcionalidade das matrizes podemos definir a estrutura da nossa rede de petri e por fim, definir os algoritmos que irão realizar a funcionalidade desejada da biblioteca.

Como padrão serão definidas para cada estruturas de dados um struct que será inicializado a partir de uma função de criação de forma dinâmica, retornando um ponteiro para a estrutura.

## 4.1.1 Representação matricial

Primeiramente define-se a estrutura primitiva da matriz, a qual irá comportar o tamanho e array dinâmico de memória de duas dimensões. Escolhe se size\_t pois este comporta o maior valor de tamanho disponível na arquitetura a ser compilada para e int para os valores da matriz, visto que todas as matrizes que irã comportar a representação matricial podem ter valores positivos e negativos.

Código 6 – Estrutura C da matriz

```
1 typedef struct{
2    size_t x;
3    size_t y;
4    int **m;
5 }pnet_matrix_t;
```

Fonte: Do autor.

Com a definição da matriz, pode-se pensar nas definições dos arcos, para isso em uma matriz A, n por m, com n sendo a quantidade de linhas e lugares, e m sendo a quantidade de colunas e transições, arcos de tipo peso podem ser dados como no seguinte exemplo.

$$A = \begin{bmatrix} -1 & 0 \\ 1 & -1 \\ 0 & 2 \end{bmatrix}$$

Onde pode-se ver três lugares e duas transições, onde a primeira transição retira uma ficha do primeiro lugar e a coloca no segundo lugar, e a segunda transição retira do segundo lugar e coloca duas fichas no terceiro lugar.

Uma das problemáticas que irá surgir é a que a rede de petri permite dois arcos de peso de um mesmo lugar para uma mesma transição, e nossa representação não a comporta, pois para retirar e colocar uma ficha, devemos marcar como zero, mas zero significa nenhum arco, para tanto, divide-se a matriz em duas, uma matriz  $A_p$  para arcos de peso positivo, que colocam fichas em lugares e outra matriz  $A_n$  para arcos negativos, que retiram fichas de lugares. Assim teremos para o mesmo exemplo:

$$A_p = \begin{bmatrix} -1 & 0 \\ 0 & -1 \\ 0 & 0 \end{bmatrix}$$

$$A_n = \begin{bmatrix} 0 & 0 \\ 1 & 0 \\ 0 & 2 \end{bmatrix}$$

Assim temos a distinção entre arcos não existentes e dois arcos distintos entre um lugar e uma transição.

Para os arcos negados e de *reset* a representação é mais simples, pois devemos apenas marcar com um, para um arco, ou zero, para sem arco. Exemplo de um arco negado da primeira transição para com o terceiro lugar.

$$A_i = \begin{bmatrix} 0 & 0 \\ 0 & 0 \\ 1 & 0 \end{bmatrix}$$

E um exemplo de um arco *reset* que após o disparo da primeira transição zera todos os lugares.

$$A_r = \begin{bmatrix} 1 & 0 \\ 1 & 0 \\ 1 & 0 \end{bmatrix}$$

Tendo já os arcos podemos agora definir o estado inicial da rede de petri, que são as fichas iniciais que alguns lugares possuem conforme o design do autor da rede. É relativamente simples, apenas uma matriz unidimensional com os valores para cada lugar. Exemplo de lugar inicial de um de três lugares.

$$P_i = \begin{bmatrix} 1 & 0 & 0 \end{bmatrix}$$

Também para as transições é necessário definir o valor da temporização dos arcos, que também é uma matriz unidimensional. Exemplo de um tempo de 5 s para a primeira de duas transições.

$$T = \begin{bmatrix} 5000 & 0 \end{bmatrix}$$

Para as entradas da rede define-se, como nos arcos, uma matriz de relação entre entradas e transições. Assim define-se uma matriz de n linhas, as entradas, por m colunas, as transições, e também deve se definir uma forma de codificar em um número inteiro, devido a implementação da matriz, o tipo de evento a ser usado, para isso define-se uma enumerador C com as três possíveis bordas propostas, de subida, descida e ambas.

Código 7 – Codificação dos eventos de entrada

Fonte: Do autor.

Observa-se que definimos também um evento nulo, zero, que simboliza nenhuma relação, e um evento max, que tem como função apenas a checagem, que será vista mais a frente. Note também como a borda de ambos, any\_edge, é a soma do valor das outras duas bordas, assim que utilizar a biblioteca para criação de redes de petri pode utilizar do operador lógico "ou"para indicar verbosamente ambas as bordas, pnet\_event\_any\_edge = pnet\_event\_pos\_edge | pnet\_event\_neg\_edge.

Um exemplo de matriz onde deseja-se que a transição um seja acionada somente quando haja uma borda de subida da entrada dois.

$$I_{in} = \begin{bmatrix} 0 & 0 \\ 1 & 0 \end{bmatrix}$$

As saídas são definidas como a quantidade de fichas maior ou igual necessárias a ativação, portanto nenhuma codificação é necessária, apenas o valor numérico, assim tempos uma matriz de n linhas, lugares, or m colunas, saídas. Um exemplo caso desejarmos que a saída um seja acionada quando o número e fichas no lugar três seja maior ou igual a quatro.

$$O_{out} = \begin{bmatrix} 0 & 0 \\ 0 & 0 \\ 4 & 0 \end{bmatrix}$$

Com estas definições podemos então definir a nossa estrutura em C da rede de petri, listagem de código 8.

Código 8 – Estrutura C da rede de petri

```
1 struct pnet_t{
      // size
      size_t num_places;
3
      size_t num_transitions;
4
      size_t num_inputs;
5
      size_t num_outputs;
6
7
8
      // maps
9
      pnet_matrix_t *neg_arcs_map;
10
      pnet_matrix_t *pos_arcs_map;
11
      pnet_matrix_t *inhibit_arcs_map;
12
      pnet_matrix_t *reset_arcs_map;
      pnet_matrix_t *places_init;
13
      pnet_matrix_t *transitions_delay;
14
15
      pnet_matrix_t *inputs_map;
      pnet_matrix_t *outputs_map;
16
17
      // validation
18
      bool valid;
19
20
      // net state
21
      pnet_matrix_t *places;
22
      pnet_matrix_t *sensitive_transitions;
23
24
      // input edges state
25
      pnet_matrix_t *inputs_last;
26
27
      // output values
28
29
      pnet_matrix_t *outputs;
30
      // async
31
32
      pnet_callback_t function;
      void *user_data;
33
      pthread_t thread;
34
      pthread_mutex_t lock;
35
      transition_queue_t *transition_to_fire;
36
37 };
```

Fonte: Do autor.

Nota-se a presença de mapas, estes são as matrizes até então mencionadas, mesmo as fichas inicias e temporização, o restante dos valores são pertinentes ao processamento e execução de rede e serão explorados posteriormente.

## 4.1.2 Checagem

Dada a estrutura da rede de petri a primeira coisa a ser pensada é que as matrizes são de livre preenchimento e portanto podem haver definições de redes de petri inválidas, comportamento que deve ser devidamente checado para que a rede de petri possa ser minimamente executável, para tanto implementa-se um algoritmo de checagem baseado nas definições propostas, na capacidade de execução e viabilidade do ponto de vista de software.

Destaca-se um problema em particular, que seria a mínima rede válida permitida, pois diz respeito as capacidades permitidas para execução da rede sem erros e garantindo a comportamento desejado dadas as especificações. Para isto vamos considerar quais seriam as matrizes opcionais da rede de petri. Entradas e saídas são opcionais por não interferem no funcionamento da rede. Temporização também não é requerida para todas as transições.

A quantidade de lugares e transições é dada pelos tamanhos das matrizes de arcos e da marcação inicial das fichas, assim ao menos uma dessas deve ser não nula, e mesmo que não hajam arcos, uma rede pode ainda ser válida, com somente lugares e transições, somente um lugar e uma transição ou ainda um único lugar ou única transição. Para uma rede de petri com uma única transição devemos ter ao menos uma das quatro matrizes de arcos, e dado que estas também definem a existência de lugares, pois não podem haver zero linhas em uma matriz, então uma rede de apenas uma transição não é possível neste tipo de implementação, sendo assim a menor rede viável é a de apenas um lugar, a qual pode ser definida apenas pela matriz de marcação inicial.

É interessante ainda expandir essa definição se escolhermos que nossa rede de petri seja passível de execução, sendo necessário assim uma transição e um tipo de arco capaz de alterar a quantidade de fichas, arcos de peso ou arcos de reset. Assim então definimos a rede mínima para esta implementação uma rede com no mínimo a matriz de marcação inicial e ao menos umas das duas matrizes de arco, de reset ou de peso, as quais podem ser matrizes de minimamente de um por um, sendo assim as redes mínimas as redes de um lugar e uma transição com um arco de peso ou arco de reset.

Dadas as especificações e a análise da rede mínima viável, os seguintes pontos devem ser reforçados para criação das redes:

- Consistência na quantidade de lugares e transições nas matrizes. Erro se qualquer matriz não for adequada.
- Existência de somente números positivos na matriz de arcos positivos. Outros va-

lores serão zerados.

- Existência de somente números negativos na matriz de arcos negativos. Outros valores serão zerados.
- Existência de somente um ou zero na matriz de arcos negados e arcos de *reset*. Outros valores serão considerados como um.
- Existência da matriz de marcações iniciais. Erro se não detectado.
- Números negativos de fichas na inicialização dos lugares, interpretado como erro.
- Existência de pelo menos um arco de pelo menos um tipo um de arco, garantindo dinâmica na rede. Erro se não detectado.
- Números negativos na temporizações das transições, interpretado como erro.
- Existência de apenas eventos válidos de entrada. Corrigido para evento nulo, zero, caso contrário.
- Existência de apenas um evento de entrada por uma transição. Erro caso contrário.

Ao final da checagem, uma variável binária é marcada como um dentro da estrutura da rede de petri, indicando que esta pode-se executada normalmente.

É importante ressaltar que a criação da rede e sua checagem é referente a esta implementação em específico dada as necessidades de *software*, tipo de rede proposto e limitações, não necessariamente refletindo os conceitos formais de rede de petri mínima e implicações relacionadas.

Também referente a implementação são realizados os seguintes testes práticos na implementação da rede de petri para garantir funcionalidade não só de execução mínima mas também das outras capacidades que irão serão posteriormente desenvolvidas nas seções seguintes.

- Teste para retorno não nulo e código de erro ok
- Teste para criação com todos os argumentos
- Teste para todos os argumentos nulos
- Teste para apenas lugares, sem arcos
- Teste para autocorreção de valores
- Teste para arcos sem peso ou negados
- Teste para múltiplas entradas em uma única transição

- Teste de sensibilização
- Teste de disparo sem arcos de peso ou reset
- Teste de disparo com entradas para matriz de entrada nulo
- Teste de disparo com entradas nulas para matriz de entrada não nula
- Teste de disparo de uma única transição com evento de entrada do tipo none
- Teste de falha no disparo
- Teste de transição com arco de peso, negados e reset e eventos de entrada
- Teste de arco de peso e reset
- Teste para sensibilização de transições sem matriz de entradas e entrada nula para pnet\_fire
- Teste de disparo manual
- Teste para estado de saída no início
- Teste para múltiplas transições mútuas e um arco bidirecional
- Teste para estado de saída no final
- Teste de rede temporizada sem passar função de callback
- Teste de acurácia de transições temporizada
- Teste de petri multi temporizadas
- Testes com redes de petri de demonstração
- Testes de serialização de matriz
- Testes de desserialização da matriz
- Comparação se matriz é igual a desserialização da serialização da matriz
- Teste para criação de matrizes grandes
- Teste matriz de um por um é igual a desserialização da serialização
- Teste de tamanho máximo factível da desserialização da serialização
- Teste de desserialização da serialização da rede de petri
- Teste de compilação de rede de petri para lista de instrução

# 4.1.3 Sensibilização

Dado uma rede petri válida, para que antes de a executarmos em tempo real, precisamos executá-la uma única vez, um disparo, e para isso necessitamos antes verificar quais transições estão sensibilizadas dado o estado atual, dado pela matriz places, visto no código 8. Para isso utilizaremos um simples algoritmo que irá varrer e comparar os mapas dos arcos de condição, que são os arcos de peso negativos e arcos negados, com o estado atual da rede.

Código 9 – Sensibilização das transições

```
1 for(size_t transition = 0; transition < pnet->num_transitions;
     transition++){
      // set transition to sensibilized
2
      pnet->sensitive_transitions->m[0][transition] = 1;
3
      for(size_t place = 0; place < pnet->num_places; place++){
4
5
           // check desensibilization
6
          if(
7
8
               // negative arcs
9
               (
                   // when there are negative arcs
10
                   (pnet->neg_arcs_map != NULL) &&
11
12
                   // when there is a neg arc for this transition/place
                   (pnet->neg_arcs_map->m[place][transition] != 0) &&
13
                   // and there is not enough tokens
14
                   ((pnet->neg_arcs_map->m[place][transition] + pnet->
15
     places ->m[0][place]) < 0)</pre>
               ) ||
16
               // inhibit arcs
17
               (
18
                   // when there are inhibit arcs
19
                   (pnet->inhibit_arcs_map != NULL) &&
20
                   // if there is a inhibit arc
21
                   (pnet->inhibit_arcs_map->m[place][transition] == 1) &&
22
                   // and place has token, then do not trigger, otherwise
23
      trigger
                   (pnet->places->m[0][place] != 0)
24
               )
25
          ){
26
27
               // desensibilize
               pnet->sensitive_transitions->m[0][transition] = 0;
28
               break;
29
30
          }
      }
31
32 }
```

Fonte: Do autor.

Na implementação primeiramente dizemos que, para determinada transição, ela se encontra sensibilizada por padrão, então verificamos se existem as matrizes dos arcos de condição e para cada tipo de arco, se existe relação entre o lugar e transição atual, e ainda, se o lugar possui a condição necessária, seja de fichas necessárias para o arcos de peso negativo ou zero fichas para o arco negado. Chegando ao final de todas as das condições, se algo não for verdadeiro a transição será marcada como não sensibilizada.

Note que a matriz das transições sensitive\_trasitions é interna a rede de petri e que a cada verificação de sensibilidade retorna várias transições sensibilizadas, que está em contradição com nossa especificação da seção 3.1.4, porém será tratada posteriormente na realização de disparos.

O ponto de entrada para realização análise de sensibilização será uma função distinta denominada pnet\_sense.

## 4.1.4 Disparo

Dada uma matriz com as transições a serem sensibilizadas é necessário ainda verificar se as transições possuem relação de entrada, caso no qual deverá ser feita uma verificação das bordas de entrada. A verificação das bordas de entrada é feita comparando se o valor atual é diferente do anterior, para isso é necessário guardar o estado anterior das entradas, seria a matriz inputs\_last dada no código 8.

O ponto de entrada para realização dos disparos será uma função distinta denominada pnet\_fire, a qual dentro si realiza uma chamada a função pnet\_sense, incorporando assim a lógica completa de disparos, exceto pelos disparos temporizados, que irão ocorrer de forma assíncrona, discutidos na próxima seção. Na função pnet\_fire, também é repassado o valor atual das entradas externas, assim pode-se verificar se houve mudança e portanto se houver um evento, uma borda, e se o evento irá contribuir para o disparo de alguma transição. Para isso verifica-se a entrada com o estado anterior e armazena se o resultado na matriz temporária edges.

Código 10 - Processamento dos eventos de borda de entrada

```
1 // run for every input given
2 for(size_t input = 0; input < pnet->num_inputs; input++){
3
      // check for pos edges
      if(pnet->inputs_last->m[0][input] == 0 && inputs->m[0][input] ==
4
     1){
          edges ->m[0][input] = pnet_event_pos_edge;
5
6
      // check for neg edges
7
      else if(pnet->inputs_last->m[0][input] == 1 && inputs->m[0][input]
8
          edges->m[0][input] = pnet_event_neg_edge;
9
10
```

```
11 }
12
13 // store last inputs
14 pnet_matrix_copy(pnet->inputs_last, inputs);
```

Primeiramente para cada transição, se assume que ela está sensibilizada, então se verifica-se para cada uma se existe um mapa de entrada, caso não ela permanece sensibilizada, caso não, verifica-se se a mesma possui um evento nulo atrelado, caso qual pulamos e a transição continua sensibilizada, caso contrário, se a borda for a mesma especificada pela matriz de entrada então a mesma continua sensibilizada. Caso nenhuma dessas condições não for verdadeira, a transição é marcada como dessensibilizada.

Código 11 – Sensibilização de transições pelos evetnos de entrada

```
1 for(size_t transition = 0; transition < pnet->num_transitions;
     transition++){
      transitions -> m[0][transition] = 0;
2
3
      // if input map is null all transitions can occurr
4
      if (pnet -> inputs_map == NULL) {
5
           transitions ->m[0][transition] = 1;
 6
           continue;
7
      }
8
9
10
      for(size_t input = 0; input < pnet->num_inputs; input++){
11
12
13
           // if event type is none mark as firable, run until the end of
       inputs
           if(pnet->inputs_map->m[input][transition] == pnet_event_none){
14
               transitions -> m[0][transition] = 1;
15
16
          }
           // if the transitions has an event. When a single event is
17
     found then this event must be satisfied,
           // otherwise the transition stay desensibilized, so we exit
18
     the loop when we reach it
19
               // using the & operator to check edge type, see
20
     pnet_event_t for why
               if(pnet->inputs_map->m[input][transition] & edges->m[0][
21
     input]){
                   transitions -> m[0][transition] = 1;
22
               }
23
               else{
24
                   transitions -> m[0][transition] = 0;
25
```

```
26 }
27
28 break;
29 }
30 }
```

É interessante notar como no código 7, as bordas de subida e descida são distintas mas a borda ambos é a soma do valor delas, basta realizar a operação lógica "e"para poder determinar o resultado, se o valor da matriz de entrada for igual ao detectado, então "e"será verdadeiro, e se o valor da matriz for para ambos, se a detectada for por exemplo de subida, então teremos 0x03 & 0x01, que será também verdadeiro, como pode ser visto na linha 21 do código 11.

Tendo agora duas matrizes, a matriz de transições sensibilizadas pelos arcos condicionais e a matriz de transições sensibilizadas pelas entradas, portanto as transições que devem ser disparadas serão o resultado lógico "e"de ambas as matrizes.

Código 12 – Disparo das transições

```
1 // transitions that are sensibilized and got the event
2 pnet_matrix_t *transitions_able_to_fire = pnet_matrix_and(
     input_event_transitions, pnet->sensitive_transitions);
3
4 // fire transitions
5 for(size_t transition = 0; transition < pnet->num_transitions;
     transition++){
      if(transitions_able_to_fire->m[0][transition] == 1){
6
                // firable transition
7
          if(
               (pnet->transitions_delay == NULL) ||
8
                // not timed
               (
9
                   (pnet->transitions_delay != NULL) &&
10
                   (pnet->transitions_delay->m[0][transition] == 0)
11
                // but instant
              )
12
          ) {
13
              // move and callback
14
              pnet_move(pnet, transition);
15
              if(pnet->function != NULL) pnet->function(pnet, transition
16
      , pnet->user_data);
              break;
17
                // only one instant transitions
18
```

No código 12 linha 2 vemos a operação lógica "e", e temos então a matriz de transições para executar, porém, conforme mencionado na seção 3.1.4, destas, podemos executar ao máximo uma transição, então a partir do momento que temos uma transição não temporizada que está apta, esta será executada e o irá quebrar o laço de repetição, assim outras transições não serão executadas. Aqui vê se a ambiguidade de que a primeira transição que for declara tem prioridade, devido ao laço de repetição, essa é a justificativa para implementação de prioridade de transições, que vem do tipo de rede de petri priorizada, porém, novamente, a escolha deste tipo de implementação é justificada pela finalidade de aplicação, assumindo que esse tipo de ambiguidade não será de grande problema.

A função pnet\_move realiza a movimentação das fichas conforme os arcos atuadores, os arcos de peso negativo e positivo e o arco reset, a forma como a movimentação é alcançada pode ser dada em termos matemáticos, para os arcos de peso, de uma única matriz A com arcos positivos e negativos, e uma matriz T que representa a transição a ser disparada, o resultado da movimentação das fichas pode ser dado como:

$$T_{t=0} = \begin{bmatrix} 1 & 0 & \dots \end{bmatrix}$$

$$P_n = P + T \cdot A^T$$

Onde a é uma célula, m o índice da coluna,  $A^T$  a transposta de, P a matriz dos lugares atuais e  $P_n$  os novos lugares após o disparo.

Esse algoritmo funciona de forma adequada, exceto pelo fato da multiplicação matricial ser extremamente custosa para grandes matrizes. Como sabe-se o tipo de matriz que temos, podemos começar extraindo a coluna da matriz de arcos que representa a única transição a ser disparada, diminuindo drasticamente o cálculo envolvido. Com essa nova matriz unidimensional, pode-se simplesmente realizar a soma diretamente aos lugares da rede de petri.

Para a implementação desse trabalho necessitamos primeiramente extrair as colunas da matriz de arcos positiva e negativa e então realizar a soma.

$$a = A(m = t)$$

Onde  $A_n$  são os arcos negativos e (m = t) representa a coluna da transição t, sendo  $a_n$  um vetor de m elementos e uma única linha.

$$P_n = \left(P + a_n^T + a_p^T\right) \odot \neg a_r$$

Onde  $a_n$  são a coluna transição dos arcos negativos, p positivos e r de reset,  $\odot$  é a multiplicação de elemento a elemento, diferente da multiplicação matricial normal, e  $\neg$  a negação lógica dos elementos matriciais. Em código temos:

Código 13 – Movimentação das fichas

```
1 // for weighted arcs, only if at least one of them is not null
2 if(pnet->pos_arcs_map != NULL || pnet->neg_arcs_map != NULL) {
      pnet_matrix_t *weighted_matrix = NULL;
4
      pnet_matrix_t *pos_weighted_matrix = NULL;
5
      pnet_matrix_t *neg_weighted_matrix = NULL;
6
7
      // extract the values from the arcs for the transition
8
      if (pnet->pos_arcs_map != NULL)
9
          pos_weighted_matrix = pnet_matrix_extract_col(pnet->
10
     pos_arcs_map, transition);
      if (pnet->neg_arcs_map != NULL)
11
          neg_weighted_matrix = pnet_matrix_extract_col(pnet->
12
     neg_arcs_map, transition);
13
      // sum the pos and neg values or returning only the non null maps
14
      if(pos_weighted_matrix != NULL && neg_weighted_matrix != NULL){
15
          weighted_matrix = pnet_matrix_add(pos_weighted_matrix,
16
     neg_weighted_matrix);
17
          pnet_matrix_delete(pos_weighted_matrix);
          pnet_matrix_delete(neg_weighted_matrix);
18
19
      else if(neg_weighted_matrix != NULL)
20
          weighted_matrix = neg_weighted_matrix;
21
      else
22
          weighted_matrix = pos_weighted_matrix;
23
24
      // finnaly adding the difference in tokens for the places,
25
     effectly moving the tokens
      pnet_matrix_t *buffer = pnet_matrix_add(places_res,
26
     weighted_matrix);
      pnet_matrix_copy(places_res, buffer);
27
      pnet_matrix_delete(weighted_matrix);
28
      pnet_matrix_delete(buffer);
29
30 }
31
```

```
32 // for reset arcs
33 if(pnet->reset_arcs_map != NULL){
      // get places to reset for a given transition
34
      pnet_matrix_t *places_reset = pnet_matrix_extract_col(pnet->
35
     reset_arcs_map, transition);
36
37
      // negate so its a 0 for a place to be reset, that way we can then
      multiply it with the places element by element, zeroing out the
     places where needed
      pnet_matrix_t *places_reset_neg = pnet_matrix_neg(places_reset);
38
      pnet_matrix_delete(places_reset);
39
40
      pnet_matrix_t *buffer = pnet_matrix_mul_by_element(places_res,
41
     places_reset_neg);
      pnet_matrix_copy(places_res, buffer);
42
43
      pnet_matrix_delete(places_reset_neg);
44
      pnet_matrix_delete(buffer);
45
46 }
```

# 4.1.5 Temporização e assincronia

Nota-se que no código 12 que a transição só é de fato disparada caso não haja matriz de temporização ou quando a temporização para a mesma for zero, porém quando há temporização, a mesma deve iniciar o processo de contagem juntamente com outras transições que possam começar a contar, isso pelo fato de que o inicio da contagem pode ocorrer de forma simultânea, somente o disparo das transições deve ser único, comportamento que pode-se tornar complexo, para tanto deve se visionar um sistema de contagem que garanta o mesmo.

Dentro deste *thread* código erá executado para efetuar a contagem das transições temporizadas quando forem sensibilizadas, porém como visto, isso pode levar a situações onde podem haver múltiplas transições simultâneas. Para lidar com esse aspecto e facilitar a organização e contagem destas transições, é implementado uma estrutura de dados específica, uma fila priorizada.

A implementação da fila é feita da seguinte forma, uma lista ligada que guarda elementos especiais, responsáveis pela metainformação de cada transição temporizada, e que trabalha a inserção e retirada dos elementos conforme a prioridade da contagem atual, de forma que os elementos com menor quantidade de tempo restante fiquem ao fim da lista, pois tem maior prioridade, assim podemos apenas verificar o último elemento da lista para saber se o mesmo deve ser disparado ou não, sem necessidade de checar o restantes dos elementos.

Os elementos da lista ligada são transições, que são da seguinte forma:

Código 14 – Elemento de transição temporizado

```
1 typedef struct{
2   size_t transition;
3   int start;
4   int delay;
5 }transition_t;
```

Fonte: Do autor.

Onde guardam-se o número da transição, o tempo inicial  $t_0$  onde a transição foi ativada e o tempo total de temporização t. Para realizar a inserção na lista teremos uma função transition\_queue\_push, qual insere os elementos de menor  $t_0 + t$ .

Código 15 – Inserção de elemento na lista priorizada por tempo

```
1 typedef struct{
2   size_t transition;
3   int start;
4   int delay;
5 }transition_t;
```

Fonte: Do autor.

Percorre-se a lista com um laço de repetição começando-se ao final, onde estão as transições mais próximas de executar, de volta ao inicio. Verifica-se então se a transição já está presente na lista, se sim retornar da chamada pois não há necessidade de inserção. Se o tempo final  $t_0 + t$  for menor que o da transição atual, inserir o elemento a frente daquela transição, e caso se chegar ao inicio da lista sem que não seja inserido nada, inserir ao inicio da lista como a transição mais longe de ser executada. Tal processo é assegurado pelo uso de um mutex definido para a instância da lista, evitando possíveis erros por race conditions.

Essa chamada foi usada no código 12 linha 21, onde verifica-se que a inserção é realizada para quantas transições forem sensibilizadas naquele momento.

Para retirada do transição da lista implementa-se a chamada transition\_queue\_pop, que é definida como:

Código 16 – Retirada de elemento na lista priorizada por tempo

```
1 int now = CLOCK_TO_MS(clock());
2
3 transition_t node_value = queue_value(queue->q->last, transition_t);
4 if((now - node_value.start) >= node_value.delay){
5 *transition = node_value;
6
```

```
7
    queue_node_t *node = queue->q->last;
    queue->q->last = node->prev;
8
9
    if (node->prev != NULL)
10
       node->prev->next = NULL;
11
12
13
    queue->q->size--;
14
    if (queue -> q -> first == node)
15
       queue -> q -> first = NULL;
16
17
    free(node->value);
18
    free(node);
19
20
    queue_unlock();
21
    return true;
22
23 }
```

Na linha 1 obtém se o tempo atual em milissegundos usando a função clock, que nos dá a quantidade de tempo passado desde o inicio do programa até o momento atual. Então verifica-se se o último elemento da lista é passível de ser retirado, então compara-se o tempo passado até o momento desde a inserção da transição com o tempo de temporização total. Caso tenha sido alcançado a função retorna sucesso e o elemento pode ser acessado pelo ponteiro de referência passado como parâmetro, caso contrário, o retorno é zero.

Assim a contagem no *thread*, dado pela própria estrutura da rede de petri, pode ser feita com a seguinte rotina de execução:

Código 17 – Rotina de execução do thread temporizador

```
1 while(1){
2
      pthread_testcancel();
3
      transition_t transition;
4
      if(transition_queue_pop(pnet->transition_to_fire, &transition)){
5
          pnet_sense(pnet);
6
          if(pnet->sensitive_transitions->m[0][transition.transition] ==
7
      1){
                // re check sensibility
              pnet_move(pnet, transition.transition);
8
                // FIRE!! move tokens and call callback
              if(pnet->function != NULL)
9
                   pnet->function(pnet, transition.transition, pnet->
10
     user_data);
          }
11
      }
12
13 }
```

Observa-se que a rotina é executa de forma infinita, sempre tentando retirar elementos da fila priorizada, caso não consiga, o laço simplesmente se repete. Lembrando que este processo ocorre de forma paralela ao código da rede. Caso o retorno seja verdadeiro podemos verificar se a transição ainda encontra se sensibilizada, comportamento desejado, pois se ela deixar de ser sensibilizada então não deve ser dispara-la. Caso se encontrar, então disparamos a rede e executamos o *callback* de retorno, deixando o usuário da rede atento que uma transição foi executada de forma assíncrona, fora to tempo de execução da chamada pnet\_fire.

Tal callback é definido na criação da rede, visto no código 8 juntamente a fila priorizada, uma simples função que é executada de forma assíncrona e recebe como parâmetros os dados do usuário, dados no momento da criação, e o número da transição que foi executada, que pode ser utilizado para lógicas auxiliares de escolha do usuário.

As funções de checagem de sensibilização, e de movimentação de fichas são thread safe também graças a um outro mutex localizado também na estrutura rede de petri, assim, quando as transições temporizadas ocorrerem, elas não entrarão em conflito com as transições instantâneas da execução normal.

Com essa última definição chegamos ao final da estrutura dada no código 8, onde cada elemento definido além das matrizes de definição, os mapas, é necessário para que a rede funcione como um todo.

## 4.1.6 Serialização de matrizes

A serialização de dados é bastante simples quando têm-se acesso ao baixo nível, como é o caso da linguagem C, onde podemos ler dados da memória e interpretá-los de maneira livre. Uma das formas de fazer isso é utilizando uma técnica chamada *pointer casting*, onde uma estrutura é definida e então atrelada como ponteiro a uma região de memória, funcionando para leitura e escrita, isso se torna uma forma conveniente de se serializar e desserializar dados.

As matrizes que compõe nossa rede de petri ocupam um espaço relevante de memória, portanto guardar uma matriz n por m de números inteiros ocuparia minimamente  $n \cdot m \cdot 4$  bytes, tamanho que cresce exponencialmente, porém, como a maioria das matrizes possuem muitos valores nulos, devido as relações de arcos e marcação inicial, podemos tomar vantagem disso e gravar apenas os valores não nulos, usando marcadores de linha e coluna.

Define-se assim então para cada matriz um cabeçalho de memória, onde serão contidas a quantidade de linhas e colunas e então os valores não nulos.

```
2
      uint32_t x;
      uint32_t y;
3
      uint8_t first_byte;
5 }pnet_matrix_header_t;
7 uint32_t data[] = {
8
      4, 4,
9
      0x80000000, 3, 1,
      0x80000001, 0, 64, 3, 1,
10
      0x80000003, 2, 0x7FFFFFFF, 3, 16
11
12 };
13
14 pnet_matrix_header_t *header = (pnet_matrix_header_t*)data;
```

Como estamos usando matrizes de número inteiro, será fixado o tamanho em *bits* das variáveis como uint32\_t para índices e int32\_t para os valores.

Os valores serão escritos da seguinte forma, marca-se primeiro a linha, caso haja no mínimo um valor não nulo nela, usando o índice da linha e marcando o último bit da representação binária do número, como forma de diferenciar que este número se trata do índice da linha. Assim o número a ser gravado será de  $n = c \mid 0x80000000$ , e a leitura pode ser feita como c = n & 0x80000000.

Dada a linha, cada valor da linha será um par de dois números, o índice da coluna e o valor.

Para seguinte matriz de exemplo teremos:

$$\begin{bmatrix} 0 & 0 & 0 & 1 \\ 64 & 0 & 0 & 1 \\ 0 & 0 & 0 & 0 \\ 0 & 0 & 0x7FFFFFFF & 16 \end{bmatrix}$$

$$Ser = \Big\{4, 4, 0 \\ \text{x} 800000000, 3, 1, 0 \\ \text{x} 800000001, 0, 64, 3, 1, 0 \\ \text{x} 800000003, 2, 0 \\ \text{x} 7 \\ \text{FFFFFF}, 3, 16\Big\}$$

Assim dado um *array* de memória de tipo uint32\_t \* de tamanho size\_t bytes, podemos realizar um casting conforme o código 18 para encontrar o tamanho da matiz e assim pre alocar a matriz para ler novamente os dados.

## 4.1.7 Serialização da Rede de petri

Dada a serialização das matrizes a rede de petri torna-se fácil, pois é composta de várias matrizes, deixando somente a definição de alguns metadados necessários. O cabeçalho do arquivo pode ser dado como:

Código 19 – Cabeçalho de arquivo da rede de petri

```
1 #pragma pack(push,1)
2 typedef struct{
3
       char magic[4];
       uint16_t version;
4
      uint8_t valid;
5
      uint8_t matrix_size;
6
      uint32_t size;
7
       uint32_t crc32;
8
       uint32_t num_places;
9
10
       uint32_t num_transitions;
       uint32_t num_inputs;
11
12
       uint32_t num_outputs;
13
       uint32_t neg_arcs_map_size;
14
       uint8_t neg_arcs_map_first_byte;
15
16
        * ...
17
18
        * neg_arcs_map
19
        * pos_arcs_map
        * inhibit_arcs_map
20
        * reset_arcs_map
21
        * places_init
22
        * transitions_delay
23
        * inputs_map
24
        * outputs_map
25
        * places
26
        * sensitive_transitions
27
        * outputs
28
        * inputs_last
29
31 }pnet_file_header_t;
32 #pragma pack(pop)
```

Nota-se que o primeiro membro, magic, trata-se de um identificação visual, este guarda quatro caracteres de texto de valor sempre "PNET", de forma a indicar que este arquivo é um arquivo de tipo rede de petri, seguido do número de versão do arquivo, que têm-seu caso de uso caso a partir deste valor no cabeçalho ocorra mudanças no futuro, os programas podem alterar a leitura e escrita com base no número de versão, dando flexibilidade e opções de versionamento.

Temos o valor binário de validez da rede de petri, seguido do valor de tamanho das matrizes, que como discutido anteriormente foi dado como de trinta e dois bits, 0x20, mas poderia ser dado como matrizes de dezesseis bits, 0x10, ou outros tamanhos caso a

necessidade de armazenamento de memória.

O campo size armazena o valor total de bytes ocupado pela serialização rede excepto pelo cabeçalho fixo, ou seja, a partir do campo neg\_arcs\_map\_size. Existe também a checagem CRC32 (GNU, 2012) dos dados dados serializados, exceto pelo cabeçalho, para que se possa detectar erros por corrupção de memória. E finalmente as quantidades de elementos, transições, lugares, entradas e saídas, e então as matrizes e seus tamanhos em bytes, que compõem nossa rede de petri.

Matrizes serializadas

- Matriz de arcos de peso negativos
- Matriz de arcos de peso positivos
- Matriz de arcos negados
- Matriz de arcos de reset
- Matriz de marcação inicial
- Matriz de temporização das transições
- Matriz de entrada
- Matriz de saída
- Matriz de estado atual dos lugares
- Matriz de transições sensíveis
- Matriz de estados das saídas
- Matriz do estado anterior das entradas

Como temos um número fixo de matrizes, basta realizar a leitura do tamanho de cada uma, e então a leitura da matriz em si.

Para este arquivo, dado o nome da biblioteca e do indicador no cabeçalho, sua extensão será denominada .pnet. Um arquivo preenchido tem o seguinte formato visto sobre um analisador hexadecimal:

# 4.2 ALGORITMO DE COMPILAÇÃO PARA LISTA DE INSTRUÇÃO

A compilação para lista de instrução foi baseado no método de compilação para Ladder apresentado no trabalho Petrilab (SOUZA, 2015), porém para lista de instrução, fato que é possível devido a característica de que programação Ladder pode ser gerada a partir de uma lista de instrução e vice-versa.

Como está sendo implementando um compilador para lista de instrução, sabe-se que funcionalidades de borda de eventos e temporizadores são de fácil acesso, dispensando

Figura 10 – Visão do arquivo pnet um uma analisador hexadecimal

criação de estruturas adicionais para implementação dos mesmos, e portanto podemos compilar a rede de petri de forma mais simples, mapeando estas funcionalidade diretamente conforme necessário.

Para a definição de funcionalidade, serão seguidos os passos como implementados em sumo na biblioteca em C.

- Marcação inicial
- sensibilização
- Disparo
- Lógica de saída

Cada passo será desenvolvido de forma sequencial, para ter-se esta ordem de execução de código no PLC dado. Para a definição deste compilador será utilizada como referência a lista de instrução dada pelo fabricante WEG para o PLC TPW04 (WEG, 2015).

Para compilação para código C usaremos uma função facilitadora, string\_cat\_fmt, que é uma função feita com base na biblioteca C padrão, parecida com a função printf mas que recebe uma string de formatação, a expande e concatena o resultado em um buffer de saída, assim temos ao inicio um buffer vazio, e vamos inserindo as instruções até que ao final tenhamos o programa compilado. Também iremos utilizar as constantes INITIAL\_RE, que se refere a memória M8001 que é ativa por apenas um ciclo quando o PLC é inicializado, e ALWAYS, a memória especial M8000, que é sempre ativa durante a execução do código.

Também iremos considerar a posição inicial de algumas memórias, por exemplo, as transições sensibilizadas são armazenadas em memórias binárias simples do tipo M enquanto os lugares são do tipo word, no PLC, D. Também as entradas X, saídas Y e o índice do pulo condicional J. Todas essa variáveis internas serão usadas e necessitem de

um índice numérico, e portanto, o algoritmo de compilação deverá ser ciente do índice inicial para cada uma delas, assim temos a seguinte chamada de função para realização da compilação da rede de petri para lista de instrução.

Código 20 – Definição da função de compilação para lista de instrução

```
1 char *pnet_compile_il_weg_tpw04(pnet_t *pnet, int input_offset, int
    output_offset, int transition_offset, int place_offset, int
    timer_offset, int timer_min, int jump_offset){
```

Fonte: Do autor.

Dada uma rede de petri exemplo, figura 11, vamos analisar o processo de compilação, mostrando a implementação das diferentes partes, a lista de instrução para cada componente e a implementação em código C. A rede apresenta todas as funcionalidades proposta exceto pelo arco de *reset*, e demonstra uma implementação simples de um processo de exclusão mútua utilizando arcos negados, redundante pois só há apenas uma ficha em cada lugar por vez mas demonstra a ideia.

 $x_{0} \uparrow \qquad y_{0}, > 1$  5000 ms  $x_{1} \uparrow \qquad y_{1}, > 1$   $x_{2} \uparrow \qquad x_{3} \uparrow$ 

Figura 11 – Rede de petri exemplo

Onde abaixo das transições temos o evento de borda relacionado a entrada e abaixo dos lugares temos a legenda da saída a ser acionada e a condição lógica de ativação.

Para marcação inicial é utilizada a memória especial M8001 e a instrução MOV para mover as fichas para os lugares iniciais, que são no PLC memórias do tipo *word*, a partir de dado endereço.

Código 21 – Exemplo de lista de instrução - Marcação inicial

```
1 LD M8001
2 MOV K1 D200
```

Em código C isso pode ser implementado por um laço de repetição pelos lugares da rede de petri, onde para cada lugar, se há marcação, concatena se a expressão em formato de lista de instrução no texto de saída final.

Código 22 – Compilação da marcação dos lugares iniciais

```
for(size_t place = 0; place < pnet->num_places; place++){
  if(pnet->places_init->m[0][place]){
    string_cat_fmt(buffer, "LD M%u\nMOV K%u D%u\n", BUFFER_SIZE,
    INITIAL_RE, 1, place + place_offset);
}
```

Fonte: Do autor.

A detecção de entrada e sensibilização de transições é relativamente fácil e será dada como os eventos de entrada mapeados para uma memória auxiliar que irá ser usada para movimentar as fichas. Para entradas normais temos:

Código 23 – Exemplo de lista de instrução - Sensibilização

```
1 LDP X1
2 MPS
3 OUT M31
4 MPP
5 CJ PO
```

Fonte: Do autor.

Observa-se a instrução a mais de pulo condicional tal condição é usada aqui para implementar a limitação dada à rede acerca de disparos mútuos, discutidos na seção 3.1.4. O pulo irá ocorrer nas sensibilizações pois pode-se definir o resultado de temporizações em sequência devido a arquitetura do ambiente de execução do PLC, diferente da biblioteca C, facilitando assim a definição desses disparos assíncronos. Esse pulo condicional será para uma marcação de execução, por exemplo PO, posteriormente feita no inicio da sequência de disparo.

Código 24 – Exemplo de lista de instrução - Sensibilização com temporizador

```
1 LD X0
2 OUT Y0
3
4 LD X0
5 MPS
6 OUT Y0
```

```
7 MRD
8 SET D201
9 MPP
10 MOV K1 D204
```

No disparo de transições temporizadas temos que a entrada aciona o temporizador, que quando for acionado, irá ao mesmo tempo acionar a memória de sensibilização, irá se auto reinicializar e fará o pulo condicional para os disparos.

A implementação em C das entradas e timers pode ser dada por um laço de repetição pelas transições e entradas, caso não houverem entradas, a transição sempre dispara, e caso houverem verifica-se se são temporizadas ou não, em código C temos:

Código 25 – Compilação da sensibilização das entradas

```
1 for(size_t transition = 0; transition < pnet->num_transitions;
     transition++){
    if (pnet -> inputs_map == NULL) {
2
3
      string_cat_fmt(buffer, "LD M%u\nMPS\nOUT %u\n", BUFFER_SIZE,
     ALWAYS, transition + transition_offset);
      string_cat_fmt(buffer, "MPP\nCJ P%u\n", BUFFER_SIZE, jump_offset);
4
      continue;
5
    }
6
7
    for(size_t input = 0; input < pnet->num_inputs; input++){
8
9
        (input == pnet->num_inputs - 1) &&
10
        (pnet->inputs_map->m[input][transition] == 0)
11
12
      ) {
13
        string_cat_fmt(buffer, "LD M%u\nMPS\nOUT %u\n", BUFFER_SIZE,
     ALWAYS, transition + transition_offset);
14
      else if(
15
        pnet->transitions_delay != NULL &&
16
        pnet->transitions_delay->m[0][transition]
17
      ) {
18
        switch(pnet->inputs_map->m[input][transition]){
19
          case pnet_event_pos_edge:
20
             string_cat_fmt(buffer, "LDP X%u\nSET T%u K%u\nLD T%u\n",
21
     BUFFER_SIZE, input + input_offset, transition + timer_offset, pnet
     ->transitions_delay->m[0][transition] / timer_min, transition +
     timer_offset);
             string_cat_fmt(buffer, "MPS\nOUT M%u\nMRD\nRST T%u\n",
22
     BUFFER_SIZE, transition + transition_offset, transition +
     timer_offset);
            break;
23
```

```
24
25
          case pnet_event_neg_edge:
             string_cat_fmt(buffer, "LDF X%u\nSET T%u K%u\nLD T%u\n",
26
     BUFFER_SIZE, input + input_offset, transition + timer_offset, pnet
     ->transitions_delay->m[0][transition] / timer_min, transition +
     timer_offset);
             string_cat_fmt(buffer, "MPS\nOUT M%u\nMRD\nRST T%u\n",
27
     BUFFER_SIZE, transition + transition_offset, transition +
     timer_offset);
28
            break:
29
30
          case pnet_event_any_edge:
             string cat fmt(buffer, "LDP X%u\nORP X%u\nSET T%u K%u\nLD T%
31
     u\n", BUFFER_SIZE, input + input_offset, input + input_offset,
     transition + timer_offset, pnet->transitions_delay->m[0][transition
     ] / timer_min, transition + timer_offset);
             string_cat_fmt(buffer, "MPS\nOUT M%u\nMRD\nRST T%u\n",
32
     BUFFER_SIZE, transition + transition_offset, transition +
     timer_offset);
            break;
33
        }
34
      }
35
      else if(pnet->inputs_map->m[input][transition]){
36
        switch(pnet->inputs_map->m[input][transition]){
37
          case pnet_event_pos_edge:
38
             string_cat_fmt(buffer, "LDP X%u\nMPS\nOUT M%u\n",
39
     BUFFER_SIZE, input + input_offset, transition + transition_offset);
            break:
40
41
          case pnet_event_neg_edge:
42
43
             string_cat_fmt(buffer, "LDF X%u\nMPS\nOUT M%u\n",
     BUFFER_SIZE, input + input_offset, transition + transition_offset);
44
            break;
45
          case pnet_event_any_edge:
46
             string_cat_fmt(buffer, "LDP X%u\nORP X%u\nMPS\nOUT M%u\n",
47
     BUFFER_SIZE, input + input_offset, input + input_offset, transition
      + transition_offset);
            break;
48
        }
49
      }
50
      else{
51
52
        continue;
      }
53
54
      string_cat_fmt(buffer, "MPP\nCJ P%u\n", BUFFER_SIZE, jump_offset);
55
56
      break;
```

```
57 } 58 }
```

Onde na linha 2 verifica-se há uma matriz de entrada, caso não todas as transições são sensibilizadas por padrão. Na linha 8 vemos o laço pelas entradas.

Linha 9, caso tenhamos varrido todas as entradas e não houver entrada atrelada a esta transição, então marcar a transição como sensibilizada.

Linha 15, se houver entrada e transição for temporizada, então realizamos um comparação com o tipo de entrada na linha 19. Para entrada de borda de subida, linha 21, para borda de descida 25, e para ambas bordas, linha 30.

Linha 36, caso a transição não tenha temporização mas tenha entrada, temos os mesmos casos de borda que para temporização, ma o código de lista de instrução é diferente.

E por fim, linha 51, caso encontrarmos um caso onde que não seja temporizado e nem com entrada, os laço de repetição para as entradas será continuado normalmente, até que a condição da linha 9 ative e a transição seja marcada como sempre acionada, como mencionado anteriormente.

Para a movimentação dos arcos pelo disparo primeiramente vamos verificar as condições de sensibilização, os arcos negativos e os arcos negados utilizando as instruções AND>= e AND=.

Código 26 – Exemplo de lista de instrução - Disparo das transições

```
1 LD M32
2 AND>= D201 K2
3 AND= D202 K0
4 MPS
5 ADD D200 K1 D200
6 MPP
7 SUB D201 K2 D201
```

Fonte: Do autor.

Dada a verificação, os arcos de movimentação serão acionados. No exemplo, arcos de peso negativos e positivos, que irão usar as instruções de subtração e adição para movimentação das fichas, um arco de *reset* em contra partida usaria uma instrução MOV com constante zero.

Em código C temos:

Código 27 – Compilação dos arcos condicionais e movimentação das fichas

```
1 string_cat_fmt(buffer, "P%u\n", BUFFER_SIZE, jump_offset);
2
3
```

```
4 for(size_t transition = 0; transition < pnet->num_transitions;
     transition++){
5
    string_cat_fmt(buffer, "LD M%u\n", BUFFER_SIZE, transition +
6
     transition_offset);
7
    for(size_t place = 0; place < pnet->num_places; place++){
8
9
10
        pnet->inhibit_arcs_map != NULL &&
11
        pnet -> inhibit_arcs_map -> m[place][transition]
      ) {
12
        string_cat_fmt(buffer, "AND= D%u KO\n", BUFFER_SIZE, place +
13
     place_offset);
      }
14
15
      if(
16
17
        pnet->neg_arcs_map != NULL &&
        pnet ->neg_arcs_map ->m[place][transition]
18
      ) {
19
        string_cat_fmt(buffer, "AND>= D_u K_u n", BUFFER_SIZE, place +
20
     place_offset, -pnet->neg_arcs_map->m[place][transition]);
21
    }
22
23
    if(
24
      pnet->neg_arcs_map != NULL ||
25
      pnet->pos_arcs_map != NULL ||
26
      pnet->reset_arcs_map != NULL
27
    ) {
28
      string_cat_raw(buffer, "MPS\n");
29
30
      for(size_t place = 0; place < pnet->num_places; place++){
31
32
        if(
          pnet->neg_arcs_map != NULL &&
33
          pnet->neg_arcs_map->m[place][transition]
34
35
          string_cat_fmt(buffer, "SUB D%u K%u D%u\nMRD\n", BUFFER_SIZE,
36
     place + place_offset, -pnet->neg_arcs_map->m[place][transition],
     place + place_offset);
37
38
          pnet->pos_arcs_map != NULL &&
39
          pnet->pos_arcs_map->m[place][transition]
40
41
           string_cat_fmt(buffer, "ADD D%u K%u D%u\nMRD\n", BUFFER_SIZE,
42
     place + place_offset, pnet->pos_arcs_map->m[place][transition],
     place + place_offset);
```

```
43
         if(
44
           pnet->reset_arcs_map != NULL &&
45
           pnet->reset_arcs_map->m[place][transition]
46
47
48
           string_cat_fmt(buffer, "MOV K%u D%u\nMRD\n", BUFFER_SIZE, 0,
     place + place_offset);
49
50
      string_trim_end(buffer, 4);
51
      string_cat_raw(buffer, "MPP\n");
52
    }
53
54 }
```

Na linha 1 definimos a marcação para o pulo condicional, indicando que aqui inicia-se os disparos após a detecção de entrada e temporização.

Na linha 6 começamos carregando o valor de sensibilidade da transição verificado anteriormente. Na linhas 8, um laço de repetição por todos os lugares, onde verificamos se para dado lugar e transição existe um arco de peso negativo ou arco negado, e se for o caso, adicionar uma comparação das fichas usando AND>= para o arco de peso e AND= KO para o arco negado.

Na linha 31 então iremos então adicionar as movimentações das fichas, usando as instrução de memória MPS, MRD, MPP para os vários arcos do lugar/transição, o arco de peso negativo, que usa a instrução SUB, arco positivo com a instrução ADD e de reset com MOV.

Por fim temos as saídas, as quais só utilizam a comparação entre os lugares e uma constante dada pela matriz de saída, e acionam a saída física.

Código 28 – Exemplo de lista de instrução - Saídas

```
1 LD>= D201 K1
2 OUT Y0
```

Fonte: Do autor.

Em código C temos:

Código 29 – Compilação para as codições de saída

```
if(pnet->outputs_map != NULL){
  for(size_t place = 0; place < pnet->num_places; place++){
   for(size_t output = 0; output < pnet->num_outputs; output++){
    if(pnet->outputs_map->m[place][output])
      string_cat_fmt(buffer, "LD>= D%u K%u\nOUT Y%u\n", BUFFER_SIZE,
      place + place_offset, pnet->outputs_map->m[place][output], output
```

```
+ output_offset);
6   }
7 }
8 }
```

Na linha 2 e 3, temos os laços que irão varrer os lugares e saídas, onde verificaremos a quantidade de fichas e acionaremos as saídas, na linha 4 vemos a concatenação da instrução inteira usando LD>= e OUT.

# 5 CONCLUSÃO

Dados os objetivos e aspirações deste projeto, pode-se afirmar que tanto a biblioteca em C quanto o algoritmo de compilação alcançaram o resultado esperado. Para biblioteca C (PETERSON, 2023) pode-se listar as seguintes estatísticas de projeto:

• Commits realizados: 42

• Arquivos: 21

• Linhas comentadas de código: 947

• Linhas de código escrito: 3368

• Período de trabalho: Maio 2022 - Junho 2023

Dada as especificações de projeto delimitadas, a checagem e testes deixam a biblioteca C com aspecto robusto e de pronto uso. A execução e temporização funcionam com base na biblioteca padrão C e a biblioteca pthread, e portanto irão trazer performance e generalidade para aplicações desktop e compatibilidade com sistemas embarcados, através da implementação genérica a base de threads.

Tendo opção de serialização da rede de petri em um formato de arquivo robusto, com versionamento e checagem de erro, capaz também de guardar estado, trás-se a possibilidade de um opção extra a compilação, onde dispositivos podem embarcar essas redes em formato de arquivo e carregá-las em memória no momento de execução.

A própria estrutura trás um aspecto trivial para utilização da mesma, seja na utilização das entradas e saídas de forma direta, no acesso ao estado interno, no aspecto assíncrono e também na parte da compilação, onde para maioria dos casos de utilização, dois laços de repetição, um para os lugares e outro para as transições, podem ser usados para varrer toda rede de petri, como por exemplo utilizado na compilação para lista de instrução.

Trabalhos futuros podem ser feitos sobre a parte assíncrona de temporização visto que temos o objectivo de levar essa biblioteca para sistemas embarcados. A compilação condicional de um sistema assíncrono baseado em timers de *hardware* traria toda precisão e acurácia que se espera do sistema embarcado, com a rede de petri nele embutida.

O algoritmo de compilação para rede de petri foi desenvolvido de maneira bastante simples, dado a estrutura da rede em mãos e o fato da lista de instrução já suportar alguns aspectos de arquitetura já de forma nativa, como eventos de borda e temporização, abrindo então portas para implementação de outras plataformas que também usam a lista de instrução, por vezes sendo extremamente similar a referência WEG TPW04 (WEG, 2015). Podendo ainda também possuir conversão da lista para outras linguagens como o próprio Ladder, aumentando o alcance dessa implementação além da própria lista de instrução.

Em sumo, os objetivos foram alcançados e espera-se que este trabalho sirva como semente para outras implementações e avanços, bem como sirva para difusão da utilização de redes de petri para modelagem de sistemas a eventos discretos, em especial a automação de sistemas industriais.

# REFERÊNCIAS

- DUFOURD, C.; FINKEL, A.; SCHNOEBELEN, P. Reset nets between decidability and undecidability. In: LARSEN, K. G.; SKYUM, S.; WINSKEL, G. (Ed.). *Automata, Languages and Programming*. Berlin, Heidelberg: Springer Berlin Heidelberg, 1998. p. 103–115. ISBN 978-3-540-68681-1.
- GHEZZI, C. et al. A unified high-level petri net formalism for time-critical systems. *IEEE Transactions on software engineering*, IEEE Computer Society, v. 17, n. 2, p. 160, 1991.
- GITHUB. Github. 2023. Disponível em: https://github.com. Acesso em: 17 jun. 2023.
- GNU. *libiberty CRC32*. 2012. Disponível em: https://github.com/gcc-mirror/gcc/blob/master/libiberty/crc32.c. Acesso em: 17 jun. 2023.
- GNU. GNU is not Unix. 2023. Disponível em: https://www.gnu.org. Acesso em: 17 jun. 2023.
- HILLSTON, J. Stochastic Petri Nets. 2009. Disponível em: https://www.inf.ed.ac.uk/teaching/courses/pm/handouts/stochasticpetrinets.pdf. Acesso em: 20 jun. 2023.
- IEEE. *POSIX*. 2023. Disponível em: http://get.posixcertified.ieee.org. Acesso em: 17 jun. 2023.
- International Electrotechnical Commission. *IEC 61131-3: Programmable Controllers Part 3: Programming Languages.* Geneva, Switzerland, 2013.
- ISO. High-level Petri Nets Concepts, Definitions and Graphical Notation. 2000. Disponível em: http://www.petrinets.info/docs/pnstd-4.7.1.pdf. Acesso em: 20 jun. 2023.
- IZHIKEVICH, E. M. *Petri Net.* 2011. Disponível em: http://www.scholarpedia.org/article/Petri\_net.
- JENSEN, K. Coloured Petri Nets: Basic Concepts, Analysis Methods and Practical Use. 2. ed. [S.l.]: Springer Berlin Heidelberg, 1997. ISBN 3-540-60943-1.
- KAID, H. et al. Applications of petri nets based models in manufacturing systems: A review. In: *Proceedings of the International Conference on Operations Excellence & Service Engineering, Orlando, FL.* [S.l.: s.n.], 2015. p. 516–28.
- LEVESON, N.; STOLZY, J. Safety analysis using petri nets. *IEEE Transactions on Software Engineering*, SE-13, n. 3, p. 386–397, 1987.
- LINUX FOUNDATION AND MICHAEL KERRISK. pthreads(7) Linux manual page. 2021. Disponível em: https://man7.org/linux/man-pages/man7/pthreads.7.html. Acesso em: 20 jun. 2023.
- MIT. MIT License. 2023. Disponível em: https://mit-license.org. Acesso em: 17 jun. 2023.
- MOREIRA, M. V.; BASILIO, J. C. Bridging the gap between design and implementation of discrete-event controllers. *IEEE Transactions on Automation Science and Engineering*, v. 11, n. 1, p. 48–65, 2014.

PETERSON, J. pnet - a petri net library for C/C++. 2023. Disponível em: https://github.com/Joao-Peterson/pnet. Acesso em: 17 jun. 2023.

PETRI, C. A. Kommunikation mit Automaten. Bonn, 1962.

SOUZA, A. L. de. Petrilab: Uma plataforma para simulação e geração de diagramas ladder de controladores a eventos discretos modelados por redes de petri. Tese (Doutorado) — Departamento de Engenharia Elétrica da Escola Politécnica, Universidade, 2015.

WEG. Série TPW-04 - Manual de Programação. 2015. Disponível em: https://static.weg.net/medias/downloadcenter/hd8/h4d/WEG-controlador-logico-programavel-tpw04-manual-de-programacao-10003853205-manual-portugues-br.pdf. Acesso em: 17 jun. 2023.

ZAITSEV, D. A. Toward the minimal universal petri net. *IEEE Transactions on Systems*, Man, and Cybernetics: Systems, v. 44, n. 1, p. 47–58, 2014.

ZHOU, M.; HRúZ, B. Modeling and Control of Discrete-event Dynamic Systems. 1. ed. [S.l.]: Springer London, 2007. ISBN 978-1-84628-872-2.