1. 引言
在完成 Cordic 算法模块并获取正余弦数值后,即可进入图像旋转的核心计算阶段。图像旋转的本质是坐标变换,虽然资料繁多,但核心无非是两个公式的应用。以下记录具体的工程实现思路。

2. 图像旋转代码设计思路
2.1 旋转后的图像尺寸
图像旋转后,像素位置发生变化,总像素面积不变,但外接矩形尺寸会改变。为确保旋转后图像完整显示,需以原图对角线长度作为新图像的显示范围。
reg [12:0] row_size;
reg [12:0] col_size;
assign Pixel_X = row_size;
assign Pixel_Y = col_size;
wire [31:0] cosout_abs = (cosout[31]) ? -cosout : cosout;
wire [31:0] sinout_abs = (sinout[31]) ? -sinout : sinout;
always @(posedge clk_i, negedge rstn_i) begin
if (!rstn_i) begin
row_size <= 'd0;
col_size <= 'd0;
end else begin
// h --> row, w --> col
row_size <= (ROW*cosout_abs + COL*sinout_abs) >> 14;
col_size <= (COL*cosout_abs + ROW*sinout_abs) >> 14;
end
end
2.2 旋转后图像的有效位置
将旋转中心设定为屏幕中心。以常见的 480x272 LCD 为例,需根据时序参数计算数据请求信号 data_req。这里减去流水线级数是为了对齐信号。
// 时序参数定义
localparam H_SYNC = 11'd41, H_BACK = 11'd2, H_LEFT = 11'd0, H_VALID = 11'd480;
localparam H_RIGHT = 11'd0, H_FRONT = 11'd2, H_TOTAL = 11'd525;
localparam V_SYNC = 11'd10, V_BACK = 11'd2, V_TOP = 11'd0, V_VALID = 11'd272;
localparam V_BOTTOM = 11'd0, V_FRONT = 11'd2, V_TOTAL = 11'd286;
// data_req: 数据请求信号,以屏幕中点为基准
wire data_req = (((cnt_h >= (((H_VALID - Pixel_X)>>1) + H_SYNC + H_BACK - 'd5)) &&
(cnt_h < (((H_VALID - Pixel_X)>>1) + Pixel_X + H_SYNC + H_BACK - 'd5))) &&
((cnt_v >= ((V_VALID - Pixel_Y)>>1) + V_SYNC + V_BACK - 'd5) &&
(cnt_v < (((V_VALID - Pixel_Y)>>1) + Pixel_Y + V_SYNC + V_BACK - 'd5))));
2.3 第一级流水线
在图像有效信号期间进行行场计数,这是基础操作。
always @(posedge clk_i, negedge rstn_i) begin
if (!rstn_i) r_rotate_valid <= 1'b0;
else r_rotate_valid <= data_req;
end
always @(posedge clk_i, negedge rstn_i) begin
if (!rstn_i) r_rotate_end <= 'd0;
else if (r_rotate_valid && (vcnt == row_abs - 1) && (hcnt == col_abs - 2))
r_rotate_end <= 1'b1;
else r_rotate_end <= 'd0;
end
always @(posedge clk_i, negedge rstn_i) begin
if (!rstn_i) hcnt <= 'd0;
else if (r_rotate_valid && (r_rotate_end || (hcnt == col_abs - 1)))
hcnt <= 'd0;
else if (r_rotate_valid) hcnt <= hcnt + 'd1;
end
always @(posedge clk_i, negedge rstn_i) begin
if (!rstn_i) vcnt <= 'd0;
else if (r_rotate_valid && r_rotate_end) vcnt <= 'd0;
else if (r_rotate_valid && (hcnt == col_abs - 1)) vcnt <= vcnt + 'd1;
end



