Using the Analog in and LCD, turn the MBED into a simple Oscilloscope

Dependencies:   C12832_lcd DebounceIn mbed

/media/uploads/trichards1138/snapshot_20140609.jpg /media/uploads/trichards1138/snapshot_20140609_2.jpg

Introduction:

The MBED Oscilloscope uses the Analog Input on pin 17 to display connected waveforms from another MBED or a function generator. Note that the above left photo shows the Ain 1/8" phone jack on the side of the MBED being used as the input for a function generator. The crude photo on the right shows the waveform displayed on the MBED LCD screen (this is using the Application board available to plug the MBED processor board into. It includes the LCD screen as well as buttons, pots, temperature sensor, wifly socket, speaker, etc).

The waveform that is displayed on the MBED Oscilloscope is: /media/uploads/trichards1138/waveform.jpg This is from a Digilent combination function generator, Oscilloscope, and logic analyzer. The arbitrary waveform generator shown is one of the features of the unit and allows output of several types of signals from its analog output.

Notice that the frequency is set to 1kHz. The MBED Oscilloscope is set to 512 uS per division (due to display limitations, the settings cannot be displayed with the waveform). The displayed waveform shows a complete period in approximately 2 divisions. The scope is indicating that the period is 1mS or 1kHz.

Capabilities and Limitations:

The scope can display waveforms up to 5kHz with some clarity (although 5kHz is pushing its resolution 2-3kHz is a good practical limit). For displaying waveforms from a function generator running on another MBED, this is not a significant limitation. Except for the square-wave function, the function generator output of the MBED is around 1kHz realistically (the square-wave can output much higher frequencies since it only has two levels). (NOTE: this limitation is not strictly due to the ADC input. it also has to do with loop delay i.e. display update times...etc)

