An introduction to electronic weighing

Software Development

The use of IDEs can be controversial and very much a matter of taste, and it's certainly possible to do this type of microcontroller development without one. The ARM compilers and standard libraries can be downloaded from your distro's repository, and you're off, using any editor that suits you and Make or Cmake – again, your choice. Once you have a compiled binary, utilities for ST-Link allow you to program your device, and you can use GDB (the GNU debugger) to debug you program.

That said, ST's STM32CubeIDE, based on Eclipse, does streamline the process by integrating ST's CubeMX tool, a utility that allows you to configure your microcontroller and generate a software framework that does all the initialization and leaves you with a blank main() function to add to your code. You can label the pins of the microcontroller (bonus points if you use the same names as on the schematic!).

The pin configuration for this design is shown in Figure 7. Once saved, the IDE generates a set of #define directives for the I/O pins that you can use in your code and a complete set of initialization routines. At this point, you can continue to use the IDE or ignore it and use Make with the generated makefile. However, if you stay with the IDE and have your hardware connected by an ST-Link programmer, a single mouse click compiles, downloads, and runs your code. This level of preconfiguration – including, if you want, the inclusion of a real-time operating system (RTOS) such as FreeRTOS – can leave you free to concentrate on your application code. In a commercial environment, time to market is everything, and time savings like this can be invaluable.

Figure 7: The project pin configuration in STM32CubeIDE.

Serial I/O and printf

My hardware design and microcontroller configuration includes a serial port for debugging. Being able to output data and wait for keystrokes during debugging can be invaluable as an alternative to, or as an adjunct to, a debugger. The STM32 libraries make it easy to override the low-level routines that printf and scanf eventually call and redirect them to a serial port (Listing 1). Once this is done, these functions are available from whatever terminal (e.g., the Linux minicom tool) you choose to connect to the serial port. This serial port could also be used in an application for batch weighing, in which an external computer or programmable logic controller (PLC) monitors the weight signal on a continuous basis.

Listing 1

Overriding Low-Level I/O Primitives

01 /** read from the serial port */
02 int _read(int file, char *ptr, int len)
03 {
04   HAL_UART_Receive(&huart1, (uint8_t *)ptr, 1, -1);
05   return 1;
06 }
08 /** write to the serial port */
09 int _write(int file, char *ptr, int len)
10 {
11   int DataIdx;
13   for (DataIdx = 0; DataIdx < len; DataIdx++)
14   {
15       if (*ptr == '\n')
16       {
17           HAL_UART_Transmit(&huart1, (uint8_t *)"\r\n", 2, -1);
18       }
19       else
20       {
21           HAL_UART_Transmit(&huart1, (uint8_t *)ptr, 1, -1);
22       }
23     ptr++;
24   }
25   return len;
26 }
28 /** wait for a keystroke */
29 uint8_t kbhit(void)
30 {
31     return __HAL_UART_GET_FLAG(&huart1, UART_FLAG_RXNE);
32 }

ADC Driver

The ADS1232 has a very simple two-wire serial interface with no configuration registers, so all setup (clock speed, gain, etc.) happens from pins on the chip. To read the data from the chip, the software must first wait for a valid sample, which it does by waiting for the DOUT (data out) line to go low (Listing 2). At this point, data is clocked out in a serial manner by pulsing the clock pin from low to high and reading the resulting data bit from DOUT. This process is repeated 24 times to clock out a complete 24-bit sample. A further clock pulse resets DOUT to its "data ready" function.

Listing 2

Reading the ADC

01 /** wait for ADC sample ready and read it out */
02 uint32_t ADS1232_Read(void)
03 {
04   int32_t value = 0;
06   // wait for DOUT to go low : data ready
08     ;
10   for (uint8_t i = 0; i < 24; i++)
11   {
12     value = value << 1;
14     // send clock pulse
15     HAL_GPIO_WritePin(ADC_SCLK_GPIO_Port, ADC_SCLK_Pin, 1);
16     HAL_GPIO_WritePin(ADC_SCLK_GPIO_Port, ADC_SCLK_Pin, 0);
18     // read the data in
20     {
21       value |= 1;
22     }
23   }
25   // reset DOUT
26   HAL_GPIO_WritePin(ADC_SCLK_GPIO_Port, ADC_SCLK_Pin, 1);
27   HAL_GPIO_WritePin(ADC_SCLK_GPIO_Port, ADC_SCLK_Pin, 0);
29   // 24-32 bit 2s complement conversion
30   value <<= 8;
31   value /= 256;
33   return (value);
34 }

Buy this article as PDF

Express-Checkout as PDF
Price $2.95
(incl. VAT)

Buy Linux Magazine

Get it on Google Play

US / Canada

Get it on Google Play

UK / Australia

Related content

  • Programming the Cell

    The Cell architecPIture is finding its way into a vast range of computer systems – from huge supercomputers to inauspicious Playstation game consoles. We'll show you around the Cell and take a look at a sample Cell application.

  • Neural Networks

    3, 4, 8, 11… ? A neural network can complete this series without knowledge of the underlying algorithm – by a kind of virtual gut feeling. We’ll show you how neural networks solve problems by simulating the behavior of a human brain.

  • Solar-Powered Mini PC with Puppy Linux

    The Aleutia E1 is a Mini PC with Puppy Linux on board designed for deployment in areas far away from power sources.

  • WiFi Thermo-Hygrometer

    A WiFi sensor monitors indoor humidity and temperature and a Node-RED dashboard reports the results, helping you to maintain a pleasant environment.

  • Latitude 2100: Netbook Not Just for Students

    Dell takes the new 10" 2100 netbook out of its Latitude business series to target students, but it also serves as a mobile business device. Ubuntu 8.10 runs on it.

comments powered by Disqus

Direct Download

Read full article as PDF:

Price $2.95