proj
This commit is contained in:
		
							parent
							
								
									085c1023a6
								
							
						
					
					
						commit
						6a914a6eaa
					
				
					 1 changed files with 93 additions and 86 deletions
				
			
		|  | @ -53,8 +53,8 @@ | |||
| 
 | ||||
| // 7-Segment patterns
 | ||||
| const unsigned char seven_seg[16] = { | ||||
|     0x3F,0x06,0x5B,0x4F,0x66,0x6D,0x7D,0x07, | ||||
|     0x7F,0x6F,0x77,0x7C,0x39,0x5E,0x79,0x71 | ||||
|     0x3F, 0x06, 0x5B, 0x4F, 0x66, 0x6D, 0x7D, 0x07, | ||||
|     0x7F, 0x6F, 0x77, 0x7C, 0x39, 0x5E, 0x79, 0x71 | ||||
| }; | ||||
| 
 | ||||
| // Keypad defines
 | ||||
|  | @ -65,16 +65,16 @@ const unsigned char seven_seg[16] = { | |||
| 
 | ||||
| // 7-Segment defines
 | ||||
| #define SEG_SHIFT 4 | ||||
| #define DIGIT_EN (1<<23) | ||||
| #define DIGIT_EN (1 << 23) | ||||
| 
 | ||||
| // LCD defines
 | ||||
| #define LCD_DATA_SHIFT 23 | ||||
| #define LCD_DATA_MASK (0x0F << LCD_DATA_SHIFT) | ||||
| #define LCD_RS (1<<27) | ||||
| #define LCD_EN (1<<28) | ||||
| #define LCD_RS (1 << 27) | ||||
| #define LCD_EN (1 << 28) | ||||
| 
 | ||||
| // Mode button defines
 | ||||
| #define MODE_BUTTON (1<<12) | ||||
| #define MODE_BUTTON (1 << 12) | ||||
| 
 | ||||
| // Calculator states
 | ||||
| #define MODE_BIN 2 | ||||
|  | @ -94,17 +94,28 @@ int result = 0; | |||
| unsigned int current_base = MODE_DEC; | ||||
| unsigned int operation = OP_NONE; | ||||
| unsigned char result_displayed = 0; | ||||
| unsigned int key = 0xFF; | ||||
| unsigned int last_key = 0xFF; | ||||
| unsigned int stable = 0; | ||||
| unsigned int mode_held = 0; | ||||
| unsigned int mode_press_counter = 0; | ||||
| unsigned int key_pressed_with_mode = 0; | ||||
| unsigned int mode_current = 0; | ||||
| int is_operator_key = 0; | ||||
| int operator_mode = 0; | ||||
| unsigned int seg_pattern = 0; | ||||
| 
 | ||||
| void delay(volatile unsigned int d){ | ||||
|     while(d--) __NOP(); | ||||
| void delay(volatile unsigned int d) { | ||||
|     while(d--) | ||||
|         __NOP(); | ||||
| } | ||||
| 
 | ||||
| void lcd_delay(unsigned long r){ | ||||
| void lcd_delay(unsigned long r) { | ||||
|     unsigned long i; | ||||
|     for(i=0; i<r; i++); | ||||
|     for(i = 0; i < r; i++); | ||||
| } | ||||
| 
 | ||||
| void lcd_write_nibble(unsigned char nibble, unsigned char is_data){ | ||||
| void lcd_write_nibble(unsigned char nibble, unsigned char is_data) { | ||||
|     unsigned long temp; | ||||
|     temp = (nibble & 0x0F) << LCD_DATA_SHIFT; | ||||
|     LPC_GPIO0->FIOPIN = (LPC_GPIO0->FIOPIN & ~LCD_DATA_MASK) | temp; | ||||
|  | @ -120,17 +131,17 @@ void lcd_write_nibble(unsigned char nibble, unsigned char is_data){ | |||
|     lcd_delay(500000); | ||||
| } | ||||
| 
 | ||||
| void lcd_cmd(unsigned char cmd){ | ||||
| void lcd_cmd(unsigned char cmd) { | ||||
|     lcd_write_nibble(cmd >> 4, 0); | ||||
|     lcd_write_nibble(cmd, 0); | ||||
| } | ||||
| 
 | ||||
| void lcd_data(unsigned char data){ | ||||
| void lcd_data(unsigned char data) { | ||||
|     lcd_write_nibble(data >> 4, 1); | ||||
|     lcd_write_nibble(data, 1); | ||||
| } | ||||
| 
 | ||||
| void lcd_init(void){ | ||||
| void lcd_init(void) { | ||||
|     lcd_delay(5000000); | ||||
|     lcd_write_nibble(0x03, 0); | ||||
|     lcd_delay(500000); | ||||
|  | @ -148,23 +159,23 @@ void lcd_init(void){ | |||
|     lcd_cmd(0x06); | ||||
| } | ||||
| 
 | ||||
| void lcd_print_str(const char* str){ | ||||
|     while(*str){ | ||||
| void lcd_print_str(const char* str) { | ||||
|     while(*str) { | ||||
|         lcd_data(*str++); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| void lcd_print_num(int num, unsigned int base){ | ||||
| void lcd_print_num(int num, unsigned int base) { | ||||
|     char buffer[17]; | ||||
|     int i = 0; | ||||
| 
 | ||||
|     if(base != MODE_DEC){ | ||||
|     if(base != MODE_DEC) { | ||||
|         unsigned int unum = (unsigned int)num; | ||||
|         if(unum == 0){ | ||||
|         if(unum == 0) { | ||||
|             lcd_data('0'); | ||||
|             return; | ||||
|         } | ||||
|         while(unum > 0 && i < 16){ | ||||
|         while(unum > 0 && i < 16) { | ||||
|             unsigned int digit = unum % base; | ||||
|             if(digit < 10) | ||||
|                 buffer[i++] = '0' + digit; | ||||
|  | @ -173,28 +184,28 @@ void lcd_print_num(int num, unsigned int base){ | |||
|             unum = unum / base; | ||||
|         } | ||||
|     } else { | ||||
|         if(num < 0){ | ||||
|         if(num < 0) { | ||||
|             lcd_data('-'); | ||||
|             num = -num; | ||||
|         } | ||||
| 
 | ||||
|         if(num == 0){ | ||||
|         if(num == 0) { | ||||
|             lcd_data('0'); | ||||
|             return; | ||||
|         } | ||||
| 
 | ||||
|         while(num > 0 && i < 16){ | ||||
|         while(num > 0 && i < 16) { | ||||
|             buffer[i++] = '0' + (num % 10); | ||||
|             num = num / 10; | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     while(i > 0){ | ||||
|     while(i > 0) { | ||||
|         lcd_data(buffer[--i]); | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| void display_mode(void){ | ||||
| void display_mode(void) { | ||||
|     lcd_cmd(0x80); | ||||
|     lcd_print_str("Mode: "); | ||||
|     if(current_base == MODE_BIN) | ||||
|  | @ -207,27 +218,28 @@ void display_mode(void){ | |||
|         lcd_print_str("HEX     "); | ||||
| } | ||||
| 
 | ||||
| void display_status(void){ | ||||
| void display_status(void) { | ||||
|     lcd_cmd(0xC0); | ||||
| 
 | ||||
|     if(operation != OP_NONE){ | ||||
|     if(operation != OP_NONE) { | ||||
|         // Show operation pending
 | ||||
|         lcd_print_str("Op:"); | ||||
|         if(operation == OP_ADD) lcd_data('+'); | ||||
|         else if(operation == OP_SUB) lcd_data('-'); | ||||
|         else if(operation == OP_MUL) lcd_data('*'); | ||||
|         if(operation == OP_ADD) | ||||
|             lcd_data('+'); | ||||
|         else if(operation == OP_SUB) | ||||
|             lcd_data('-'); | ||||
|         else if(operation == OP_MUL) | ||||
|             lcd_data('*'); | ||||
| 
 | ||||
|         lcd_print_str(" Val:"); | ||||
|         lcd_print_num(stored_num, current_base); | ||||
|         lcd_print_str("    "); | ||||
|     } | ||||
|     else if(result_displayed){ | ||||
|     } else if(result_displayed) { | ||||
|         // Show result
 | ||||
|         lcd_print_str("Res: "); | ||||
|         lcd_print_num(result, current_base); | ||||
|         lcd_print_str("        "); | ||||
|     } | ||||
|     else { | ||||
|     } else { | ||||
|         // Show current input
 | ||||
|         lcd_print_str("Inp: "); | ||||
|         lcd_print_num(input_num, current_base); | ||||
|  | @ -235,22 +247,22 @@ void display_status(void){ | |||
|     } | ||||
| } | ||||
| 
 | ||||
| unsigned int scan_keypad(void){ | ||||
| unsigned int scan_keypad(void) { | ||||
|     unsigned int col, row; | ||||
|     unsigned int row_bits; | ||||
| 
 | ||||
|     for(col=0; col<4; col++){ | ||||
|     for(col = 0; col < 4; col++) { | ||||
|         LPC_GPIO0->FIOSET = COL_MASK; | ||||
|         delay(50); | ||||
|         LPC_GPIO0->FIOCLR = (1 << (COL_BASE + col)); | ||||
|         delay(200); | ||||
|         row_bits = (LPC_GPIO0->FIOPIN & ROW_MASK) >> ROW_BASE; | ||||
| 
 | ||||
|         if(row_bits != 0x0F){ | ||||
|             for(row=0; row<4; row++){ | ||||
|                 if((row_bits & (1<<row)) == 0){ | ||||
|         if(row_bits != 0x0F) { | ||||
|             for(row = 0; row < 4; row++) { | ||||
|                 if((row_bits & (1 << row)) == 0) { | ||||
|                     LPC_GPIO0->FIOSET = COL_MASK; | ||||
|                     return col*4 + row; | ||||
|                     return col * 4 + row; | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | @ -259,19 +271,23 @@ unsigned int scan_keypad(void){ | |||
|     return 0xFF; | ||||
| } | ||||
| 
 | ||||
| unsigned int is_mode_button_pressed(void){ | ||||
| unsigned int is_mode_button_pressed(void) { | ||||
|     return ((LPC_GPIO2->FIOPIN & MODE_BUTTON) == 0) ? 1 : 0; | ||||
| } | ||||
| 
 | ||||
| unsigned int is_valid_digit(unsigned int key){ | ||||
|     if(key >= 16) return 0; | ||||
|     if(current_base == MODE_BIN && key >= 2) return 0; | ||||
|     if(current_base == MODE_OCT && key >= 8) return 0; | ||||
|     if(current_base == MODE_DEC && key >= 10) return 0; | ||||
| unsigned int is_valid_digit(unsigned int key) { | ||||
|     if(key >= 16) | ||||
|         return 0; | ||||
|     if(current_base == MODE_BIN && key >= 2) | ||||
|         return 0; | ||||
|     if(current_base == MODE_OCT && key >= 8) | ||||
|         return 0; | ||||
|     if(current_base == MODE_DEC && key >= 10) | ||||
|         return 0; | ||||
|     return 1; | ||||
| } | ||||
| 
 | ||||
| void change_mode(void){ | ||||
| void change_mode(void) { | ||||
|     if(current_base == MODE_DEC) | ||||
|         current_base = MODE_BIN; | ||||
|     else if(current_base == MODE_BIN) | ||||
|  | @ -291,13 +307,7 @@ void change_mode(void){ | |||
|     display_status(); | ||||
| } | ||||
| 
 | ||||
| int main(void){ | ||||
|     unsigned int key, last_key = 0xFF; | ||||
|     unsigned int stable = 0; | ||||
|     unsigned int mode_held = 0; | ||||
|     unsigned int mode_press_counter = 0; | ||||
|     unsigned int key_pressed_with_mode = 0; | ||||
| 
 | ||||
| int main(void) { | ||||
|     // Configure pins
 | ||||
|     LPC_PINCON->PINSEL0 = 0; | ||||
|     LPC_PINCON->PINSEL1 = 0; | ||||
|  | @ -323,28 +333,30 @@ int main(void){ | |||
|     display_mode(); | ||||
|     display_status(); | ||||
| 
 | ||||
|     for(;;){ | ||||
|     for(;;) { | ||||
|         // Check if MODE button is currently pressed
 | ||||
|         unsigned int mode_current = is_mode_button_pressed(); | ||||
|         mode_current = is_mode_button_pressed(); | ||||
| 
 | ||||
|         // Scan keypad
 | ||||
|         key = scan_keypad(); | ||||
| 
 | ||||
|         // Debounce keypad
 | ||||
|         if(key == last_key){ | ||||
|             if(stable < 5) stable++; | ||||
|         if(key == last_key) { | ||||
|             if(stable < 5) | ||||
|                 stable++; | ||||
|         } else { | ||||
|             last_key = key; | ||||
|             stable = 0; | ||||
|         } | ||||
| 
 | ||||
|         // Track MODE button hold
 | ||||
|         if(mode_current){ | ||||
|         if(mode_current) { | ||||
|             mode_press_counter++; | ||||
|             mode_held = 1; | ||||
|         } else { | ||||
|             // MODE button released
 | ||||
|             if(mode_held && !key_pressed_with_mode && mode_press_counter < 50){ | ||||
|             if(mode_held && !key_pressed_with_mode && | ||||
|                mode_press_counter < 50) { | ||||
|                 // Short press without key combo = change mode
 | ||||
|                 change_mode(); | ||||
|             } | ||||
|  | @ -354,47 +366,44 @@ int main(void){ | |||
|         } | ||||
| 
 | ||||
|         // Handle keypad input
 | ||||
|         if(stable == 3 && key != 0xFF){ | ||||
|         if(stable == 3 && key != 0xFF) { | ||||
|             // Key pressed and stable
 | ||||
| 
 | ||||
|             // Check if this is an operator key (A-E)
 | ||||
|             int is_operator_key = (key >= 10 && key <= 14); | ||||
|             is_operator_key = (key >= 10 && key <= 14); | ||||
| 
 | ||||
|             // In HEX mode, operators require MODE button
 | ||||
|             // In other modes, operator keys work directly
 | ||||
|             int operator_mode = 0; | ||||
|             if(current_base == MODE_HEX){ | ||||
|             operator_mode = 0; | ||||
|             if(current_base == MODE_HEX) { | ||||
|                 operator_mode = (mode_held && is_operator_key); | ||||
|             } else { | ||||
|                 operator_mode = is_operator_key; | ||||
|             } | ||||
| 
 | ||||
|             if(operator_mode){ | ||||
|             if(operator_mode) { | ||||
|                 // OPERATOR INPUT
 | ||||
|                 key_pressed_with_mode = 1; | ||||
| 
 | ||||
|                 if(key == 10){  // A = Addition
 | ||||
|                 if(key == 10) {  // A = Addition
 | ||||
|                     stored_num = input_num; | ||||
|                     operation = OP_ADD; | ||||
|                     input_num = 0; | ||||
|                     result_displayed = 0; | ||||
|                     display_status(); | ||||
|                 } | ||||
|                 else if(key == 11){  // B = Subtraction
 | ||||
|                 } else if(key == 11) {  // B = Subtraction
 | ||||
|                     stored_num = input_num; | ||||
|                     operation = OP_SUB; | ||||
|                     input_num = 0; | ||||
|                     result_displayed = 0; | ||||
|                     display_status(); | ||||
|                 } | ||||
|                 else if(key == 12){  // C = Multiplication
 | ||||
|                 } else if(key == 12) {  // C = Multiplication
 | ||||
|                     stored_num = input_num; | ||||
|                     operation = OP_MUL; | ||||
|                     input_num = 0; | ||||
|                     result_displayed = 0; | ||||
|                     display_status(); | ||||
|                 } | ||||
|                 else if(key == 13){  // D = Equals
 | ||||
|                 } else if(key == 13) {  // D = Equals
 | ||||
|                     if(operation == OP_ADD) | ||||
|                         result = stored_num + input_num; | ||||
|                     else if(operation == OP_SUB) | ||||
|  | @ -408,8 +417,7 @@ int main(void){ | |||
|                     operation = OP_NONE; | ||||
|                     result_displayed = 1; | ||||
|                     display_status(); | ||||
|                 } | ||||
|                 else if(key == 14){  // E = Clear
 | ||||
|                 } else if(key == 14) {  // E = Clear
 | ||||
|                     input_num = 0; | ||||
|                     stored_num = 0; | ||||
|                     operation = OP_NONE; | ||||
|  | @ -417,14 +425,15 @@ int main(void){ | |||
|                     result_displayed = 0; | ||||
|                     display_status(); | ||||
|                 } | ||||
|             } | ||||
|             else { | ||||
|             } else { | ||||
|                 // DIGIT INPUT MODE
 | ||||
|                 if(is_valid_digit(key)){ | ||||
|                 if(is_valid_digit(key)) { | ||||
|                     result_displayed = 0; | ||||
|                     input_num = input_num * current_base + key; | ||||
|                     if(input_num > 32767) input_num = input_num % 32768; | ||||
|                     if(input_num < -32768) input_num = -32768; | ||||
|                     if(input_num > 32767) | ||||
|                         input_num = input_num % 32768; | ||||
|                     if(input_num < -32768) | ||||
|                         input_num = -32768; | ||||
|                     display_status(); | ||||
|                 } | ||||
|             } | ||||
|  | @ -433,18 +442,14 @@ int main(void){ | |||
|         } | ||||
| 
 | ||||
|         // ===== STATE MACHINE DISPLAY ON 7-SEGMENT =====
 | ||||
|         unsigned int seg_pattern; | ||||
| 
 | ||||
|         if(operation == OP_NONE && result_displayed == 0) { | ||||
|             seg_pattern = 0x06;  // Display '1'
 | ||||
|         } | ||||
|         else if(operation != OP_NONE) { | ||||
|         } else if(operation != OP_NONE) { | ||||
|             seg_pattern = 0x5B | 0x80;  // Display '2' with DP
 | ||||
|         } | ||||
|         else if(result_displayed == 1) { | ||||
|         } else if(result_displayed == 1) { | ||||
|             seg_pattern = 0x4F | 0x80;  // Display '3' with DP
 | ||||
|         } | ||||
|         else { | ||||
|         } else { | ||||
|             seg_pattern = 0x3F;  // Display '0'
 | ||||
|         } | ||||
| 
 | ||||
|  | @ -455,4 +460,6 @@ int main(void){ | |||
| 
 | ||||
|         delay(3000); | ||||
|     } | ||||
| 
 | ||||
|     return 0; | ||||
| } | ||||
|  |  | |||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue