首页 > 技术 > 内容

FPGA图像处理-Sobel边缘检测原理

时间:2026-01-31  作者:Diven  阅读:0

因为在做3*3卷积的时候,图像大小会变小,具体计算公式如下

其中O是输出特征图的大小,I是输入特征图的大小,P是Padding的大小,K是卷积核的大小,S是指Stride的大小,当K的值是3,P的值是1,S的值也是1,的时候O的值和I的值相等。

为了保持输出图像的大小在经过卷积后和输入的大小一样,我们需要进行Padding操作,在这里我采用了复制周围一圈的方式来完成。

采用python完成Sobel算法的参考模型

 

import cv2 as cvimport numpy as npimg = cv.imread(r"G:shiyanIDc.jpg")img_gray = cv.cvtColor(img, cv.COLOR_RGB2GRAY)h, w = img_gray.shapeimg_padding = np.zeros((h + 2, w + 2), np.uint8)img_padding[1:h + 1, 1:w + 1] = img_grayimg_padding[0:1, 1:w + 1] = img_gray[0:1, :]img_padding[h + 1:h + 2, 1:w + 1] = img_gray[h - 1:h, :]img_padding[:, 0:1] = img_padding[:, 1:2]img_padding[:, w + 1:w + 2] = img_padding[:, w:w + 1]th = 200sobel_rf = np.zeros((h, w), np.uint8)for i in range(1, h): for j in range(1, w): gx1 = img_padding[i - 1][j + 1] + 2 * img_padding[i][j + 1] + img_padding[i + 1][j + 1] gx2 = img_padding[i - 1][j - 1] + 2 * img_padding[i][j - 1] + img_padding[i + 1][j - 1] gy1 = img_padding[i - 1][j - 1] + 2 * img_padding[i - 1][j] + img_padding[i - 1][j + 1] gy2 = img_padding[i + 1][j - 1] + 2 * img_padding[i + 1][j] + img_padding[i + 1][j + 1] gx = abs(gx1 - gx2) gy = abs(gy1 - gy2) if gx + gy > th: sobel_rf[i - 1][j - 1] = 255 else: sobel_rf[i - 1][j - 1] = 0cv.imshow("sobel_rf", sobel_rf)cv.imshow("src", img_gray)cv.waitKey()cv.destroyAllWindows()

 

根据算法模型完成HDL:提供SpinalHDL源码

 

