Merge branch 'riscv' of g.csail.mit.edu:xv6-dev into riscv
This commit is contained in:
		
						commit
						005773c0c3
					
				
					 2 changed files with 27 additions and 12 deletions
				
			
		| 
						 | 
					@ -5,6 +5,7 @@
 | 
				
			||||||
#include "defs.h"
 | 
					#include "defs.h"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void main();
 | 
					void main();
 | 
				
			||||||
 | 
					void timerinit();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// entry.S needs one stack per CPU.
 | 
					// entry.S needs one stack per CPU.
 | 
				
			||||||
__attribute__ ((aligned (16))) char stack0[4096 * NCPU];
 | 
					__attribute__ ((aligned (16))) char stack0[4096 * NCPU];
 | 
				
			||||||
| 
						 | 
					@ -36,15 +37,31 @@ start()
 | 
				
			||||||
  w_medeleg(0xffff);
 | 
					  w_medeleg(0xffff);
 | 
				
			||||||
  w_mideleg(0xffff);
 | 
					  w_mideleg(0xffff);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  // ask for clock interrupts.
 | 
				
			||||||
 | 
					  timerinit();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  // keep each CPU's hartid in its tp register, for cpuid().
 | 
				
			||||||
  int id = r_mhartid();
 | 
					  int id = r_mhartid();
 | 
				
			||||||
 | 
					  w_tp(id);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					  // switch to supervisor mode and jump to main().
 | 
				
			||||||
 | 
					  asm volatile("mret");
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// set up to receive timer interrupts in machine mode,
 | 
					// set up to receive timer interrupts in machine mode,
 | 
				
			||||||
// which arrive at timervec in kernelvec.S,
 | 
					// which arrive at timervec in kernelvec.S,
 | 
				
			||||||
// which turns them into software interrupts for
 | 
					// which turns them into software interrupts for
 | 
				
			||||||
// devintr() in trap.c.
 | 
					// devintr() in trap.c.
 | 
				
			||||||
 | 
					void
 | 
				
			||||||
 | 
					timerinit()
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
					  // each CPU has a separate source of timer interrupts.
 | 
				
			||||||
 | 
					  int id = r_mhartid();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  // ask the CLINT for a timer interrupt.
 | 
					  // ask the CLINT for a timer interrupt.
 | 
				
			||||||
  int interval = 1000000; // cycles; about 1/10th second in qemu.
 | 
					  int interval = 1000000; // cycles; about 1/10th second in qemu.
 | 
				
			||||||
  *(uint64*)CLINT_MTIMECMP(id) = *(uint64*)CLINT_MTIME + interval;
 | 
					  *(uint64*)CLINT_MTIMECMP(id) = *(uint64*)CLINT_MTIME + interval;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  // prepare information in scratch[] for timervec.
 | 
					  // prepare information in scratch[] for timervec.
 | 
				
			||||||
  // scratch[0..3] : space for timervec to save registers.
 | 
					  // scratch[0..3] : space for timervec to save registers.
 | 
				
			||||||
  // scratch[4] : address of CLINT MTIMECMP register.
 | 
					  // scratch[4] : address of CLINT MTIMECMP register.
 | 
				
			||||||
| 
						 | 
					@ -53,16 +70,13 @@ start()
 | 
				
			||||||
  scratch[4] = CLINT_MTIMECMP(id);
 | 
					  scratch[4] = CLINT_MTIMECMP(id);
 | 
				
			||||||
  scratch[5] = interval;
 | 
					  scratch[5] = interval;
 | 
				
			||||||
  w_mscratch((uint64)scratch);
 | 
					  w_mscratch((uint64)scratch);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  // set the machine-mode trap handler.
 | 
					  // set the machine-mode trap handler.
 | 
				
			||||||
  w_mtvec((uint64)timervec);
 | 
					  w_mtvec((uint64)timervec);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  // enable machine-mode interrupts.
 | 
					  // enable machine-mode interrupts.
 | 
				
			||||||
  w_mstatus(r_mstatus() | MSTATUS_MIE);
 | 
					  w_mstatus(r_mstatus() | MSTATUS_MIE);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  // enable machine-mode timer interrupts.
 | 
					  // enable machine-mode timer interrupts.
 | 
				
			||||||
  w_mie(r_mie() | MIE_MTIE);
 | 
					  w_mie(r_mie() | MIE_MTIE);
 | 
				
			||||||
 | 
					 | 
				
			||||||
  // keep each CPU's hartid in its tp register, for cpuid().
 | 
					 | 
				
			||||||
  w_tp(id);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  // switch to supervisor mode and jump to main().
 | 
					 | 
				
			||||||
  asm volatile("mret");
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -15,8 +15,9 @@
 | 
				
			||||||
// address of one of the registers.
 | 
					// address of one of the registers.
 | 
				
			||||||
#define Reg(reg) ((volatile unsigned char *)(UART0 + reg))
 | 
					#define Reg(reg) ((volatile unsigned char *)(UART0 + reg))
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// the registers. some have different meanings for
 | 
					// the UART control registers.
 | 
				
			||||||
// read and write.
 | 
					// some have different meanings for
 | 
				
			||||||
 | 
					// read vs write.
 | 
				
			||||||
// http://byterunner.com/16550.html
 | 
					// http://byterunner.com/16550.html
 | 
				
			||||||
#define RHR 0 // receive holding register (for input bytes)
 | 
					#define RHR 0 // receive holding register (for input bytes)
 | 
				
			||||||
#define THR 0 // transmit holding register (for output bytes)
 | 
					#define THR 0 // transmit holding register (for output bytes)
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue