-
Notifications
You must be signed in to change notification settings - Fork 12
/
Copy pathtv_controller.vhd
334 lines (290 loc) · 14.1 KB
/
tv_controller.vhd
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
-------------------------------------------------------------------------------
--
-- 15 kHz TV output for Apple //e
-- Szombathelyi György
--
-- based on vga_controller by
-- Stephen A. Edwards, [email protected]
--
--
-- FIXME: This is all wrong
--
-- The Apple ][ uses a 14.31818 MHz master clock. It outputs a new
-- horizontal line every 65 * 14 + 2 = 912 14M cycles. The extra two
-- are from the "extended cycle" used to keep the 3.579545 MHz
-- colorburst signal in sync. Of these, 40 * 14 = 560 are active video.
--
-- In graphics mode, the Apple effectively generates 140 four-bit pixels
-- output serially (i.e., with 3.579545 MHz pixel clock). In text mode,
-- it generates 280 one-bit pixels (i.e., with a 7.15909 MHz pixel clock).
--
-- We capture 140 four-bit nibbles for each line and interpret them in
-- one of the two modes. In graphics mode, each is displayed as a
-- single pixel of one of 16 colors. In text mode, each is displayed
-- as two black or white pixels.
--
-------------------------------------------------------------------------------
library ieee;
use ieee.std_logic_1164.all;
use ieee.numeric_std.all;
entity tv_controller is
port (
clk_14m : in std_logic; -- 14.31818 MHz master clock
VIDEO : in std_logic; -- from the Apple video generator
COLOR_LINE : in std_logic;
SCREEN_MODE: in std_logic_vector(1 downto 0); -- 00: Color, 01: B&W, 10: Green, 11: Amber
COLOR_PALETTE : in std_logic_vector(1 downto 0); -- 00: Original, 01: //gs, 02: //e, 03: //e alternative
HBL : in std_logic;
VBL : in std_logic;
VGA_CLK : out std_logic;
VGA_HS : out std_logic; -- Active low
VGA_VS : out std_logic; -- Active low
VGA_BLANK : out std_logic;
VGA_R : out unsigned(7 downto 0);
VGA_G : out unsigned(7 downto 0);
VGA_B : out unsigned(7 downto 0)
);
end tv_controller;
architecture rtl of tv_controller is
-- RGB values from Linards Ticmanis,
-- http://newsgroups.derkeiler.com/Archive/Comp/comp.sys.apple2/2005-09/msg00534.html
type basis_color is array(0 to 3) of unsigned(7 downto 0);
constant basis_r : basis_color := ( X"88", X"38", X"07", X"38" );
constant basis_g : basis_color := ( X"22", X"24", X"67", X"52" );
constant basis_b : basis_color := ( X"2C", X"A0", X"2C", X"07" );
signal shift_reg : unsigned(5 downto 0); -- Last six pixels
signal last_hbl : std_logic;
signal last_vbl : std_logic;
signal hcount : unsigned(10 downto 0);
signal vcount : unsigned(8 downto 0);
signal hactive : std_logic;
signal vactive : std_logic;
constant VGA_SCANLINE : integer := 456*2; -- Must be 456*2 (set by the Apple)
constant VGA_HSYNC : integer := 64;
constant VGA_BACK_PORCH : integer := 158;
constant VGA_ACTIVE : integer := 282 * 2;
constant VGA_FRONT_PORCH : integer := 130;
-- VGA_HSYNC + VGA_BACK_PORCH + VGA_ACTIVE + VGA_FRONT_PORCH = VGA_SCANLINE
constant VBL_TO_VSYNC : integer := 33;
constant VGA_VSYNC_LINES : integer := 4;
signal VGA_VS_I, VGA_HS_I : std_logic;
signal video_active : std_logic;
signal color_line_delayed : std_logic;
begin
hcount_vcount_control : process (clk_14m)
begin
if rising_edge(clk_14m) then
if HBL = '0' and last_hbl = '1' then -- Falling edge
last_vbl <= VBL;
color_line_delayed <= COLOR_LINE;
hcount <= (others => '0');
vcount <= vcount + 1;
if last_VBL = '0' and VBL = '1' then
vcount <= (others => '0');
end if;
else
hcount <= hcount + 1;
end if;
last_hbl <= HBL;
end if;
end process hcount_vcount_control;
hsync_gen : process (clk_14m)
begin
if rising_edge(clk_14m) then
if hcount = VGA_ACTIVE + VGA_FRONT_PORCH then
VGA_HS_I <= '0';
elsif hcount = VGA_ACTIVE + VGA_FRONT_PORCH + VGA_HSYNC then
VGA_HS_I <= '1';
end if;
if hcount = 8 then
hactive <= '1';
vactive <= not VBL;
elsif hcount = VGA_ACTIVE + 10 then
hactive <= '0';
vactive <= not VBL;
end if;
end if;
end process hsync_gen;
VGA_HS <= VGA_HS_I;
vsync_gen : process (clk_14m)
begin
if rising_edge(clk_14m) then
if vcount = VBL_TO_VSYNC and hcount = VGA_ACTIVE + VGA_FRONT_PORCH + VGA_HSYNC then
VGA_VS_I <= '0';
elsif vcount = VBL_TO_VSYNC + VGA_VSYNC_LINES and hcount = VGA_ACTIVE + VGA_FRONT_PORCH then
VGA_VS_I <= '1';
end if;
end if;
end process vsync_gen;
VGA_VS <= VGA_VS_I;
shifter: process (clk_14m)
begin
if rising_edge(clk_14m) then
shift_reg <= VIDEO & shift_reg(5 downto 1);
end if;
end process;
video_active <= hactive and vactive;
pixel_generator: process (clk_14m)
variable r, g, b : unsigned(7 downto 0);
variable shift_color : unsigned(3 downto 0); -- subset of shift register to determine color in LUT
begin
if rising_edge(clk_14m) then
r := X"00";
g := X"00";
b := X"00";
-- alternate background for monochrome modes
case SCREEN_MODE is
when "00" =>
r := X"00"; g := X"00"; b := X"00"; -- color mode background
when "01" =>
r := X"00"; g := X"00"; b := X"00"; -- B&W mode background
when "10" =>
r := X"00"; g := X"0F"; b := X"01"; -- green mode background color
when "11" =>
r := X"20"; g := X"08"; b := X"01"; -- amber mode background color
end case;
if video_active = '1' then
if color_line_delayed = '0' then -- Monochrome mode
if shift_reg(2) = '1' then
-- handle green/amber color modes
case SCREEN_MODE is
when "00" =>
r := X"FF"; g := X"FF"; b := X"FF"; -- white (color mode)
when "01" =>
r := X"FF"; g := X"FF"; b := X"FF"; -- white (B&W mode)
when "10" =>
r := X"00"; g := X"C0"; b := X"01"; -- green
when "11" =>
r := X"FF"; g := X"80"; b := X"01"; -- amber
end case;
end if;
elsif shift_reg(0) = shift_reg(4) and shift_reg(5) = shift_reg(1) then
-- rotate the 4-bit value based on contents of shift register
-- then apply the color mapping
shift_color := shift_reg(4 downto 1) rol to_integer(hcount);
if COLOR_PALETTE = "00" then
-- lticmanis 2024 (default)
-- 'correct' palette derived by mathematical NTSC formulas,
-- calibrated to approximate GS colors as much NTSC can
case shift_color is
when "0000" => r := X"00"; g := X"00"; b := X"00"; -- black
when "0010" => r := X"9F"; g := X"1B"; b := X"48"; -- magenta
when "0100" => r := X"48"; g := X"32"; b := X"EB"; -- dark blue
when "0110" => r := X"D6"; g := X"43"; b := X"EF"; -- purple
when "1000" => r := X"19"; g := X"75"; b := X"44"; -- dark green
when "0101" => r := X"81"; g := X"81"; b := X"81"; -- gray 1
when "1100" => r := X"36"; g := X"92"; b := X"FF"; -- med blue
when "1110" => r := X"D8"; g := X"9E"; b := X"FF"; -- light blue
when "0001" => r := X"49"; g := X"65"; b := X"00"; -- brown
when "0011" => r := X"D8"; g := X"73"; b := X"00"; -- orange
when "1010" => r := X"81"; g := X"81"; b := X"81"; -- gray 2
when "0111" => r := X"FB"; g := X"8F"; b := X"BC"; -- pink
when "1001" => r := X"3C"; g := X"CC"; b := X"00"; -- green
when "1011" => r := X"BC"; g := X"D6"; b := X"00"; -- yellow
when "1101" => r := X"6C"; g := X"E6"; b := X"B8"; -- aquamarine
when "1111" => r := X"FF"; g := X"FF"; b := X"FF"; -- white
end case;
elsif COLOR_PALETTE = "01" then
-- Apple IIgs & LC (//e mode) palette
-- see "IIGS Technical Note #63" from Apple ,
-- and the 'clut' resource "Apple IIe Colors" in the IIe card's "IIe Startup" 68K Mac executable version 2.2.1d.
case shift_color is
when "0000" => r := X"00"; g := X"00"; b := X"00"; -- 0 - black
when "0010" => r := X"DB"; g := X"1F"; b := X"42"; -- 1 - magenta
when "0100" => r := X"0C"; g := X"11"; b := X"A4"; -- 2 - dark blue
when "0110" => r := X"DC"; g := X"43"; b := X"E1"; -- 3 - purple
when "1000" => r := X"1C"; g := X"82"; b := X"31"; -- 4 - dark green
when "0101" => r := X"B3"; g := X"B3"; b := X"B3"; -- 5 - gray 1
when "1100" => r := X"39"; g := X"3D"; b := X"FF"; -- 6 - med blue
when "1110" => r := X"7A"; g := X"B3"; b := X"FF"; -- 7 - light blue
when "0001" => r := X"91"; g := X"64"; b := X"00"; -- 8 - brown
when "0011" => r := X"FA"; g := X"77"; b := X"00"; -- 9 - orange
when "1010" => r := X"63"; g := X"63"; b := X"63"; -- 10 - gray 2
when "0111" => r := X"FB"; g := X"A5"; b := X"93"; -- 11 - pink
when "1001" => r := X"40"; g := X"DE"; b := X"00"; -- 12 - green
when "1011" => r := X"FE"; g := X"FE"; b := X"00"; -- 13 - yellow
when "1101" => r := X"67"; g := X"FC"; b := X"A4"; -- 14 - aquamarine
when "1111" => r := X"FF"; g := X"FF"; b := X"FF"; -- 15 - white
end case;
--elsif COLOR_PALETTE = "10" then
-- older 2005 palette from Linards Ticmanis posted on comp.sys.apple2 in 29-Sep-2005
-- (kept for historical comparison)
--case shift_color is
--when "0000" => r := X"00"; g := X"00"; b := X"00"; -- black
--when "0010" => r := X"8A"; g := X"21"; b := X"40"; -- magenta
--when "0100" => r := X"2C"; g := X"22"; b := X"a5"; -- dark blue
--when "0110" => r := X"d0"; g := X"43"; b := X"E3"; -- purple
--when "1000" => r := X"07"; g := X"70"; b := X"28"; -- dark green
--when "0101" => r := X"7b"; g := X"7e"; b := X"80"; -- gray 1
--when "1100" => r := X"30"; g := X"8F"; b := X"e3"; -- med blue
--when "1110" => r := X"B9"; g := X"A9"; b := X"FD"; -- light blue
--when "0001" => r := X"3b"; g := X"51"; b := X"07"; -- brown
--when "0011" => r := X"B6"; g := X"8F"; b := X"00"; -- orange
--when "1010" => r := X"7b"; g := X"7E"; b := X"80"; -- gray 2
--when "0111" => r := X"F3"; g := X"9A"; b := X"C2"; -- pink
--when "1001" => r := X"2f"; g := X"B8"; b := X"1F"; -- green
--when "1011" => r := X"BF"; g := X"D3"; b := X"5A"; -- yellow
--when "1101" => r := X"6E"; g := X"E1"; b := X"C0"; -- aquamarine
--when "1111" => r := X"FF"; g := X"FF"; b := X"FF"; -- white
--end case;
elsif COLOR_PALETTE = "10" then
-- AppleWin palette as of 1.13.18.0
case shift_color is
when "0000" => r := X"00"; g := X"00"; b := X"00"; -- black
when "0010" => r := X"93"; g := X"0B"; b := X"7C"; -- magenta
when "0100" => r := X"1F"; g := X"35"; b := X"D3"; -- dark blue
when "0110" => r := X"BB"; g := X"36"; b := X"FF"; -- purple
when "1000" => r := X"00"; g := X"76"; b := X"0C"; -- dark green
when "0101" => r := X"7E"; g := X"7E"; b := X"7E"; -- gray 1
when "1100" => r := X"07"; g := X"A8"; b := X"E0"; -- med blue
when "1110" => r := X"9D"; g := X"AC"; b := X"FF"; -- light blue
when "0001" => r := X"62"; g := X"4C"; b := X"00"; -- brown
when "0011" => r := X"F9"; g := X"56"; b := X"1D"; -- orange
when "1010" => r := X"7E"; g := X"7E"; b := X"7E"; -- gray 2
when "0111" => r := X"FF"; g := X"81"; b := X"EC"; -- pink
when "1001" => r := X"43"; g := X"C8"; b := X"00"; -- green
when "1011" => r := X"DC"; g := X"CD"; b := X"16"; -- yellow
when "1101" => r := X"5D"; g := X"F7"; b := X"84"; -- aquamarine
when "1111" => r := X"FF"; g := X"FF"; b := X"FF"; -- white
end case;
else
-- original implementaiton of Apple II FPGA core (sedwards, 2009)
-- Tint of adjacent pixels is consistent : display the color
if shift_reg(3) = '1' then
r := r + basis_r(to_integer(hcount + 1));
g := g + basis_g(to_integer(hcount + 1));
b := b + basis_b(to_integer(hcount + 1));
end if;
if shift_reg(4) = '1' then
r := r + basis_r(to_integer(hcount + 2));
g := g + basis_g(to_integer(hcount + 2));
b := b + basis_b(to_integer(hcount + 2));
end if;
if shift_reg(1) = '1' then
r := r + basis_r(to_integer(hcount + 3));
g := g + basis_g(to_integer(hcount + 3));
b := b + basis_b(to_integer(hcount + 3));
end if;
if shift_reg(2) = '1' then
r := r + basis_r(to_integer(hcount));
g := g + basis_g(to_integer(hcount));
b := b + basis_b(to_integer(hcount));
end if;
end if;
else
-- Tint is changing: display only black, gray, or white
case shift_reg(3 downto 2) is
when "11" => r := X"FF"; g := X"FF"; b := X"FF";
when "01" | "10" => r := X"80"; g := X"80"; b := X"80";
when others => r := X"00"; g := X"00"; b := X"00";
end case;
end if;
end if;
VGA_R <= r;
VGA_G <= g;
VGA_B <= b;
end if;
end process pixel_generator;
VGA_CLK <= clk_14m;
VGA_BLANK <= not video_active;
end rtl;