Good afternoon all,
I am trying to write code that will produce 10 Mbit/s, Manchester-encoded data. Manchester-encoded means a '1' looks like '01' and a '0' will look like '10'. 10 Mbit/s means a bit will last 100 ns, so a half-bit will last for 50 ns; i.e. 5 clock cycles. The good news is that the mbed controller seems to be quite capable of this, and I have code that works: almost.
My first attempt was using FastOut (found here on mbed.org), and looks like this:
template <PinName pin> class FastOut
{
// pin = LPC_GPIO0_BASE + port * 32 + bit
// port = (pin - LPC_GPIO0_BASE) / 32
// bit = (pin - LPC_GPIO0_BASE) % 32
// helper function to calculate the GPIO port definition for the pin
// we rely on the fact that port structs are 0x20 bytes apart
inline LPC_GPIO_TypeDef* portdef() { return (LPC_GPIO_TypeDef*)(LPC_GPIO_BASE + ((pin - P0_0)/32)*0x20); };
// helper function to calculate the mask for the pin's bit in the port
inline uint32_t mask() { return 1UL<<((pin - LPC_GPIO0_BASE) % 32); };
public:
FastOut()
{
// set FIODIR bit to 1 (output)
portdef()->FIODIR |= mask();
}
void write(int value)
{
if ( value )
portdef()->FIOSET = mask();
else
portdef()->FIOCLR = mask();
}
int read()
{
return (portdef()->FIOPIN) & mask() != 0;
}
FastOut& operator= (int value) { write(value); return *this; };
FastOut& operator= (FastOut& rhs) { return write(rhs.read()); };
operator int() { return read(); };
};
FastOut<p5> txd;
void snd(uint32_t cmd)
{
uint32_t invcmd;
invcmd=~cmd;
__disable_irq();
txd=0; // start marker
txd=1; // start marker
txd=0; // start marker
txd=(invcmd&1);
__nop();
txd=(cmd&1);
__nop();
txd=0; // end marker
txd=1; // end marker
txd=0; // end marker
__enable_irq();
}
This is not fully functional code, but fortunately, it is not too complex, and you need a high-speed probe & scope in any case to debug the code. The main point here is the snd routine: It sends a start marker (010) to trigger the scope, next it sends the LSB of cmd: '10' if it is a '0', and '01' if it is a '1'. Finally, it sends an end marker to verify that the total duration of the bit is correct.
The results are as follows:
For a '0':

For a '1':

Analysing what we see (expressed in clock cycles), there is :
For a 0: 010 0011111100 010
For a 1: 010 0000000111 010
The good news is that the duration of the bit is OK, (in fact the nop's are helping to stretch the bit out to 100 ns and we would be able to do it in 80 ns if we needed to), but my problem is that the 1->0 transition of the '0' is appearing 1 clock cycle after the 0->1 transition of the '1'.
Next, I tried this:
inline LPC_GPIO_TypeDef* portdef() { return (LPC_GPIO_TypeDef*)(537509888); }; // p5
void snd(uint32_t cmd)
{
uint32_t invcmd;
invcmd=~cmd;
__disable_irq();
portdef()->FIODIR |= 512; // set direction of p5 to out
portdef()->FIOCLR =512; // generate start marker bit
portdef()->FIOSET =512; // generate start marker bit
portdef()->FIOCLR =512; // generate start marker bit
if (cmd&1){
portdef()->FIOCLR=512;
portdef()->FIOSET=512;
}
else{
portdef()->FIOSET=512;
portdef()->FIOCLR=512;
}
portdef()->FIOCLR =512; // generate end marker bit
portdef()->FIOSET =512; // generate end marker bit
portdef()->FIOCLR =512; // generate end marker bit
__enable_irq();
}
And the result was:
For a '0': 010 001000 010
For a '1': 010 000001 010
Now, the problem has gotten worse: The 1->0 transition of the '0' appears two clock cycles before the 0->1 transition of the '1'. (The bit time is also no longer correct, but we can pad that out with nops later)
I suppose my questions are:
a) Is there any way to examine the assembly output of the compiler to better understand what it is doing?
b) Is there any way to change the compiler optimisation settings? For example, to prevent it from eliminating 'pointless' nop's?
c) Is there any way to simply 'write data to a pin'? I suppose that would get rid of the the data dependency of the transition time. But if (deep down inside) we always need to work with FIOSET and FIOCLR and conditional execution depending on some value, then the problem may be quite difficult to work around, short of self-modifying code.
- Does anybody have other suggestions?
I am quite new to this so if there is something obvious I have overlooked, please feel free to comment!
Kind regards,
Arnoud.
Good afternoon all,
I am trying to write code that will produce 10 Mbit/s, Manchester-encoded data. Manchester-encoded means a '1' looks like '01' and a '0' will look like '10'. 10 Mbit/s means a bit will last 100 ns, so a half-bit will last for 50 ns; i.e. 5 clock cycles. The good news is that the mbed controller seems to be quite capable of this, and I have code that works: almost.
My first attempt was using FastOut (found here on mbed.org), and looks like this:
This is not fully functional code, but fortunately, it is not too complex, and you need a high-speed probe & scope in any case to debug the code. The main point here is the snd routine: It sends a start marker (010) to trigger the scope, next it sends the LSB of cmd: '10' if it is a '0', and '01' if it is a '1'. Finally, it sends an end marker to verify that the total duration of the bit is correct.
The results are as follows:
For a '0':
For a '1':
Analysing what we see (expressed in clock cycles), there is :
For a 0: 010 0011111100 010
For a 1: 010 0000000111 010
The good news is that the duration of the bit is OK, (in fact the nop's are helping to stretch the bit out to 100 ns and we would be able to do it in 80 ns if we needed to), but my problem is that the 1->0 transition of the '0' is appearing 1 clock cycle after the 0->1 transition of the '1'.
Next, I tried this:
And the result was:
For a '0': 010 001000 010
For a '1': 010 000001 010
Now, the problem has gotten worse: The 1->0 transition of the '0' appears two clock cycles before the 0->1 transition of the '1'. (The bit time is also no longer correct, but we can pad that out with nops later)
I suppose my questions are:
a) Is there any way to examine the assembly output of the compiler to better understand what it is doing?
b) Is there any way to change the compiler optimisation settings? For example, to prevent it from eliminating 'pointless' nop's?
c) Is there any way to simply 'write data to a pin'? I suppose that would get rid of the the data dependency of the transition time. But if (deep down inside) we always need to work with FIOSET and FIOCLR and conditional execution depending on some value, then the problem may be quite difficult to work around, short of self-modifying code.
- Does anybody have other suggestions?
I am quite new to this so if there is something obvious I have overlooked, please feel free to comment!
Kind regards,
Arnoud.