import spinal.core._import spinal.lib._class Sobel(th: Int, imageColNum: Int, imageRowNum: Int) extends Component { val io = new Bundle { val dataIn = slave(ImageStream(8, imageColNum, imageRowNum, 1)) val dataOut = master(ImageStream(8, imageColNum, imageRowNum, 1)) } noIoPrefix() val genMatrix = new GenMatrix(scala.math.pow(2, log2Up(imageColNum)).toInt, imageColNum, imageRowNum) genMatrix.io.dataIn <> io.dataIn val genMatrixOut = ImageStream(8, imageColNum, imageRowNum, 9) genMatrixOut := genMatrix.io.dataOut val GX1 = RegNext(genMatrixOut.data(0).asUInt +^ (genMatrixOut.data(1) ## B"1'b0").asUInt +^ genMatrixOut.data(2).asUInt) val GX2 = RegNext(genMatrixOut.data(6).asUInt +^ (genMatrixOut.data(7) ## B"1'b0").asUInt +^ genMatrixOut.data(8).asUInt) val GX = Reg(UInt(11 bits)) val GY1 = RegNext(genMatrixOut.data(6).asUInt +^ (genMatrixOut.data(3) ## B"1'b0").asUInt +^ genMatrixOut.data(0).asUInt) val GY2 = RegNext(genMatrixOut.data(8).asUInt +^ (genMatrixOut.data(5) ## B"1'b0").asUInt +^ genMatrixOut.data(2).asUInt) val GY = Reg(UInt(11 bits)) when(GX1 > GX2) { GX := GX1 - GX2 } otherwise { GX := GX2 - GX1 } when(GY1 > GY2) { GY := GY1 - GY2 } otherwise { GY := GY2 - GY1 } val G = RegNext(GX + GY) val sobelOut = Reg(Bits(8 bits)) when(G > th) { sobelOut := 255 } otherwise { sobelOut := 0 } io.dataOut.data(0) := sobelOut io.dataOut.row := Delay(genMatrixOut.row, 4) io.dataOut.col := Delay(genMatrixOut.col, 4) io.dataOut.c.hsync := Delay(genMatrixOut.c.hsync, 4,init = False) io.dataOut.c.vsync := Delay(genMatrixOut.c.vsync, 4,init = False) io.dataOut.c.de := Delay(genMatrixOut.c.de, 4,init = False)}object Sobel extends App { SpinalConfig().generateVerilog(new Sobel(200, 640, 480))}

 

仿真代码:

 

import spinal.lib._import spinal.core._import spinal.core.sim._import scala.collection.mutable.Queueimport java.io.FileOutputStreamimport scala.io.Sourceclass tbSobelC(th: Int) extends Sobel(th, 430, 430) { var src = Array[String]() var destDut = Array[String]() var destRef = Array[String]() // var srcLen = 0 var width = Array[Int]() var high = Array[Int]() val dutData = Queue[Int]() val refData = Queue[Int]() var frameLen = 0 def init(srcFile: Array[String], destDutFile: Array[String], destRefFile: Array[String], imgShape: Array[(Int, Int)]) = { clockDomain.forkStimulus(10) io.dataIn.data(0) #= 0 io.dataIn.row #= 0 io.dataIn.col #= 0 src = srcFile destDut = destDutFile destRef = destRefFile io.dataIn.c.de #= false io.dataIn.c.vsync #= false io.dataIn.c.hsync #= false frameLen = src.length width = imgShape.map(i => i._1) high = imgShape.map(i => i._2) clockDomain.waitSampling(10) } def FRAMe(src: String, width: Int, high: Int) = { val srcFile = Source.fROMFile(src) val srcData = srcFile.getLines() var colCnt = 0 var rowCnt = 0 io.dataIn.row #= width io.dataIn.col #= high io.dataIn.c.de #= false io.dataIn.c.vsync #= false io.dataIn.c.hsync #= false clockDomain.waitSampling(20) while (srcData.hasNext) { val data = srcData.next() io.dataIn.data(0) #= data.toInt io.dataIn.c.de #= true if (colCnt == 0 && rowCnt == 0) { io.dataIn.c.vsync #= true println("xx") } else { io.dataIn.c.vsync #= false } if (colCnt == width - 1 && rowCnt == high - 1) { clockDomain.waitSampling(1) io.dataIn.c.de #= false clockDomain.waitSampling(200) } if (colCnt == 0) { io.dataIn.c.hsync #= true } else { io.dataIn.c.hsync #= false } if (colCnt == width - 1 && rowCnt != high - 1) { clockDomain.waitSampling(1) io.dataIn.c.de #= false clockDomain.waitSampling(20) } if (colCnt == width - 1) { colCnt = 0 if (rowCnt == high - 1) { rowCnt = 0 } else { rowCnt = rowCnt + 1 } } else { colCnt = colCnt + 1 } clockDomain.waitSampling() } clockDomain.waitSampling(1000) srcFile.close() } def driver = { val dri = fork { for (i <- 0 until fRAMeLen) { println(s"FRAMe = ${i}") fRAMe(src(i), width(i), high(i)) } } } def dutOut = { val dutOutFile = new FileOutputStream(destDut(0)) val d = fork { while (true) { if (io.dataOut.c.de.toBoolean) { dutData.enqueue(io.dataOut.data(0).toInt) dutOutFile.write((io.dataOut.data(0).toInt.toString + "").getBytes()) } clockDomain.waitSampling() } } } def refFun = { val d = fork { while (true) { for (i <- 0 until frameLen) { val file = Source.fROMFile(destRef(i)) val srcData = file.getLines() while (srcData.hasNext) { clockDomain.waitSampling() val data = srcData.next().toInt refData.enqueue(data) } } } } } def scoreBoard = { val d = fork { var index = 0 while (true) { while (dutData.nonEmpty && refData.nonEmpty) { clockDomain.waitSampling() val dut = dutData.dequeue() val ref = refData.dequeue() // if(dut != ref){ // println(s"i:${index} dutData:${dut} refData:${ref}") // } index = index + 1 assert(scala.math.ABS(ref - dut) < 5, s"index:${index}, dutData:${dut} refData:${ref}") // if (scala.math.ABS(ref - dut) != 0) { // println(s"ref = ${ref} , dut = ${dut}") // } } clockDomain.waitSampling() } } } def waitSimDone = { val d = fork { var index = 0 while (index < width(0) * high(0)) {        clockDomain.waitSampling()        if (io.dataOut.c.de.toBoolean) {          index = index + 1        }      }      clockDomain.waitSampling(3000)      simSuccess()    }.join()  }}class tbSobel {  val testFile = Array("testGray.txt")  val dutFile = Array("testDut.txt")  val refFile = Array("testSobel.txt")  val imgShape = Array((430, 430))    val dut = SimConfig.withConfig(SpinalConfig(inlineRom = true)).withWave.compile(new tbSobelC(100))  dut.doSim { dut => dut.init(testFile, dutFile, refFile, imgShape) dut.driver dut.refFun dut.dutOut dut.scoreBoard dut.waitSimDone }}object tbSobel extends App { val tb = new tbSobel}

 

经过分析之后,该代码可以跑到238MHz,占用330LUT,312FF。


审核编辑:刘清

猜您喜欢


寻找可靠的贴片电阻供应商?东莞市众多贴片电阻生产厂家,为您提供高品质、高性能的电子元件。我们专注于贴片电阻的研发与生产,拥有先进的生产设备和严格的质量控制体系,...
2024-11-29 10:26:25
光敏电阻作为重要的光电传感元件,是不可少的配件。VITROHM作为知名的光敏电阻系列品牌,优异的性能和稳定的质量,深受广大电子工程师和制造商的青睐。本文将为您详...
2015-09-06 22:25:30
贴片电阻1210,指的是长宽分别为1.2mm和1.0mm的矩形表面贴装电阻,是电子电路中不可或缺的基础元件。它的小巧尺寸使其非常适合应用于空间有限的现代电子设备...
2025-04-14 15:03:50
焊条是焊接过程中不可少的材料,其种类繁多,主要可以根据不同的标准进行分类。按焊接工艺分类,焊条可分为手工电弧焊条、气体保护焊条和埋弧焊条等。手工电弧焊条适用于多...
2022-03-24 00:00:00
典型图像传感器的核心是CCD单元(charge-coupled device,电荷耦合器件)或标准CMOS单元(complementary meta-oxide...
2023-08-01 00:26:00
防静电玻璃盒是专门设计用于保护电子元件和敏感设备的容器,其参数主要包括以下几个方面:材料选择非常重要。防静电玻璃盒通常采用高品质的耐高温玻璃,具有良好的透明度和...
2008-08-14 00:00:00
现代电子设备中,封装技术的选择对性能和效率有着重要影响。SOP-8(SmallOutlinePackage)是常见的表面贴装封装类型,广泛应用于集成电路(IC)...
2025-02-21 12:40:06
夹是一个多义词,通常指的是夹持、固定或约束的动作或工具。在日常生活中,夹可以指用手或工具将物体夹紧的行为,比如夹子、夹板等。夹子的基本功能是通过施加压力,将物品...
2022-03-10 00:00:00
贴片电阻上的数字和字母可不是随便印的,代表着电阻值。理解这些标记,你就能快速识别电阻的大小。一般来说,贴片电阻会用三位数或四位数来表示阻值。三位数表示法: 前两...
2024-11-29 10:26:07
铁皮剪是专门用于剪切金属材料的工具,应用于建筑、制造和维修等领域。刀片采用高强度钢材制成,具有良好的耐磨性和锋利度,能够轻松应对各种厚度的铁皮、铝板及其金属材料...
2013-06-20 00:00:00