Síntese de bloco de IP de vídeo Vivado HLS: 12 etapas
Síntese de bloco de IP de vídeo Vivado HLS: 12 etapas
Anonim
Vivado HLS Video IP Block Synthesis
Vivado HLS Video IP Block Synthesis

Você sempre quis processar vídeo em tempo real sem adicionar muita latência ou em um sistema integrado? FPGAs (Field Programmable Gate Arrays) às vezes são usados para fazer isso; no entanto, escrever algoritmos de processamento de vídeo em linguagens de especificação de hardware como VHDL ou Verilog é, na melhor das hipóteses, frustrante. Digite Vivado HLS, a ferramenta Xilinx que permite programar em um ambiente C ++ e gerar código de linguagem de especificação de hardware a partir dele.

Softwares necessários:

  • Vivado HLS
  • Vivado
  • (Se você usar os registros AXI) Vivado SDK

(Opcional) Baixe os exemplos feitos pelo Xilinx aqui:

Exemplos de vídeo Xilinx HLS

Etapa 1: O que é Vivado HLS?

Vivado HLS é uma ferramenta usada para transformar o código c ++ em estruturas de hardware que podem ser implementadas em um FPGA. Inclui um IDE para fazer este desenvolvimento. Depois de concluir o desenvolvimento do código para HLS, você pode exportar o IP gerado em um formato para uso com Vivado.

Baixe os arquivos anexados e coloque-os perto de onde você criará seu projeto. (renomeie-os de volta para "top.cpp" e "top.h" se eles tiverem um nome aleatório)

Etapa 2: Biblioteca de vídeo HLS

HLS Video Library
HLS Video Library
HLS Video Library
HLS Video Library

A HLS Video Library tem documentação com designs de referência neste artigo: XAPP1167 Outro bom recurso é a página Wiki do Xilinx sobre o assunto.

Inicie o Vivado HLS.

Crie um novo projeto.

Pegue os arquivos que você baixou na etapa anterior e adicione-os como arquivos de origem. (Observação: os arquivos não são copiados para o projeto, mas permanecem onde estão)

Em seguida, use o botão Navegar para selecionar a função superior.

Na próxima página, selecione a peça Xilinx que você está usando.

Etapa 3: Sintetizando

Sintetizando
Sintetizando

Solução => Executar Síntese C => Solução Ativa

Depois de aproximadamente 227,218 segundos, isso deve ser feito. (Nota: o seu tempo de síntese real irá variar com base em muitos fatores)

Etapa 4: controle de versão e outras informações para exportação

Controle de versão e outras informações para exportação
Controle de versão e outras informações para exportação

Os números de versão interagem com o Vivado para permitir que você atualize o IP em um design. Se for uma alteração de versão secundária, isso pode ser feito no local enquanto as alterações de versão principais exigem que você adicione manualmente o novo bloco e remova o antigo. Se suas interfaces não foram alteradas e a atualização da versão é secundária, a atualização pode ser feito de forma totalmente automática, pressionando o botão de atualização de IP. Você pode executar "report_ip_status" no console tcl do Vivado para ver o status do seu IP.

Defina os números da versão e outras informações em Solução => Configurações da solução …

Como alternativa, essas configurações podem ser definidas durante a exportação.

Etapa 5: Exportando para uma biblioteca Vivado IP

Exportando para uma Biblioteca IP Vivado
Exportando para uma Biblioteca IP Vivado
Exportando para uma Biblioteca IP Vivado
Exportando para uma Biblioteca IP Vivado

Solução => Exportar RTL

Se você não configurou os detalhes da biblioteca de IP na etapa anterior, pode fazer isso agora.

Etapa 6: Análise de Síntese e Exportação

Análise de Síntese e Exportação
Análise de Síntese e Exportação
Análise de Síntese e Exportação
Análise de Síntese e Exportação
Análise de Síntese e Exportação
Análise de Síntese e Exportação

Nesta tela podemos ver as estatísticas sobre nosso módulo exportado, mostrando que ele cumpre nosso período de clock de 10ns (100MHz) e quanto de cada recurso ele usa.

Com uma combinação disso, nosso relatório de síntese e nossa análise de fluxo de dados, podemos ver que leva 317.338 ciclos de relógio * período de 10 ns * 14 estágios de pipeline = 0,04442732 segundos. O que significa que a latência total adicionada pelo nosso processamento de imagem é inferior a um vigésimo de segundo (quando cronometrado no alvo de 100 MHz).

Etapa 7: Adicionando a Biblioteca IP no Vivado

Adicionando a Biblioteca IP no Vivado
Adicionando a Biblioteca IP no Vivado
Adicionando a Biblioteca IP no Vivado
Adicionando a Biblioteca IP no Vivado
Adicionando a Biblioteca IP no Vivado
Adicionando a Biblioteca IP no Vivado
Adicionando a Biblioteca IP no Vivado
Adicionando a Biblioteca IP no Vivado

