diff --git a/core/load_store_unit.sv b/core/load_store_unit.sv index 95e622f757..14f523c0a3 100644 --- a/core/load_store_unit.sv +++ b/core/load_store_unit.sv @@ -171,26 +171,15 @@ module load_store_unit // ); if (MMU_PRESENT) begin : gen_mmu - localparam HYP_EXT = 0; //CVA6Cfg.CVA6ConfigHExtEn localparam ASID_LEN = (riscv::XLEN == 64) ? 16 : 9; - localparam VPN_LEN = (riscv::XLEN == 64) ? (HYP_EXT ? 29 : 27) : 20; + localparam VPN_LEN = (riscv::XLEN == 64) ? 27 : 20; localparam PT_LEVELS = (riscv::XLEN == 64) ? 3 : 2; - localparam int unsigned mmu_ASID_WIDTH [HYP_EXT:0] = {ASID_WIDTH}; - logic [mmu_ASID_WIDTH[0]-1:0] mmu_asid_i [HYP_EXT:0]; - logic [mmu_ASID_WIDTH[0]-1:0] mmu_asid_to_be_flushed_i [HYP_EXT:0]; - logic [riscv::VLEN-1:0] mmu_vaddr_to_be_flushed_i [HYP_EXT:0]; - logic [riscv::VLEN-1:0] mmu_lsu_vaddr_i[HYP_EXT:0]; - - assign mmu_asid_i[0] = asid_i; - assign mmu_asid_to_be_flushed_i[0] =asid_to_be_flushed_i; - assign mmu_vaddr_to_be_flushed_i[0] =vaddr_to_be_flushed_i; - assign mmu_lsu_vaddr_i[0]= mmu_vaddr; cva6_mmu #( .CVA6Cfg (CVA6Cfg), .INSTR_TLB_ENTRIES(ariane_pkg::INSTR_TLB_ENTRIES), .DATA_TLB_ENTRIES (ariane_pkg::DATA_TLB_ENTRIES), - .ASID_WIDTH (mmu_ASID_WIDTH), + .ASID_WIDTH (ASID_WIDTH), .ASID_LEN (ASID_LEN), .VPN_LEN (VPN_LEN), .PT_LEVELS (PT_LEVELS) @@ -199,7 +188,7 @@ module load_store_unit .misaligned_ex_i(misaligned_exception), .lsu_is_store_i (st_translation_req), .lsu_req_i (translation_req), - .lsu_vaddr_i (mmu_lsu_vaddr_i), + .lsu_vaddr_i (mmu_vaddr), .lsu_valid_o (translation_valid), .lsu_paddr_o (mmu_paddr), .lsu_exception_o(mmu_exception), @@ -210,14 +199,11 @@ module load_store_unit .req_port_o (dcache_req_ports_o[0]), // icache address translation requests .icache_areq_i (icache_areq_i), - // .asid_to_be_flushed_i, - // .vaddr_to_be_flushed_i, + .asid_to_be_flushed_i, + .vaddr_to_be_flushed_i, .icache_areq_o (icache_areq_o), .pmpcfg_i, .pmpaddr_i, - .asid_i(mmu_asid_i), - .asid_to_be_flushed_i(mmu_asid_to_be_flushed_i), - .vaddr_to_be_flushed_i(mmu_vaddr_to_be_flushed_i), .* ); end else begin : gen_no_mmu diff --git a/core/mmu_unify/cva6_mmu.sv b/core/mmu_unify/cva6_mmu.sv index bc3df48eb4..cf65db5fc3 100644 --- a/core/mmu_unify/cva6_mmu.sv +++ b/core/mmu_unify/cva6_mmu.sv @@ -23,11 +23,10 @@ import ariane_pkg::*; parameter config_pkg::cva6_cfg_t CVA6Cfg = config_pkg::cva6_cfg_empty, parameter int unsigned INSTR_TLB_ENTRIES = 4, parameter int unsigned DATA_TLB_ENTRIES = 4, - parameter int unsigned ASID_WIDTH [HYP_EXT:0]= {1}, + parameter int unsigned ASID_WIDTH = 1, parameter int unsigned ASID_LEN = 1, parameter int unsigned VPN_LEN = 1, - parameter int unsigned PT_LEVELS = 1, - parameter logic HYP_EXT = 0 + parameter int unsigned PT_LEVELS = 1 ) ( input logic clk_i, input logic rst_ni, @@ -42,7 +41,7 @@ import ariane_pkg::*; // in the LSU as we distinguish load and stores, what we do here is simple address translation input exception_t misaligned_ex_i, input logic lsu_req_i, // request address translation - input logic [riscv::VLEN-1:0] lsu_vaddr_i[HYP_EXT:0], // virtual address in + input logic [riscv::VLEN-1:0] lsu_vaddr_i, // virtual address in input logic lsu_is_store_i, // the translation is requested by a store // if we need to walk the page table we can't grant in the same cycle // Cycle 0 @@ -59,9 +58,9 @@ import ariane_pkg::*; input logic mxr_i, // input logic flag_mprv_i, input logic [riscv::PPNW-1:0] satp_ppn_i, - input logic [ASID_WIDTH[0]-1:0] asid_i [HYP_EXT:0], - input logic [ASID_WIDTH[0]-1:0] asid_to_be_flushed_i [HYP_EXT:0], - input logic [riscv::VLEN-1:0] vaddr_to_be_flushed_i [HYP_EXT:0], + input logic [ASID_WIDTH-1:0] asid_i, + input logic [ASID_WIDTH-1:0] asid_to_be_flushed_i, + input logic [riscv::VLEN-1:0] vaddr_to_be_flushed_i, input logic flush_tlb_i, // Performance counters output logic itlb_miss_o, @@ -92,13 +91,12 @@ localparam type pte_cva6_t = struct packed { localparam type tlb_update_cva6_t = struct packed { // typedef struct packed { logic valid; // valid flag - logic [HYP_EXT:0][PT_LEVELS-2:0] is_page; // + logic [PT_LEVELS-2:0] is_page; // logic [VPN_LEN-1:0] vpn; // - logic [HYP_EXT:0][ASID_LEN-1:0] asid; // - pte_cva6_t [HYP_EXT:0] content; + logic [ASID_LEN-1:0] asid; // + pte_cva6_t content; } ; - logic iaccess_err; // insufficient privilege to access this instruction page logic daccess_err; // insufficient privilege to access this data page logic ptw_active; // PTW is currently walking a page table @@ -112,12 +110,12 @@ logic [riscv::VLEN-1:0] update_vaddr; tlb_update_cva6_t update_itlb, update_dtlb, update_shared_tlb; logic itlb_lu_access; -pte_cva6_t [HYP_EXT:0] itlb_content ; +pte_cva6_t itlb_content; logic [PT_LEVELS-2:0] itlb_is_page; logic itlb_lu_hit; logic dtlb_lu_access; -pte_cva6_t [HYP_EXT:0] dtlb_content; +pte_cva6_t dtlb_content; logic [PT_LEVELS-2:0] dtlb_is_page; logic dtlb_lu_hit; @@ -131,8 +129,6 @@ logic itlb_req; assign itlb_lu_access = icache_areq_i.fetch_req; assign dtlb_lu_access = lsu_req_i; -logic [riscv::VLEN-1:0] lu_vaddr_i [HYP_EXT:0]; -assign lu_vaddr_i[0]=icache_areq_i.fetch_vaddr; cva6_tlb #( .CVA6Cfg (CVA6Cfg), @@ -154,12 +150,11 @@ cva6_tlb #( .lu_asid_i (asid_i), .asid_to_be_flushed_i (asid_to_be_flushed_i), .vaddr_to_be_flushed_i(vaddr_to_be_flushed_i), - .lu_vaddr_i (lu_vaddr_i), + .lu_vaddr_i (icache_areq_i.fetch_vaddr), .lu_content_o (itlb_content), .lu_is_page_o(itlb_is_page), - .lu_hit_o (itlb_lu_hit), - .v_st_enbl_i(1) + .lu_hit_o (itlb_lu_hit) ); cva6_tlb #( @@ -186,15 +181,14 @@ cva6_tlb #( .lu_content_o (dtlb_content), .lu_is_page_o(dtlb_is_page), - .lu_hit_o (dtlb_lu_hit), - .v_st_enbl_i(1) + .lu_hit_o (dtlb_lu_hit) ); cva6_shared_tlb #( .CVA6Cfg (CVA6Cfg), .SHARED_TLB_DEPTH(64), .SHARED_TLB_WAYS (2), - .ASID_WIDTH (ASID_WIDTH[0]), + .ASID_WIDTH (ASID_WIDTH), .ASID_LEN (ASID_LEN), .VPN_LEN(VPN_LEN), .PT_LEVELS(PT_LEVELS), @@ -208,7 +202,7 @@ cva6_shared_tlb #( .enable_translation_i (enable_translation_i), .en_ld_st_translation_i(en_ld_st_translation_i), - .asid_i (asid_i[0]), + .asid_i (asid_i), // from TLBs // did we miss? .itlb_access_i(itlb_lu_access), @@ -217,7 +211,7 @@ cva6_shared_tlb #( .dtlb_access_i(dtlb_lu_access), .dtlb_hit_i (dtlb_lu_hit), - .dtlb_vaddr_i (lsu_vaddr_i[0]), + .dtlb_vaddr_i (lsu_vaddr_i), // to TLBs, update logic .itlb_update_o(update_itlb), @@ -238,7 +232,7 @@ cva6_shared_tlb #( cva6_ptw #( .CVA6Cfg (CVA6Cfg), - .ASID_WIDTH(ASID_WIDTH[0]), + .ASID_WIDTH(ASID_WIDTH), .VPN_LEN(VPN_LEN), .PT_LEVELS(PT_LEVELS), .pte_cva6_t(pte_cva6_t), @@ -263,7 +257,7 @@ cva6_ptw #( .update_vaddr_o(update_vaddr), - .asid_i(asid_i[0]), + .asid_i(asid_i), // from shared TLB // did we miss? @@ -318,8 +312,8 @@ localparam PPNWMin = (riscv::PPNW - 1 > 29) ? 29 : riscv::PPNW - 1; assign icache_areq_o.fetch_paddr [11:0] = icache_areq_i.fetch_vaddr[11:0]; assign icache_areq_o.fetch_paddr [riscv::PLEN-1:PPNWMin+1] = // (enable_translation_i) ? // - itlb_content[0].ppn[riscv::PPNW-1:(riscv::PPNW - (riscv::PLEN - PPNWMin-1))] : // - (riscv::PLEN-PPNWMin-1)'(icache_areq_i.fetch_vaddr[((riscv::PLEN > riscv::VLEN) ? riscv::VLEN : riscv::PLEN )-1:PPNWMin+1]); + itlb_content.ppn[riscv::PPNW-1:(riscv::PPNW - (riscv::PLEN - PPNWMin-1))] : // + riscv::PLEN'(icache_areq_i.fetch_vaddr[((riscv::PLEN > riscv::VLEN) ? riscv::VLEN : riscv::PLEN )-1:PPNWMin+1]); genvar a; generate @@ -327,7 +321,7 @@ generate for (a=0; a < PT_LEVELS-1; a++) begin assign icache_areq_o.fetch_paddr [PPNWMin-((VPN_LEN/PT_LEVELS)*(a)):PPNWMin-((VPN_LEN/PT_LEVELS)*(a+1))+1] = // (enable_translation_i && (|itlb_is_page[a:0]==0)) ? // - itlb_content[0].ppn [(riscv::PPNW - (riscv::PLEN - PPNWMin-1)-((VPN_LEN/PT_LEVELS)*(a))-1):(riscv::PPNW - (riscv::PLEN - PPNWMin-1)-((VPN_LEN/PT_LEVELS)*(a+1)))] : // + itlb_content.ppn [(riscv::PPNW - (riscv::PLEN - PPNWMin-1)-((VPN_LEN/PT_LEVELS)*(a))-1):(riscv::PPNW - (riscv::PLEN - PPNWMin-1)-((VPN_LEN/PT_LEVELS)*(a+1)))] : // icache_areq_i.fetch_vaddr[PPNWMin-((VPN_LEN/PT_LEVELS)*(a)):PPNWMin-((VPN_LEN/PT_LEVELS)*(a+1))+1]; end @@ -342,8 +336,8 @@ always_comb begin : instr_interface // 2. We got an access error because of insufficient permissions -> throw an access exception icache_areq_o.fetch_exception = '0; // Check whether we are allowed to access this memory region from a fetch perspective - iaccess_err = icache_areq_i.fetch_req && (((priv_lvl_i == riscv::PRIV_LVL_U) && ~itlb_content[0].u) - || ((priv_lvl_i == riscv::PRIV_LVL_S) && itlb_content[0].u)); + iaccess_err = icache_areq_i.fetch_req && (((priv_lvl_i == riscv::PRIV_LVL_U) && ~itlb_content.u) + || ((priv_lvl_i == riscv::PRIV_LVL_S) && itlb_content.u)); // MMU enabled: address from TLB, request delayed until hit. Error when TLB // hit and no access right or TLB hit and translated address not valid (e.g. @@ -450,11 +444,11 @@ assign lsu_paddr_o [11:0] = lsu_vaddr_q[11:0]; assign lsu_paddr_o [riscv::PLEN-1:PPNWMin+1] = (en_ld_st_translation_i && !misaligned_ex_q.valid) ? // dtlb_pte_q.ppn[riscv::PPNW-1:(riscv::PPNW - (riscv::PLEN - PPNWMin-1))] : // - (riscv::PLEN-PPNWMin-1)'(lsu_vaddr_q[((riscv::PLEN > riscv::VLEN) ? riscv::VLEN : riscv::PLEN )-1:PPNWMin+1]); + (riscv::PLEN-PPNWMin)'(lsu_vaddr_q[((riscv::PLEN > riscv::VLEN) ? riscv::VLEN : riscv::PLEN )-1:PPNWMin+1]); assign lsu_dtlb_ppn_o [11:0] = (en_ld_st_translation_i && !misaligned_ex_q.valid) ? // - dtlb_content[0].ppn[11:0] : // + dtlb_content.ppn[11:0] : // lsu_vaddr_n[23:12]; genvar i; @@ -468,14 +462,14 @@ genvar i; assign lsu_dtlb_ppn_o[PPNWMin-((VPN_LEN/PT_LEVELS)*(i)):PPNWMin-((VPN_LEN/PT_LEVELS)*(i+1))+1] = // (en_ld_st_translation_i && !misaligned_ex_q.valid && (|dtlb_is_page_q[i:0]==0)) ? // - dtlb_content[0].ppn[PPNWMin-((VPN_LEN/PT_LEVELS)*(i)):PPNWMin-((VPN_LEN/PT_LEVELS)*(i+1))+1] : // + dtlb_content.ppn[PPNWMin-((VPN_LEN/PT_LEVELS)*(i)):PPNWMin-((VPN_LEN/PT_LEVELS)*(i+1))+1] : // (en_ld_st_translation_i && !misaligned_ex_q.valid && (|dtlb_is_page_q[i:0]!=0)? lsu_vaddr_n[PPNWMin-((VPN_LEN/PT_LEVELS)*(i)):PPNWMin-((VPN_LEN/PT_LEVELS)*(i+1))+1]:// (VPN_LEN/PT_LEVELS)'(lsu_vaddr_n[((riscv::PLEN > riscv::VLEN) ? riscv::VLEN -1 : (24 + (VPN_LEN/PT_LEVELS)*(PT_LEVELS-i-1) ) -1): (riscv::PLEN > riscv::VLEN) ? 24 :24 + (VPN_LEN/PT_LEVELS)*(PT_LEVELS-i-2)])); end if(riscv::IS_XLEN64) begin assign lsu_dtlb_ppn_o[riscv::PPNW-1:PPNWMin+1] = (en_ld_st_translation_i && !misaligned_ex_q.valid) ? - dtlb_content[0].ppn[riscv::PPNW-1:PPNWMin+1] : + dtlb_content.ppn[riscv::PPNW-1:PPNWMin+1] : lsu_vaddr_n[riscv::PLEN-1:PPNWMin+1] ; end @@ -483,7 +477,7 @@ genvar i; // The data interface is simpler and only consists of a request/response interface always_comb begin : data_interface // save request and DTLB response - lsu_vaddr_n = lsu_vaddr_i[0]; + lsu_vaddr_n = lsu_vaddr_i; lsu_req_n = lsu_req_i; misaligned_ex_n = misaligned_ex_i; dtlb_pte_n = dtlb_content; diff --git a/core/mmu_unify/cva6_ptw.sv b/core/mmu_unify/cva6_ptw.sv index 425e12e679..e7457c34a0 100644 --- a/core/mmu_unify/cva6_ptw.sv +++ b/core/mmu_unify/cva6_ptw.sv @@ -141,7 +141,7 @@ genvar x; for (x=0; x < PT_LEVELS-1; x++) begin // update the correct page table level - assign shared_tlb_update_o.is_page[0][x] = (ptw_lvl_q == (x)); + assign shared_tlb_update_o.is_page[x] = (ptw_lvl_q == (x)); // check if the ppn is correctly aligned: // 6. If i > 0 and pa.ppn[i − 1 : 0] != 0, this is a misaligned superpage; stop and raise a page-fault @@ -157,7 +157,7 @@ genvar x; // output the correct ASID - assign shared_tlb_update_o.asid[0] = tlb_update_asid_q; + assign shared_tlb_update_o.asid = tlb_update_asid_q; // set the global mapping bit assign shared_tlb_update_o.content = pte | (global_mapping_q << 5); diff --git a/core/mmu_unify/cva6_shared_tlb.sv b/core/mmu_unify/cva6_shared_tlb.sv index 49d1c26843..e26b7f63ff 100644 --- a/core/mmu_unify/cva6_shared_tlb.sv +++ b/core/mmu_unify/cva6_shared_tlb.sv @@ -237,14 +237,14 @@ module cva6_shared_tlb if (itlb_req_q) begin itlb_update_o.valid = 1'b1; itlb_update_o.vpn = itlb_vpn_q; - itlb_update_o.is_page[0] = shared_tag_rd[i].is_page; - itlb_update_o.asid[0] = tlb_update_asid_q; + itlb_update_o.is_page = shared_tag_rd[i].is_page; + itlb_update_o.asid = tlb_update_asid_q; itlb_update_o.content = pte[i]; end else if (dtlb_req_q) begin dtlb_update_o.valid = 1'b1; dtlb_update_o.vpn = dtlb_vpn_q; - dtlb_update_o.is_page[0] = shared_tag_rd[i].is_page; - dtlb_update_o.asid[0] = tlb_update_asid_q; + dtlb_update_o.is_page = shared_tag_rd[i].is_page; + dtlb_update_o.asid = tlb_update_asid_q; dtlb_update_o.content = pte[i]; end end @@ -300,10 +300,10 @@ module cva6_shared_tlb end end //update_flush - assign shared_tag_wr.asid = shared_tlb_update_i.asid[0]; + assign shared_tag_wr.asid = shared_tlb_update_i.asid; // assign shared_tag_wr.vpn[1] = shared_tlb_update_i.vpn[19:10]; // assign shared_tag_wr.vpn[0] = shared_tlb_update_i.vpn[9:0]; - assign shared_tag_wr.is_page = shared_tlb_update_i.is_page[0]; + assign shared_tag_wr.is_page = shared_tlb_update_i.is_page; genvar z; @@ -318,7 +318,7 @@ module cva6_shared_tlb assign tag_wr_data = shared_tag_wr; assign pte_wr_addr = shared_tlb_update_i.vpn[$clog2(SHARED_TLB_DEPTH)-1:0]; - assign pte_wr_data = shared_tlb_update_i.content[0]; + assign pte_wr_data = shared_tlb_update_i.content; assign way_valid = shared_tag_valid_q[shared_tlb_update_i.vpn[$clog2(SHARED_TLB_DEPTH)-1:0]]; assign repl_way = (all_ways_valid) ? rnd_way : inv_way; diff --git a/core/mmu_unify/cva6_tlb.sv b/core/mmu_unify/cva6_tlb.sv index e28fb552fa..60f8d2bc83 100644 --- a/core/mmu_unify/cva6_tlb.sv +++ b/core/mmu_unify/cva6_tlb.sv @@ -12,15 +12,15 @@ // Date: 20/11/2023 // // Description: Translation Lookaside Buffer, parameterizable to Sv32 or Sv39 , -// or sv39x4 fully set-associative +// fully set-associative // This module is an merge of the Sv32 TLB developed by Sebastien -// Jacq (Thales Research & Technology), the Sv39 TLB developed -// by Florian Zaruba and David Schaffenrath and the Sv39x4 by Bruno Sá. +// Jacq (Thales Research & Technology) and the Sv39 TLB developed +// by Florian Zaruba and David Schaffenrath. // // =========================================================================== // // Revisions : // Date Version Author Description -// 2023-12-13 0.2 A.Gonzalez Generic TLB for CVA6 with Hypervisor support +// 2023-11-20 0.1 A.Gonzalez Generic TLB for CVA6 // =========================================================================== // module cva6_tlb @@ -29,43 +29,39 @@ parameter type pte_cva6_t = logic, parameter type tlb_update_cva6_t = logic, parameter config_pkg::cva6_cfg_t CVA6Cfg = config_pkg::cva6_cfg_empty, - parameter logic HYP_EXT = 0, parameter int unsigned TLB_ENTRIES = 4, - parameter int unsigned ASID_WIDTH [HYP_EXT:0] = {1}, //[vmid_width,asid_width] - parameter int unsigned ASID_LEN[HYP_EXT:0] = 1, //[vmid_len,asid_len] + parameter int unsigned ASID_WIDTH = 1, + parameter int unsigned ASID_LEN = 1, parameter int unsigned VPN_LEN = 1, parameter int unsigned PT_LEVELS = 1 ) ( input logic clk_i, // Clock input logic rst_ni, // Asynchronous reset active low - input logic [HYP_EXT*2:0] flush_i, // Flush signal [g_stage,vs stage, normal translation signal] - input logic [HYP_EXT*2:0] v_st_enbl_i, // v_i,g-stage enabled, s-stage enabled + input logic flush_i, // Flush signal // Update TLB input tlb_update_cva6_t update_i, // Lookup signals input logic lu_access_i, - input logic [ASID_WIDTH[0]-1:0] lu_asid_i [HYP_EXT:0], //[lu_vmid,lu_asid] - input logic [riscv::VLEN-1:0] lu_vaddr_i [HYP_EXT:0], //[gp_addr,vaddr] - output pte_cva6_t [HYP_EXT:0] lu_content_o , - input logic [ASID_WIDTH[0]-1:0] asid_to_be_flushed_i [HYP_EXT:0], //[vmid,asid] - input logic [riscv::VLEN-1:0] vaddr_to_be_flushed_i [HYP_EXT:0], // [gpaddr,vaddr] + input logic [ASID_WIDTH-1:0] lu_asid_i, + input logic [riscv::VLEN-1:0] lu_vaddr_i, + output pte_cva6_t lu_content_o, + input logic [ASID_WIDTH-1:0] asid_to_be_flushed_i, + input logic [riscv::VLEN-1:0] vaddr_to_be_flushed_i, output logic [PT_LEVELS-2:0] lu_is_page_o, output logic lu_hit_o ); // Sv32 defines two levels of page tables, Sv39 defines 3 struct packed { - logic [HYP_EXT:0][ASID_LEN-1:0] asid; - logic [PT_LEVELS+HYP_EXT-1:0][(VPN_LEN/PT_LEVELS)-1:0] vpn; - logic [HYP_EXT:0][PT_LEVELS-2:0] is_page; - logic [HYP_EXT*2:0] v_st_enbl_i; // v_i,g-stage enabled, s-stage enabled + logic [ASID_LEN-1:0] asid; + logic [PT_LEVELS-1:0][(VPN_LEN/PT_LEVELS)-1:0] vpn; + logic [PT_LEVELS-2:0] is_page; logic valid; } [TLB_ENTRIES-1:0] tags_q, tags_n; - pte_cva6_t [HYP_EXT:0][TLB_ENTRIES-1:0] content_q , content_n; + pte_cva6_t [TLB_ENTRIES-1:0] content_q, content_n; logic [TLB_ENTRIES-1:0][PT_LEVELS-1:0] vpn_match; - logic [TLB_ENTRIES-1:0][HYP_EXT:0] asid_match; logic [TLB_ENTRIES-1:0][PT_LEVELS-1:0] page_match; logic [TLB_ENTRIES-1:0][PT_LEVELS-1:0] level_match; logic [TLB_ENTRIES-1:0][PT_LEVELS-1:0] vaddr_vpn_match; @@ -82,23 +78,21 @@ //AND the page_match is also set //At level 0 the page match is always set, so this level will have a match //if all vpn levels match - genvar i,x,z; + genvar i,x; generate for (i=0; i < TLB_ENTRIES; i++) begin - - for(z=0;z= 1) + assert (ASID_WIDTH >= 1) else begin $error("ASID width must be at least 1"); $stop();