| 
	
	|  |  |  
	
		| View previous topic :: View next topic |  
		| Author | Message |  
		| Joe Porthouse 
 
 
 Joined: 21 Sep 2003
 Posts: 8
 Location: orlando, FL
 
 
			      
 
 | 
			
				| Changing baud rates, parity, etc. at run time? |  
				|  Posted: Mon Feb 28, 2005 12:37 pm |   |  
				| 
 |  
				| #use rs232(baud= 38400,xmit=PIN_C6,rcv=PIN_C7, errors) 
 This allows a single set of communication parameters to be configured. My problem is that our project uses dip switchs to select the communication parameters to be used.
 
 Does PCH has some method to support this?
 
 In the meantime I am attempting to write directly to the PIC UART registers to modify the configuration.  I believe I may have a problem when modifying the parity or data size without the PCH routine's knowledge.
 
 Any ideas?
 |  |  
		|  |  
		| dyeatman 
 
 
 Joined: 06 Sep 2003
 Posts: 1968
 Location: Norman, OK
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Mon Feb 28, 2005 12:51 pm |   |  
				| 
 |  
				| Joe, I had this requirement in one of my projects.  I had a eight DIP switch and used an eight input mux chip to scan the switches and store the settings (set flags), then I  executed the appropriate #use RS232 statement etc..
 
 You can use (reuse) the RS232 statement multiple times in the code and change the settings on the fly whenever it is required.  There is a command to change just the baud rate (SET_UART_SPEED()) for the internal UART if that's all you need to change.
 
 That approach requires 4 pins.  Another option to save pins is using a Phillips I2C expander chip to get the DIP switch settings via I2C.
 |  |  
		|  |  
		| Joe Porthouse 
 
 
 Joined: 21 Sep 2003
 Posts: 8
 Location: orlando, FL
 
 
			      
 
 | 
			
				|  |  
				|  Posted: Mon Feb 28, 2005 1:23 pm |   |  
				| 
 |  
				| Thanks for the info.  I did not see the SET_UART_SPEED() functon before. 
 I knew you could use multiple #use RS232 statements to use more then one serial port, but when I attempt to use multiple #use RS232 only the last listed in the file gets applied.
 
  	  | Code: |  	  | // setup baud rate
 switch (a)
 {
 case 0:
 // 16Mhz,   1200 Baud
 #use rs232(baud=  1200,xmit=PIN_C6,rcv=PIN_C7,parity=N,bits=8,errors)
 break;
 case 1:
 // 16Mhz,   2400 Baud
 #use rs232(baud=  2400,xmit=PIN_C6,rcv=PIN_C7, errors)
 break;
 case 2:
 // 16Mhz,   4800 Baud
 #use rs232(baud=  4800,xmit=PIN_C6,rcv=PIN_C7, errors)
 break;
 case 3:
 // 16Mhz,   9600 Baud
 #use rs232(baud=  9600,xmit=PIN_C6,rcv=PIN_C7, errors)
 break;
 case 4:
 // 16Mhz,  19200 Baud
 #use rs232(baud= 19200,xmit=PIN_C6,rcv=PIN_C7, errors)
 break;
 case 5:
 // 16Mhz,  38400 Baud
 #use rs232(baud= 38400,xmit=PIN_C6,rcv=PIN_C7, errors)
 break;
 case 6:
 // 16Mhz,  57600 Baud
 #use rs232(baud= 57600,xmit=PIN_C6,rcv=PIN_C7, errors)
 break;
 case 7:
 // 16Mhz, 115200 Baud
 #use rs232(baud=115200,xmit=PIN_C6,rcv=PIN_C7, errors)
 break;
 }
 | 
 
 (I realise SET_UART_SPEED()could have been used in the above example, but my final application requires parity, etc. to also be selectable).
 
 In my listing file it also shows that NO instructions were generated at each #use RS232.  Can you post a copy of how you got this to work?
 |  |  
		|  |  
		| dyeatman 
 
 
 Joined: 06 Sep 2003
 Posts: 1968
 Location: Norman, OK
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Mon Feb 28, 2005 2:37 pm |   |  
				| 
 |  
				| The way it is supposed to work is that the last #use RS232 encountered is the one in effect until the next #use RS232 is encountered.  Your switch statement should work.  The way the compiler works the code generated may not always be where the staement is executed.   It usually is placed near the beginning of the listing.  I will compile the code and see where it ends up. |  |  
		|  |  
		| fuzzy 
 
 
 Joined: 26 Feb 2005
 Posts: 64
 
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Mon Feb 28, 2005 2:42 pm |   |  
				| 
 |  
				| // Set baud rate based on setting // of pins B0 and B1
 
  	  | Code: |  	  | switch( input_b() & 3 ) {
 case 0 : set_uart_speed(2400);   break;
 case 1 : set_uart_speed(4800);   break;
 case 2 : set_uart_speed(9600);   break;
 case 3 : set_uart_speed(19200);  break;
 }
 
 | 
 this is a built in function of the compiler. there is an example in help.
 |  |  
		|  |  
		| Joe Porthouse 
 
 
 Joined: 21 Sep 2003
 Posts: 8
 Location: orlando, FL
 
 
			      
 
 | 
			
				|  |  
				|  Posted: Mon Feb 28, 2005 7:21 pm |   |  
				| 
 |  
				| The set_uart_speed only helps for baud rate, not parity. 
 Looking at my listing for the multiple #use RS232 shows no code generated other then the one RS232 stub at the beginning of the listing.
 
 For now I am kinda going around the built in functions and doing it brut force with:
 
  	  | Code: |  	  | struct {
 short int RX9D;
 short int OERR;
 short int FERR;
 short int ADDEN;
 short int CREN;
 short int SREN;
 short int RX9;
 short int SPEN;
 } RCSTAbits;
 #byte RCSTA     = 0xFAB
 #byte RCSTAbits = 0xFAB
 
 struct {
 short int TX9D;
 short int TRMT;
 short int BRGH;
 short int unused3;
 short int SYNC;
 short int TXEN;
 short int TX9;
 short int CSRC;
 } TXSTAbits;
 #byte TXSTA     = 0xFAC
 #byte TXSTAbits = 0xFAC
 #byte TXREG     = 0xFAD
 #byte RCREG     = 0xFAE
 #byte SPBRG     = 0xFAF
 #byte SPBRGH    = 0xFB0
 
 
 union
 {
 struct
 {
 short int serial_baud_sel:3;
 short int serial_even_parity:1;
 short int serial_odd_parity:1;
 short int serial_diag_mode:1;
 short int serial_2_stop_bits:1; // not supported, now card reader captive
 short int unused:1;
 }switches;
 char all;
 } serial_configuration;
 
 
 void configure_serial_port(void)
 {
 // stop any current transmittion
 disable_interrupts(GLOBAL);
 tx_enable = FALSE;
 
 // get current serial selection DIP settings
 serial_configuration.all = (~not_serial_config_port & 0x7F);
 
 // Reset USART registers to POR state
 TXSTA = 0;
 RCSTA = 0;
 
 // setup baud rate
 switch (serial_configuration.switches.serial_baud_sel)
 {
 // case x:
 // 16Mhz,  300 Baud
 // TXSTAbits.BRGH = 0; SPBRG = 832; // Err +0.04%
 // Invalid SPBRG
 // case x:
 // 16Mhz,  600 Baud
 // TXSTAbits.BRGH = 0; SPBRG = 416; // Err -0.08%
 // Invalid SPBRG
 case 0:
 // 16Mhz, 1200 Baud
 TXSTAbits.BRGH = 0; SPBRG = 207; // Err +0.16%
 break;
 case 1:
 // 16Mhz, 2400 Baud
 TXSTAbits.BRGH = 0; SPBRG = 103; // Err +0.16%
 break;
 case 2:
 // 16Mhz, 4800 Baud
 TXSTAbits.BRGH = 1; SPBRG = 207; // Err +0.16%
 break;
 case 3:
 // 16Mhz, 9600 Baud
 TXSTAbits.BRGH = 1; SPBRG = 103; // Err +0.16%
 break;
 case 4:
 // 16Mhz, 19200 Baud
 TXSTAbits.BRGH = 1; SPBRG =  51; // Err +0.16%
 break;
 case 5:
 // 16Mhz, 38400 Baud
 TXSTAbits.BRGH = 1; SPBRG =  25; // Err +0.16%
 break;
 case 6:
 // 16Mhz, 57600 Baud
 TXSTAbits.BRGH = 1; SPBRG =  16; // Err +2.12%
 break;
 case 7:
 // 16Mhz, 115200 Baud
 TXSTAbits.BRGH = 1; SPBRG =   8; // Err -3.55%
 break;
 default:
 // 16Mhz, 9600 Baud
 TXSTAbits.BRGH = 0; SPBRG =  25; // Err +0.16%
 break;
 }
 
 // Asynchronous Operation
 TXSTAbits.SYNC = 0;
 
 // Continuous or single reception
 RCSTAbits.CREN = 1;
 
 // Setup 8 or 9 bits
 if( serial_configuration.switches.serial_odd_parity &&
 !serial_configuration.switches.serial_even_parity)
 {
 TXSTAbits.TX9 = 1;
 RCSTAbits.RX9 = 1;
 }
 else if( serial_configuration.switches.serial_even_parity &&
 !serial_configuration.switches.serial_odd_parity)
 {
 TXSTAbits.TX9 = 1;
 RCSTAbits.RX9 = 1;
 }
 else // no parity
 {
 TXSTAbits.TX9 = 0;
 RCSTAbits.RX9 = 0;
 }
 
 // disable (for now) transmit interrupts as low priority
 disable_interrupts(INT_TBE);
 //  IPR1bits.TXIP = 0;
 //  PIR1bits.TXIF = 0;
 //  PIE1bits.TXIE = 0;
 
 // enable receive interrupts as low priority
 enable_interrupts(INT_RDA);
 //  IPR1bits.RCIP = 0;
 //  PIR1bits.RCIF = 0;
 //  PIE1bits.RCIE = 1;
 
 // enable transmitter and receiver
 TXSTAbits.TXEN = 1;
 RCSTAbits.SPEN = 1;
 
 // enable interrupts
 enable_interrupts(GLOBAL);
 }
 
 | 
 
 I know, it's ugly, but it seems to work until the PCH compiler has some clear way to do it.
 
 I still have to manually check the parity in the 9th bit in my receive ISR, and set the 9th bit in my send ISR.
 
 I'm still interested in an easier way to do it if anyone finds a way.
 |  |  
		|  |  
		| ljbeng 
 
 
 Joined: 10 Feb 2004
 Posts: 205
 
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Fri Apr 22, 2005 3:49 pm |   |  
				| 
 |  
				| I am having the same problem.... 
 I need to switch from 300,o,8,1 to 9600,n,8,1 and the compiler only sees the 9600 baud #use statement....
 
 Did anyone see this or figure it out?
 |  |  
		|  |  
		| Joe Porthouse 
 
 
 Joined: 21 Sep 2003
 Posts: 8
 Location: orlando, FL
 
 
			      
 
 | 
			
				|  |  
				|  Posted: Fri Apr 22, 2005 6:54 pm |   |  
				| 
 |  
				| Are you using the Hardware or Software UART? 
 set_uart_speed(9600); and set_uart_speed(300); would change your baud rates, but would not change your parity.
 
 The #use RS232 statements actually do not generate code, but keep the parameters specified to be used in any serial commands that follow, until the next #use RS232 is encountered.  Keep in mind that application of the #use RS232 is at compile time, NOT run time.  That is why the above listed case statement does not work and only the last #use RS232 parameters were applied.
 
 You could try,
 
  	  | Code: |  	  | #use RS232 (...300. O)
 void send_300(char c) {...};
 
 #use RS232 (...9600.. N)
 void send_9600(char c){...};
 
 ....
 if(x)
 send_300(c);
 else
 send_9600(c);
 
 | 
 
 You also may need to do the same with the kbhit() function to place the hardware uart into the correct baud/parity while waiting for a char.
 
 My brut force method listed above has been working nicely for me.  I have developed the parity bit sending/checking since then.  I have an interrupt routine that puts received chars into a queue that the application loop can pick up.  The send is also interrupt driven where the application loop fills a buffer, then enables the send interrupt.  The send interrupt is automatically disabled when the last character is sent.  The interrupt also disables my RS485 transmitter at the end of the string.  If anyone is interested I can post the complete routines.
 |  |  
		|  |  
		| PCM programmer 
 
 
 Joined: 06 Sep 2003
 Posts: 21708
 
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Fri Apr 22, 2005 7:22 pm |   |  
				| 
 |  
				| Sure.  Post them in the Code Library.   They would be a useful addition. |  |  
		|  |  
		| ljbeng 
 
 
 Joined: 10 Feb 2004
 Posts: 205
 
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Mon Apr 25, 2005 2:46 pm |   |  
				| 
 |  
				| I want to Receive at 300,o,8,1.  I want to transmit at 9600,n,8,1.  There will very little transmit and a lot of receive.  The chip is 18F452 and I want to use the hardware UART. 
 I am trying this but I don't think it worky...
 
 I still receive all data ok but it is not transmitting anything.
 
 
  	  | Code: |  	  | //abbreviated code....
 #use rs232(baud=300,parity=O,xmit=PIN_C6,rcv=PIN_C7,bits=8,errors)
 
 #int_RDA
 RDA_isr(){
 //receive 300 odd parity here
 
 }
 
 main(){
 
 //do stuff....
 
 //if I need to transmit data....
 if (txresp){
 disable_interrupts(INT_RDA);
 set_uart_speed(9600);
 #asm
 BCF    0xFAC.6  // clears the 9 bit transmit enable bit
 #endasm
 txresp = 0;
 putc('@');
 putc('h');
 putc('i');
 set_uart_speed(300);
 enable_interrupts(INT_RDA);
 }
 
 
 }
 
 | 
 |  |  
		|  |  
		| Neutone 
 
 
 Joined: 08 Sep 2003
 Posts: 839
 Location: Houston
 
 
			    
 
 | 
			
				| Re: Changing baud rates, parity, etc. at run time? |  
				|  Posted: Tue Apr 26, 2005 4:05 pm |   |  
				| 
 |  
				|  	  | Joe Porthouse wrote: |  	  | #use rs232(baud= 38400,xmit=PIN_C6,rcv=PIN_C7, errors) 
 This allows a single set of communication parameters to be configured. My problem is that our project uses dip switches to select the communication parameters to be used.
 
 Does PCH has some method to support this?
 
 In the meantime I am attempting to write directly to the PIC UART registers to modify the configuration.  I believe I may have a problem when modifying the parity or data size without the PCH routine's knowledge.
 
 Any ideas?
 | 
 
 You can have 3 #use statements that use the same serial port each with a different stream name and different parity settings. For this to work you need to replace putc with a switch that selects from 3 different putc statements, each have a different stream name. This method works for changing parity on the fly. In order to change baud rate on the fly you still have to use the set_uart_speed function.
 
 In your case you can use 2 different stream names and call set_uart_speed between transmitting and receiving.
 |  |  
		|  |  
		| henriquesv 
 
 
 Joined: 20 Feb 2011
 Posts: 8
 
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Sun Feb 20, 2011 9:16 am |   |  
				| 
 |  
				| Hello Joe, 
 Would you mind telling me a bit more about your code? Did  you manage to do this in some other way?
 
 Still lacking:
 tx_enable
 and
 not_serial_config_port
 
 Where do they come from?
 
 Thank you.
 
 Regards
 
 Henrique
 
 ps: I am using a simple PIC16f628a
 |  |  
		|  |  
		| JeffCondit 
 
 
 Joined: 29 Jul 2011
 Posts: 2
 Location: San Diega
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Fri Jul 29, 2011 2:06 pm |   |  
				| 
 |  
				| OK, still having difficulties.  We are using a PIC with 2 internal hardware EUSARTS, and the baudrates of each of the two must be different.  They must be settable by software execution and not require a reboot. 
 I've tried set_uart_speed() method before each putc() in the ISR to switch the speed at that time.  I've tried the #use rs232(baud= 1200,xmit=PIN_C6,rcv=PIN_C7,parity=N,bits=8,errors) commands before each EUSART as well.  I've tried defining a number of #use lines with stream names and referencing them in the putc() calls.
 
 The problem is that either the command affects both EUSARTs (even though referencing a specific one), or that the last #use is in effect regardless of the stream used.
 
 Can anyone give a definitive answer for what exactly is going on with these particular pre-compile directives, just how they couple to streams, EUSARTs, Bauds, etc.?  In particular, if the pre-processor goes line-by-line through the C code taking on values as it goes without generating code, I could understand why when you reach various putc() the last one encountered by a line-by-line walk through the code file by line number would be used to influence the particular putc().  But what about streams?  Is there some way the streams concept tells the pre-processor to remember various sets of parameters and associate them with a stream name?  If so, what exactly happens when the putc() to the stream is encountered by the compiler?  What does it do differently to the generated code such that when the code is run in all its nonlinear branching, that the correct baud rate or other parameters will be used?
 
 Does anyone have a sample of dual hardware EUSART code with baudrate changes controlled by software execution?
 _________________
 Jeff Condit, BSEE+38, CID
 |  |  
		|  |  
		| JeffCondit 
 
 
 Joined: 29 Jul 2011
 Posts: 2
 Location: San Diega
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Fri Jul 29, 2011 2:09 pm |   |  
				| 
 |  
				| I might have to resort to Joe Porthouse's control by direct register manipulation if the #use stuff can't be made to work.  The direct register manipulation method is bulkier but gets around all the problems with the #use and #set_baud_rate ambiguities. _________________
 Jeff Condit, BSEE+38, CID
 |  |  
		|  |  
		| PCM programmer 
 
 
 Joined: 06 Sep 2003
 Posts: 21708
 
 
 
			    
 
 | 
			
				|  |  
				|  Posted: Fri Jul 29, 2011 6:31 pm |   |  
				| 
 |  
				|  	  | Quote: |  	  | I could understand why when you reach various putc() the last one
 encountered by a line-by-line walk through the code file by line number
 would be used to influence the particular putc().
 
 | 
 Yes, that's correct.  Read the #use rs232() section in the CCS compiler
 manual.  It says:
 
  	  | Quote: |  	  | #use rs232()
 This directive tells the compiler the baud rate and pins used for serial
 I/O. This directive takes effect until another RS232 directive is
 encountered.
 
 | 
 
 
 
  	  | Quote: |  	  | But what about streams? Is there some way the streams concept tells the pre-processor to remember various sets of parameters and associate
 them with a stream name? If so, what exactly happens when the putc()
 to the stream is encountered by the compiler?
 | 
 It generates code according to the parameters defined in the #use rs232
 statement for that particular stream.
 
 
  	  | Quote: |  	  | We are using a PIC with 2 internal hardware EUSARTS. Does anyone
 have a sample of dual hardware EUSART code with baudrate changes
 controlled by software execution?
 
 | 
 Post your PIC and your CCS compiler version.
 |  |  
		|  |  
		|  |  
  
	| 
 
 | You cannot post new topics in this forum You cannot reply to topics in this forum
 You cannot edit your posts in this forum
 You cannot delete your posts in this forum
 You cannot vote in polls in this forum
 
 |  
 Powered by phpBB © 2001, 2005 phpBB Group
 
 |