Para usar seu bloco de IP sintetizado, você precisará adicioná-lo ao Vivado.

No Vivado, adicione um repositório de IP ao seu projeto indo ao catálogo de IP e clique com o botão direito e selecionando "Adicionar Repositório …"

Navegue até o diretório do projeto Vivado HLS e selecione o diretório da solução.

Ele deve relatar o IP que encontrou.

Etapa 8: fazer um upgrade

Fazendo um upgrade
Fazendo um upgrade
Fazendo um upgrade
Fazendo um upgrade
Fazendo um upgrade
Fazendo um upgrade

Às vezes, você precisa fazer alterações em seu bloco HLS após incluí-lo em um design Vivado.

Para fazer isso, você pode fazer as alterações e ressintetizar e exportar o IP com um número de versão superior (consulte os detalhes na etapa anterior sobre alterações de número de versão principal / secundária).

Após alterar a exportação da nova versão, atualize seus repositórios IP no Vivado. Isso pode ser feito quando o Vivado percebe que o IP mudou no repositório, ou pode ser ativado manualmente. (Observe, se você atualizar seus repositórios de IP após o início, mas antes que a exportação seja concluída em HLS, o IP temporariamente não estará lá, espere que ele termine e atualize novamente.)

Neste ponto, uma janela deve aparecer com a informação de que um IP foi alterado no disco e dá a você a opção de atualizá-lo com um botão "Upgrade Selecionado". Se a mudança foi uma pequena mudança de versão e nenhuma das interfaces mudou, em seguida, pressionar esse botão substituirá automaticamente o IP antigo pelo novo, caso contrário, mais trabalho pode ser necessário.

Etapa 9: detalhes e informações adicionais

As etapas a seguir fornecem mais informações sobre como funciona a síntese HLS e o que você pode fazer com ela.

Para obter um exemplo de um projeto usando um bloco IP sintetizado por HLS, consulte este instrutível.

Etapa 10: saída e entrada

Saída e entrada
Saída e entrada
Saída e entrada
Saída e entrada

Saídas e entradas para o bloco IP final são determinadas a partir de uma análise que o sintetizador faz do fluxo de dados de entrada e saída da função superior.

Semelhante ao VHDL ou verilog, o HLS permite que você especifique detalhes sobre as conexões entre IP. Essas linhas são exemplos disso:

void image_filter (AXI_STREAM & video_in, AXI_STREAM & video_out, int & x, int & y) {

#pragma Porta do eixo HLS INTERFACE = video_in bundle = INPUT_STREAM #pragma Porta do eixo HLS INTERFACE = video_out bundle = OUTPUT_STREAM #pragma HLS INTERFACE porta s_axilite = x bundle = CONTROL_BUS offset = 0x14 # pragma HLS INTERFACE s_axilite porta 0xC

Você pode ver como as portas exibidas no bloco IP são influenciadas por essas diretivas.

Etapa 11: Interface de registro AXI

Interface de registro AXI
Interface de registro AXI

Para obter entrada / saída de / para seu bloco de IP para o PS, uma boa maneira de fazer isso é por meio de uma interface AXI.

Você pode especificar isso em seu código HLS, incluindo os deslocamentos a serem usados para acessar o valor posteriormente, como este:

void image_filter (AXI_STREAM & video_in, AXI_STREAM & video_out, int & x, int & y) {

#pragma HLS INTERFACE s_axilite porta = x pacote = CONTROL_BUS deslocamento = 0x14

#pragma HLS INTERFACE s_axilite port = y bundle = CONTROL_BUS offset = 0x1C #pragma HLS dataflow

x = 42;

y = 0xDEADBEEF; }

Uma vez conectado corretamente no Vivado, você pode acessar os valores usando este código no Vivado SDK:

#include "parameters.h"

#define xregoff 0x14 #define yregoff 0x1c x = Xil_In32 (XPAR_IMAGE_FILTER_0_S_AXI_CONTROL_BUS_BASEADDR + xregoff); y = Xil_In32 (XPAR_IMAGE_FILTER_0_S_AXI_CONTROL_BUS_BASEADDR + yregoff);

Isso fará com que você termine com 42 em xe 0xdeadbeef em y

Etapa 12: Dataflow Pragma

Dataflow Pragma
Dataflow Pragma
Dataflow Pragma
Dataflow Pragma
Dataflow Pragma
Dataflow Pragma

Dentro do #pragma DATAFLOW a forma como o código é implementado muda do C ++ normal. O código é canalizado para que todas as instruções sejam executadas em todos os momentos em diferentes partes dos dados (pense nisso como uma linha de montagem em uma fábrica, cada estação está trabalhando continuamente fazendo uma função e passando para a próxima estação)

na imagem você pode ver que cada uma das diretivas

Apesar de parecerem ser variáveis normais, os objetos img são, na verdade, implementados como pequenos buffers entre os comandos. Usar uma imagem como entrada para uma função a "consome" e a torna inutilizável. (Daí a necessidade de comandos duplicados)