When an input is configured as an Analog input to the ADC, it is not 5V tolerant. The voltage on the pin must not exceed the reference-voltage-plus (set to approx 3V on the MBED board). So the amplitude input is limited to 3V (Although I've input signals at 3.3V with no immediate issues). Again this is not a significant limitation if debugging signals from another MBED. However, if you are using an external function generator, you must ensure you do not overdrive the input.

Usage:

Using the scope is fairly straight forward. Once you have the waveform source connected to pin 17 (either via the 1/8" phone jack or a wire directly connected to p17 through the header), the waveform should be displayed.

To change the time-base, use the right and left joystick buttons. You can check the current setting by pressing the middle button of the joystick. As long as you keep the middle button pressed, the settings page will be displayed. It shows the current settings of the time-base, vertical or volt-base, and the trigger mode. The photo below shows the settings page: /media/uploads/trichards1138/snapshot_20140609_8.jpg

Again, I apologize for the crudeness of the photo (a photographer I am not). Note that the time-base will change based on whether the grid is turned on. In the first photo above you see the grid lines painted on the display. You have the option to turn those off and there are instances where it is desirable. When this is done, the entire horizontal screen is viewed as one division. So the time-base will change accordingly.

Once you release the center button, the display will return to the waveform screen after about a second. If however you press the center button again within that window, you will be taken to the trigger selection window: /media/uploads/trichards1138/snapshot_20140609_9.jpg

There are five trigger modes:

1. Rising-Edge Main Signal (this finds the rising edge on the signal connected to pin 17)

2. Falling-Edge Main Signal (Same as above but finds the falling edge)

3. Rising-Edge Ext Signal (this finds the rising edge on the signal connected to pin 18)

4. Falling-Edge Ext Signal (Sam as in 3 for falling edge)

5. Auto or No Trigger (this does not look for a trigger - useful for dc signals)

Note that only three modes are displayed on the screen. The screen will scroll as you move the selection down. This is done with the "up" and "down" buttons on the joystick. Make your selection by pressing the center button again.

Once you make your trigger selection, you will then be taken to the grid selection menu: /media/uploads/trichards1138/snapshot_20140609_10.jpg

This allows the choice of whether to display the grid. Again there are situations you may want to disable the grid. Make the grid selection in the same way by pressing the center button of the joystick.

Once this selection is made, you will return the the waveform display.

The only additional control it connected to pot2. This is the pot furthest from the display. It controls the trigger level.

Know Issues:

1. The External triggering is not working at all right now. For some reason having two active signals connected to two different ADC inputs causes crosstalk that distorts both signals. I haven't looked into this yet so avoid using external triggering for the present. I will update when and if it is fixed. (any input or pointers would be appreciated)

2. The time base needs to be calibrated a little better. It is pretty close in most cases and good enough for rough work, but you would not want to keep time with it. It is a little like whack-a-mole and may never be perfect, but it can be improved.

3. The timeouts for the triggering mechanism may need some tweaking at some point as well. I think it is pretty good but may cause some fringe problems.

4. The trigger level is not currently displayed anywhere and should be part of the settings page. This would need to be added to a second settings page (and the menu system might get cumbersome at some point).

5. The pot is a little jittery for the trigger level. However, adding another menu to set it with the joystick will further push the problem in 4 of the menu system getting cumbersome. It may be good enough with some tweaks.

Future enhancements:

1. Getting the external triggering working is the first order of business.

2. Getting the time-base closer to reality will make things nicer.

3. I believe the grid would be nicer if the lines were dotted.

4. A classmate suggested adding FFT capabilities for a limited spectrum analyzer.

Any other suggestions?

Accessories:

I put together some faux scope probes by purchasing an inexpensive 3-foot audio cable (it is not shielded as the frequencies we are dealing with are fairly low). I cut this in half and soldered on some clips to connect the center and ground connections for the Ain 1/8" phone jack. The cable I got was actually stereo and I used a DMM to find which bare-wire was connected to pin 17 on the MBED. This accessory does make it nice in most cases.

However you can achieve the same results by connecting two wires directly one to pin 17 and the other to GND on the header connections of the application board directly to the input signal and GND of your waveform output.

Committer:
trichards1138
Date:
Sun Jun 01 06:19:45 2014 +0000
Revision:
1:4080ce288aea
Parent:
0:9ccc4b6355a8
Child:
2:71767eb23ac2
Fixed the time-base, volt-base, and the trigger.  Still need a user interface to select a trigger mode.

Who changed what in which revision?

UserRevisionLine numberNew contents of line
trichards1138 0:9ccc4b6355a8 1 #include "mbed.h"
trichards1138 0:9ccc4b6355a8 2 #include "C12832_lcd.h"
trichards1138 1:4080ce288aea 3 #include "DebounceIn.h"
trichards1138 0:9ccc4b6355a8 4
trichards1138 0:9ccc4b6355a8 5 Serial pc(USBTX, USBRX);
trichards1138 0:9ccc4b6355a8 6 C12832_LCD LCD;
trichards1138 0:9ccc4b6355a8 7 AnalogIn Ain(p17);
trichards1138 1:4080ce288aea 8 AnalogIn Aintrig(p18);
trichards1138 1:4080ce288aea 9
trichards1138 1:4080ce288aea 10 //AnalogIn pot1(p19);
trichards1138 0:9ccc4b6355a8 11 AnalogIn pot2(p20);
trichards1138 1:4080ce288aea 12
trichards1138 1:4080ce288aea 13 DigitalOut led1(LED1);
trichards1138 1:4080ce288aea 14 DigitalOut led2(LED2);
trichards1138 1:4080ce288aea 15
trichards1138 1:4080ce288aea 16 DebounceIn Update(p14);
trichards1138 1:4080ce288aea 17 DebounceIn Up(p15);
trichards1138 1:4080ce288aea 18 DebounceIn Down(p12);
trichards1138 1:4080ce288aea 19 DebounceIn Left(p13);
trichards1138 1:4080ce288aea 20 DebounceIn Right(p16);
trichards1138 1:4080ce288aea 21
trichards1138 1:4080ce288aea 22 unsigned int time_base = 0.1;
trichards1138 1:4080ce288aea 23 float volt_base = 3.0;
trichards1138 1:4080ce288aea 24 float trigger = 0.5, temptrig = 0.5;
trichards1138 1:4080ce288aea 25 bool rising_edge_triggered = true;
trichards1138 1:4080ce288aea 26 bool falling_edge_triggered = false;
trichards1138 1:4080ce288aea 27 bool ext_trigger_rising = false;
trichards1138 1:4080ce288aea 28 bool ext_trigger_falling = false;
trichards1138 0:9ccc4b6355a8 29
trichards1138 1:4080ce288aea 30 #define VBASEMAX 13
trichards1138 1:4080ce288aea 31 #define TBASEMAX 9
trichards1138 1:4080ce288aea 32 void Update_params(void)
trichards1138 1:4080ce288aea 33 { //divisions 1=128uS, 5=640uS, 10=1280uS, 20=2560uS, 50=6.4mS, 100=12.8mS, 500=64mS, 1000=128mS
trichards1138 1:4080ce288aea 34 unsigned int next_tbase[] = { 1, 5, 10, 20, 50, 100, 500, 700, 1000 };
trichards1138 1:4080ce288aea 35 float next_vbase[] = { 1.0, 1.4, 1.8, 2.0, 2.4, 2.8, 3.0, 3.5, 4.0, 4.5, 5.0, 5.5, 6.0 };
trichards1138 1:4080ce288aea 36 static int vsel=0, tsel=1;
trichards1138 1:4080ce288aea 37 if( Up ) {
trichards1138 1:4080ce288aea 38 while( Up )
trichards1138 1:4080ce288aea 39 wait_ms(1);
trichards1138 1:4080ce288aea 40 //led1 = !led1;
trichards1138 1:4080ce288aea 41 volt_base = next_vbase[vsel++];
trichards1138 1:4080ce288aea 42 if( vsel == VBASEMAX )
trichards1138 1:4080ce288aea 43 vsel = VBASEMAX-1;
trichards1138 1:4080ce288aea 44 }
trichards1138 1:4080ce288aea 45 if( Down ) {
trichards1138 1:4080ce288aea 46 while( Down )
trichards1138 1:4080ce288aea 47 wait_ms(1);
trichards1138 1:4080ce288aea 48 //led2 = !led2;
trichards1138 1:4080ce288aea 49 volt_base = next_vbase[vsel--];
trichards1138 1:4080ce288aea 50 if( vsel == -1 )
trichards1138 1:4080ce288aea 51 vsel = 0;
trichards1138 1:4080ce288aea 52 }
trichards1138 1:4080ce288aea 53 if( Right ) {
trichards1138 1:4080ce288aea 54 while( Right )
trichards1138 1:4080ce288aea 55 wait_ms(1);
trichards1138 1:4080ce288aea 56 //led3 = !led3;
trichards1138 1:4080ce288aea 57 time_base = next_tbase[tsel++];
trichards1138 1:4080ce288aea 58 if( tsel == TBASEMAX )
trichards1138 1:4080ce288aea 59 tsel = TBASEMAX-1;
trichards1138 1:4080ce288aea 60 }
trichards1138 1:4080ce288aea 61 if( Left ) {
trichards1138 1:4080ce288aea 62 while( Left )
trichards1138 1:4080ce288aea 63 wait_ms(1);
trichards1138 1:4080ce288aea 64 //led4 = !led4;
trichards1138 1:4080ce288aea 65 time_base = next_tbase[tsel--];
trichards1138 1:4080ce288aea 66 if( tsel == -1 )
trichards1138 1:4080ce288aea 67 tsel = 0;
trichards1138 1:4080ce288aea 68 }
trichards1138 0:9ccc4b6355a8 69
trichards1138 1:4080ce288aea 70 temptrig = pot2.read() * volt_base;
trichards1138 1:4080ce288aea 71 if( (temptrig > (trigger+0.10)) || (temptrig < (trigger-0.10)) )
trichards1138 1:4080ce288aea 72 trigger = temptrig;
trichards1138 0:9ccc4b6355a8 73 }
trichards1138 0:9ccc4b6355a8 74
trichards1138 1:4080ce288aea 75 void get_trigger(void)
trichards1138 0:9ccc4b6355a8 76 {
trichards1138 1:4080ce288aea 77 if( rising_edge_triggered ) {
trichards1138 1:4080ce288aea 78 if( (Ain.read() * volt_base) > trigger )
trichards1138 1:4080ce288aea 79 while( (Ain.read() * volt_base) > trigger )
trichards1138 1:4080ce288aea 80 wait_us(1);
trichards1138 1:4080ce288aea 81 while( (Ain.read() * volt_base) < trigger )
trichards1138 1:4080ce288aea 82 wait_us(1);
trichards1138 1:4080ce288aea 83 } else if( falling_edge_triggered ) {
trichards1138 1:4080ce288aea 84 if( (Ain.read() * volt_base) < trigger )
trichards1138 1:4080ce288aea 85 while( (Ain.read() * volt_base) < trigger )
trichards1138 1:4080ce288aea 86 wait_us(1);
trichards1138 1:4080ce288aea 87 while( (Ain.read() * volt_base) > trigger )
trichards1138 1:4080ce288aea 88 wait_us(1);
trichards1138 1:4080ce288aea 89 } else if( ext_trigger_rising ) {
trichards1138 1:4080ce288aea 90 if( (Aintrig.read() * volt_base) > trigger )
trichards1138 1:4080ce288aea 91 while( (Aintrig.read() * volt_base) > trigger )
trichards1138 1:4080ce288aea 92 wait_us(1);
trichards1138 1:4080ce288aea 93 while( (Aintrig.read() * volt_base) < trigger )
trichards1138 1:4080ce288aea 94 wait_us(1);
trichards1138 1:4080ce288aea 95 } else if( ext_trigger_falling ) {
trichards1138 1:4080ce288aea 96 if( (Aintrig.read() * volt_base) < trigger )
trichards1138 1:4080ce288aea 97 while( (Aintrig.read() * volt_base) < trigger )
trichards1138 1:4080ce288aea 98 wait_us(1);
trichards1138 1:4080ce288aea 99 while( (Aintrig.read() * volt_base) > trigger )
trichards1138 1:4080ce288aea 100 wait_us(1);
trichards1138 1:4080ce288aea 101 }
trichards1138 0:9ccc4b6355a8 102 }
trichards1138 0:9ccc4b6355a8 103
trichards1138 1:4080ce288aea 104 int main()
trichards1138 0:9ccc4b6355a8 105 {
trichards1138 1:4080ce288aea 106 pc.baud(115200);
trichards1138 0:9ccc4b6355a8 107 unsigned int i;
trichards1138 0:9ccc4b6355a8 108 unsigned int h = LCD.height()-2, w = LCD.width(), hhh;
trichards1138 0:9ccc4b6355a8 109 while(true) { // thread loop
trichards1138 0:9ccc4b6355a8 110 for (i=0; i<w; i++) {
trichards1138 1:4080ce288aea 111 hhh = (int)((h+1)-(h*volt_base*Ain.read()));
trichards1138 0:9ccc4b6355a8 112 LCD.pixel(i, hhh ,1); // print pixel
trichards1138 1:4080ce288aea 113 wait_us(time_base);
trichards1138 0:9ccc4b6355a8 114 }
trichards1138 1:4080ce288aea 115 LCD.copy_to_lcd(); // LCD.pixel does not update the lcd
trichards1138 1:4080ce288aea 116 Update_params();
trichards1138 1:4080ce288aea 117 wait_ms(50);
trichards1138 1:4080ce288aea 118 get_trigger();
trichards1138 0:9ccc4b6355a8 119 LCD.cls();
trichards1138 0:9ccc4b6355a8 120 }
trichards1138 0:9ccc4b6355a8 121 }
trichards1138 0:9ccc4b6355a8 122