From 10b0939930bc7261cc527903dbac92d0e2ebe0e3 Mon Sep 17 00:00:00 2001 From: Mark Date: Sat, 1 Mar 2025 09:31:48 -0800 Subject: [PATCH] Random generation --- README.md | 2 - tetros/Cargo.lock | 72 +++++++++++++++++++ tetros/Cargo.toml | 1 + tetros/src/drivers/vga.rs | 49 +++++++++++++ tetros/src/lib.rs | 16 ++--- tetros/src/tetrisboard/cell.rs | 31 -------- tetros/src/tetrisboard/draw.rs | 21 +++--- .../{fall/mod.rs => tetrisboard/falling.rs} | 47 ++++++++++-- tetros/src/tetrisboard/mod.rs | 18 +++-- 9 files changed, 196 insertions(+), 61 deletions(-) delete mode 100644 tetros/src/tetrisboard/cell.rs rename tetros/src/{fall/mod.rs => tetrisboard/falling.rs} (87%) diff --git a/README.md b/README.md index 4d9ee92..b81c3df 100644 --- a/README.md +++ b/README.md @@ -1,5 +1,3 @@ -- better colors -- random spawning - improve collisions - controls (input) - clear lines diff --git a/tetros/Cargo.lock b/tetros/Cargo.lock index 46cf528..6780134 100644 --- a/tetros/Cargo.lock +++ b/tetros/Cargo.lock @@ -45,6 +45,40 @@ dependencies = [ "scopeguard", ] +[[package]] +name = "proc-macro2" +version = "1.0.93" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "60946a68e5f9d28b0dc1c21bb8a97ee7d018a8b322fa57838ba31cc878e22d99" +dependencies = [ + "unicode-ident", +] + +[[package]] +name = "quote" +version = "1.0.38" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "0e4dccaaaf89514f546c693ddc140f729f958c247918a13380cccc6078391acc" +dependencies = [ + "proc-macro2", +] + +[[package]] +name = "rand" +version = "0.9.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "3779b94aeb87e8bd4e834cee3650289ee9e0d5677f976ecdb6d219e5f4f6cd94" +dependencies = [ + "rand_core", + "zerocopy", +] + +[[package]] +name = "rand_core" +version = "0.9.3" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "99d9a13982dcf210057a8a78572b2217b667c3beacbf3a0d8b454f6f82837d38" + [[package]] name = "raw-cpuid" version = "10.7.0" @@ -75,12 +109,24 @@ dependencies = [ "lock_api", ] +[[package]] +name = "syn" +version = "2.0.98" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "36147f1a48ae0ec2b5b3bc5b537d267457555a10dc06f3dbc8cb11ba3006d3b1" +dependencies = [ + "proc-macro2", + "quote", + "unicode-ident", +] + [[package]] name = "tetros" version = "1.0.0" dependencies = [ "bitflags 2.8.0", "lazy_static", + "rand", "spin", "uart_16550", ] @@ -96,6 +142,12 @@ dependencies = [ "x86", ] +[[package]] +name = "unicode-ident" +version = "1.0.17" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "00e2473a93778eb0bad35909dff6a10d28e63f792f16ed15e404fca9d5eeedbe" + [[package]] name = "x86" version = "0.52.0" @@ -106,3 +158,23 @@ dependencies = [ "bitflags 1.3.2", "raw-cpuid", ] + +[[package]] +name = "zerocopy" +version = "0.8.21" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "dcf01143b2dd5d134f11f545cf9f1431b13b749695cb33bcce051e7568f99478" +dependencies = [ + "zerocopy-derive", +] + +[[package]] +name = "zerocopy-derive" +version = "0.8.21" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "712c8386f4f4299382c9abee219bee7084f78fb939d88b6840fcc1320d5f6da2" +dependencies = [ + "proc-macro2", + "quote", + "syn", +] diff --git a/tetros/Cargo.toml b/tetros/Cargo.toml index 7741742..6d22d53 100644 --- a/tetros/Cargo.toml +++ b/tetros/Cargo.toml @@ -65,6 +65,7 @@ identity_op = "allow" [dependencies] bitflags = "2.8.0" +rand = { version = "0.9.0", features = ["small_rng"], default-features = false } spin = "0.9.8" uart_16550 = "0.3.2" diff --git a/tetros/src/drivers/vga.rs b/tetros/src/drivers/vga.rs index 0fa2f24..194882e 100644 --- a/tetros/src/drivers/vga.rs +++ b/tetros/src/drivers/vga.rs @@ -1,4 +1,53 @@ use core::slice; +use rand::seq::IndexedRandom; + +use crate::RNG; + +#[repr(u8)] +#[derive(Debug, Clone, Copy, PartialEq, Eq)] +pub enum VgaColor { + Black, + Gray, + Blue, + Cyan, + Orange, + Red, + Green, + Purple, + Yellow, +} + +impl VgaColor { + pub fn as_u8(self) -> u8 { + match self { + Self::Black => 0b0000_0000, + Self::Gray => 0b1110_0000, + Self::Blue => 0b0000_0001, + Self::Cyan => 0b0000_0011, + Self::Orange => 0b0000_0110, + Self::Red => 0b0000_0100, + Self::Green => 0b0000_0010, + Self::Purple => 0b0000_0101, + Self::Yellow => 0b1100_0000, + } + } + + /// Pick a random non-utility color + pub fn choose_rand() -> Self { + let colors = [ + VgaColor::Blue, + VgaColor::Cyan, + VgaColor::Orange, + VgaColor::Red, + VgaColor::Green, + VgaColor::Purple, + VgaColor::Yellow, + ]; + + let mut rng = RNG.lock(); + *(colors.choose(&mut rng).unwrap()) + } +} /// VGA driver for mode 0x13: /// diff --git a/tetros/src/lib.rs b/tetros/src/lib.rs index ef84702..8431b8f 100644 --- a/tetros/src/lib.rs +++ b/tetros/src/lib.rs @@ -4,18 +4,15 @@ #![feature(abi_x86_interrupt)] #![allow(internal_features)] -use core::num::NonZeroUsize; - -use fall::{FallingTetromino, Tetromino}; use lazy_static::lazy_static; +use rand::{rngs::SmallRng, SeedableRng}; use spin::Mutex; use drivers::{pic::PICDriver, vga::Vga13h}; use idt::{InterruptDescriptorTable, InterruptStackFrame}; use os::thunk::ThunkData; -use tetrisboard::TetrisBoard; +use tetrisboard::{FallingTetromino, TetrisBoard}; -mod fall; mod idt; mod os; mod tetrisboard; @@ -30,6 +27,7 @@ pub(crate) static BOARD: Mutex = Mutex::new(TetrisBoard::new()); pub(crate) static FALLING: Mutex> = Mutex::new(None); lazy_static! { + static ref RNG: Mutex = Mutex::new(SmallRng::seed_from_u64(1337)); static ref IDT: InterruptDescriptorTable = { let mut idt = InterruptDescriptorTable::new(); @@ -84,7 +82,7 @@ fn tick() { } } } else { - *fall = Some(FallingTetromino::new(Tetromino::AngleLeft, 5, 1)) + *fall = Some(FallingTetromino::random(5, 1)) } } @@ -124,15 +122,11 @@ pub unsafe extern "C" fn start( let mut v = VGA.lock(); let board = BOARD.lock(); - let mut fall = FALLING.lock(); - *fall = Some(FallingTetromino::new(Tetromino::SkewRight, 5, 1)); - - board.draw(&mut v, fall.as_ref()); + board.draw(&mut v, None); v.swap(); // Important to drop these, or tick() will deadlock drop(v); - drop(fall); drop(board); loop {} diff --git a/tetros/src/tetrisboard/cell.rs b/tetros/src/tetrisboard/cell.rs deleted file mode 100644 index 42a8dbe..0000000 --- a/tetros/src/tetrisboard/cell.rs +++ /dev/null @@ -1,31 +0,0 @@ -#[repr(u8)] -#[derive(Debug, Clone, Copy, PartialEq, Eq)] -pub enum TetrisCell { - Empty, - Blue, - Cyan, - Orange, - Red, - Green, - Purple, - Yellow, -} - -impl TetrisCell { - pub fn vga_color(&self) -> u8 { - match self { - Self::Empty => 0b0000_0000, - Self::Blue => 0b0000_0001, - Self::Cyan => 0b0000_0011, - Self::Orange => 0b0000_0110, - Self::Red => 0b0000_0100, - Self::Green => 0b0000_0010, - Self::Purple => 0b0000_0101, - Self::Yellow => 0b1100_0000, - } - } - - pub fn is_empty(&self) -> bool { - matches!(self, Self::Empty) - } -} diff --git a/tetros/src/tetrisboard/draw.rs b/tetros/src/tetrisboard/draw.rs index 3d3c553..8de472d 100644 --- a/tetros/src/tetrisboard/draw.rs +++ b/tetros/src/tetrisboard/draw.rs @@ -1,10 +1,10 @@ -use crate::{drivers::vga::Vga13h, fall::FallingTetromino}; +use crate::drivers::vga::{Vga13h, VgaColor}; -use super::{TetrisBoard, TetrisCell}; +use super::{falling::FallingTetromino, TetrisBoard, TetrisCell}; impl TetrisBoard { - fn draw_cell(&self, fb: &mut [u8], state: TetrisCell, x: usize, y: usize) { - let color = state.vga_color(); + fn draw_cell(&self, fb: &mut [u8], color: VgaColor, x: usize, y: usize) { + let color = color.as_u8(); for yo in 0..TetrisBoard::CELL_SIZE { let left = Vga13h::pix_idx(x, y + yo); let right = Vga13h::pix_idx(x + TetrisBoard::CELL_SIZE, y + yo); @@ -13,7 +13,7 @@ impl TetrisBoard { } fn draw_frame(&self, fb: &mut [u8], x: usize, y: usize) { - let color = 0b1110_0000; + let color = VgaColor::Gray.as_u8(); for yo in 0..TetrisBoard::CELL_SIZE { let left = Vga13h::pix_idx(x, y + yo); let right = Vga13h::pix_idx(x + TetrisBoard::CELL_SIZE, y + yo); @@ -23,6 +23,7 @@ impl TetrisBoard { pub fn draw(&self, vga: &mut Vga13h, falling: Option<&FallingTetromino>) { let fb = vga.get_fb(); + // Draw cells for bx in 0..TetrisBoard::BOARD_WIDTH { for by in 0..TetrisBoard::BOARD_HEIGHT { @@ -30,17 +31,21 @@ impl TetrisBoard { let dx = (bx + 1) * TetrisBoard::CELL_SIZE; let dy = (by + 1) * TetrisBoard::CELL_SIZE; - self.draw_cell(fb, cell, dx, dy); + if let TetrisCell::Filled { color } = cell { + self.draw_cell(fb, color, dx, dy); + } else { + self.draw_cell(fb, VgaColor::Black, dx, dy); + } } } + // Draw falling tetromino if let Some(falling) = falling { for (x, y) in falling.tiles() { - let cell = TetrisCell::Blue; let dx = (x + 1) * TetrisBoard::CELL_SIZE; let dy = (y + 1) * TetrisBoard::CELL_SIZE; - self.draw_cell(fb, cell, dx, dy); + self.draw_cell(fb, falling.color, dx, dy); } } diff --git a/tetros/src/fall/mod.rs b/tetros/src/tetrisboard/falling.rs similarity index 87% rename from tetros/src/fall/mod.rs rename to tetros/src/tetrisboard/falling.rs index f32b19f..eafb193 100644 --- a/tetros/src/fall/mod.rs +++ b/tetros/src/tetrisboard/falling.rs @@ -1,3 +1,8 @@ +use rand::seq::IndexedRandom; + +use crate::{drivers::vga::VgaColor, RNG}; + +#[derive(Debug, PartialEq, Eq, Clone, Copy)] pub enum Tetromino { /// The 1x4 line Straight, @@ -21,6 +26,24 @@ pub enum Tetromino { Tee, } +impl Tetromino { + /// Pick a random tetromino + pub fn choose_rand() -> Self { + let ominos = [ + Tetromino::Straight, + Tetromino::Square, + Tetromino::AngleRight, + Tetromino::AngleLeft, + Tetromino::SkewRight, + Tetromino::SkewLeft, + Tetromino::Tee, + ]; + + let mut rng = RNG.lock(); + *(ominos.choose(&mut rng).unwrap()) + } +} + #[derive(Debug, Clone, Copy)] pub enum Direction { North, @@ -54,13 +77,26 @@ pub struct FallingTetromino { direction: Direction, center_x: usize, center_y: usize, + + pub color: VgaColor, } impl FallingTetromino { - pub fn new(tetromino: Tetromino, center_x: usize, center_y: usize) -> Self { + pub fn new(tetromino: Tetromino, color: VgaColor, center_x: usize, center_y: usize) -> Self { Self { tetromino, direction: Direction::North, + color, + center_x, + center_y, + } + } + + pub fn random(center_x: usize, center_y: usize) -> Self { + Self { + tetromino: Tetromino::choose_rand(), + direction: Direction::North, + color: VgaColor::choose_rand(), center_x, center_y, } @@ -119,6 +155,7 @@ impl FallingTetromino { (self.center_x - 1, self.center_y), (self.center_x - 1, self.center_y + 1), ], + // // Right Angle // @@ -195,7 +232,7 @@ impl FallingTetromino { (self.center_x, self.center_y - 1), (self.center_x, self.center_y), (self.center_x - 1, self.center_y), - (self.center_x - 1, self.center_y - 1), + (self.center_x - 1, self.center_y + 1), ], (Tetromino::SkewLeft, Direction::South) => [ @@ -209,7 +246,7 @@ impl FallingTetromino { (self.center_x + 1, self.center_y - 1), (self.center_x + 1, self.center_y), (self.center_x, self.center_y), - (self.center_x, self.center_y - 1), + (self.center_x, self.center_y + 1), ], // @@ -223,7 +260,7 @@ impl FallingTetromino { ], (Tetromino::SkewRight, Direction::East) => [ - (self.center_x, self.center_y - 1), + (self.center_x, self.center_y + 1), (self.center_x, self.center_y), (self.center_x + 1, self.center_y), (self.center_x + 1, self.center_y - 1), @@ -240,7 +277,7 @@ impl FallingTetromino { (self.center_x - 1, self.center_y - 1), (self.center_x - 1, self.center_y), (self.center_x, self.center_y), - (self.center_x, self.center_y - 1), + (self.center_x, self.center_y + 1), ], // diff --git a/tetros/src/tetrisboard/mod.rs b/tetros/src/tetrisboard/mod.rs index 41d2f7e..fec2b23 100644 --- a/tetros/src/tetrisboard/mod.rs +++ b/tetros/src/tetrisboard/mod.rs @@ -1,8 +1,16 @@ -mod cell; mod draw; -pub use cell::*; +mod falling; -use crate::fall::FallingTetromino; +pub use falling::FallingTetromino; + +use crate::drivers::vga::VgaColor; + +#[repr(u8)] +#[derive(Debug, Clone, Copy, PartialEq, Eq)] +pub enum TetrisCell { + Empty, + Filled { color: VgaColor }, +} pub struct TetrisBoard { board: [TetrisCell; TetrisBoard::BOARD_WIDTH * TetrisBoard::BOARD_HEIGHT], @@ -24,7 +32,9 @@ impl TetrisBoard { for (x, y) in tetromino.tiles() { let cell = self.get_cell_mut(x, y); if let Some(cell) = cell { - *cell = TetrisCell::Blue; + *cell = TetrisCell::Filled { + color: tetromino.color, + }; } } }