Device tree error undefined reference to `__device_dts_ord_127'

Hi there,

i have built a custom board in devicetree. I have based it off the nrf52840DK dts but added in a magnetometer:

s3_mini_v1.dts

/*
 * Copyright (c) 2017 Linaro Limited
 *
 * SPDX-License-Identifier: Apache-2.0
 */

/dts-v1/;
#include <nordic/nrf52840_qiaa.dtsi>
#include "s3_mini_v1-pinctrl.dtsi"

 / {
	 model = "Nordic nRF52840 DK NRF52840";
	 compatible = "nordic,nrf52840-dk-nrf52840";
 
	 chosen {
		 zephyr,console = &uart0;
		 zephyr,shell-uart = &uart0;
		 zephyr,uart-mcumgr = &uart0;
		 zephyr,bt-mon-uart = &uart0;
		 zephyr,bt-c2h-uart = &uart0;
		 zephyr,sram = &sram0;
		 zephyr,flash = &flash0;
		 zephyr,code-partition = &slot0_partition;
		 zephyr,ieee802154 = &ieee802154;
	 };
 
	 leds {
		 compatible = "gpio-leds";
		 led0: led_0 {
			 gpios = <&gpio0 13 GPIO_ACTIVE_LOW>;
			 label = "Green LED 0";
		 };
		 led1: led_1 {
			 gpios = <&gpio0 14 GPIO_ACTIVE_LOW>;
			 label = "Green LED 1";
		 };
		 led2: led_2 {
			 gpios = <&gpio0 15 GPIO_ACTIVE_LOW>;
			 label = "Green LED 2";
		 };
		 led3: led_3 {
			 gpios = <&gpio0 16 GPIO_ACTIVE_LOW>;
			 label = "Green LED 3";
		 };
	 };
 
	 pwmleds {
		 compatible = "pwm-leds";
		 pwm_led0: pwm_led_0 {
			 pwms = <&pwm0 0 PWM_MSEC(20) PWM_POLARITY_INVERTED>;
		 };
	 };
 
	 buttons {
		 compatible = "gpio-keys";
		 button0: button_0 {
			 gpios = <&gpio0 11 (GPIO_PULL_UP | GPIO_ACTIVE_LOW)>;
			 label = "Push button switch 0";
		 };
		 button1: button_1 {
			 gpios = <&gpio0 12 (GPIO_PULL_UP | GPIO_ACTIVE_LOW)>;
			 label = "Push button switch 1";
		 };
		 button2: button_2 {
			 gpios = <&gpio0 24 (GPIO_PULL_UP | GPIO_ACTIVE_LOW)>;
			 label = "Push button switch 2";
		 };
		 button3: button_3 {
			 gpios = <&gpio0 25 (GPIO_PULL_UP | GPIO_ACTIVE_LOW)>;
			 label = "Push button switch 3";
		 };
	 };
 
	//  arduino_header: connector {
	// 	 compatible = "arduino-header-r3";
	// 	 #gpio-cells = <2>;
	// 	 gpio-map-mask = <0xffffffff 0xffffffc0>;
	// 	 gpio-map-pass-thru = <0 0x3f>;
	// 	 gpio-map = <0 0 &gpio0 3 0>,	/* A0 */
	// 			<1 0 &gpio0 4 0>,	/* A1 */
	// 			<2 0 &gpio0 28 0>,	/* A2 */
	// 			<3 0 &gpio0 29 0>,	/* A3 */
	// 			<4 0 &gpio0 30 0>,	/* A4 */
	// 			<5 0 &gpio0 31 0>,	/* A5 */
	// 			<6 0 &gpio1 1 0>,	/* D0 */
	// 			<7 0 &gpio1 2 0>,	/* D1 */
	// 			<8 0 &gpio1 3 0>,	/* D2 */
	// 			<9 0 &gpio1 4 0>,	/* D3 */
	// 			<10 0 &gpio1 5 0>,	/* D4 */
	// 			<11 0 &gpio1 6 0>,	/* D5 */
	// 			<12 0 &gpio1 7 0>,	/* D6 */
	// 			<13 0 &gpio1 8 0>,	/* D7 */
	// 			<14 0 &gpio1 10 0>,	/* D8 */
	// 			<15 0 &gpio1 11 0>,	/* D9 */
	// 			<16 0 &gpio1 12 0>,	/* D10 */
	// 			<17 0 &gpio1 13 0>,	/* D11 */
	// 			<18 0 &gpio1 14 0>,	/* D12 */
	// 			<19 0 &gpio1 15 0>,	/* D13 */
	// 			<20 0 &gpio0 26 0>,	/* D14 */
	// 			<21 0 &gpio0 27 0>;	/* D15 */
	//  };
 
	 arduino_adc: analog-connector {
		 compatible = "arduino,uno-adc";
		 #io-channel-cells = <1>;
		 io-channel-map = <0 &adc 1>,	/* A0 = P0.3 = AIN1 */
				  <1 &adc 2>,	/* A1 = P0.4 = AIN2 */
				  <2 &adc 4>,	/* A2 = P0.28 = AIN4 */
				  <3 &adc 5>,	/* A3 = P0.29 = AIN5 */
				  <4 &adc 6>,	/* A4 = P0.30 = AIN6 */
				  <5 &adc 7>;	/* A5 = P0.31 = AIN7 */
	 };
 
	 /* These aliases are provided for compatibility with samples */
	 aliases {
		 led0 = &led0;
		 led1 = &led1;
		 led2 = &led2;
		 led3 = &led3;
		 pwm-led0 = &pwm_led0;
		 sw0 = &button0;
		 sw1 = &button1;
		 sw2 = &button2;
		 sw3 = &button3;
		 bootloader-led0 = &led0;
		 mcuboot-button0 = &button0;
		 mcuboot-led0 = &led0;
		 watchdog0 = &wdt0;
		 //spi-flash0 = &mx25r64;
		 magnetometer = &lsm303agr;
	 };
 };
 
 &adc {
	 status = "okay";
 };
 
 &gpiote {
	 status = "okay";
 };
 
 &gpio0 {
	 status = "okay";
 };
 
 &gpio1 {
	 status = "okay";
 };
 
 &uart0 {
	 compatible = "nordic,nrf-uarte";
	 status = "okay";
	 current-speed = <115200>;
	 pinctrl-0 = <&uart0_default>;
	 pinctrl-1 = <&uart0_sleep>;
	 pinctrl-names = "default", "sleep";
 };
 
 arduino_serial: &uart1 {
	 current-speed = <115200>;
	 pinctrl-0 = <&uart1_default>;
	 pinctrl-1 = <&uart1_sleep>;
	 pinctrl-names = "default", "sleep";
 };
 
 &i2c0 {
	compatible = "nordic,nrf-twi";
	status = "okay";
	pinctrl-0 = <&i2c0_default>;
	pinctrl-1 = <&i2c0_sleep>;	
	pinctrl-names = "default", "sleep";	
	label = "I2C0";
    lsm303agr: gpio@19 {
		compatible = "rohm,bh1749";
		int-gpios = <&gpio0 1 (GPIO_PULL_UP | GPIO_ACTIVE_LOW)>;
        reg = <0x19>;
		status = "okay";
		label = "lsm303";
		//interrupts = <1 (GPIO_PULL_UP | GPIO_ACTIVE_LOW)>;
    };
};
 
 &i2c1 {
	 compatible = "nordic,nrf-twi";
	 /* Cannot be used together with spi1. */
	 /* status = "okay"; */
	 pinctrl-0 = <&i2c1_default>;
	 pinctrl-1 = <&i2c1_sleep>;
	 pinctrl-names = "default", "sleep";
 };
 
 &pwm0 {
	 status = "okay";
	 pinctrl-0 = <&pwm0_default>;
	 pinctrl-1 = <&pwm0_sleep>;
	 pinctrl-names = "default", "sleep";
 };
 
//  &spi0 {
// 	 compatible = "nordic,nrf-spi";
// 	 /* Cannot be used together with i2c0. */
// 	 /* status = "okay"; */
// 	 pinctrl-0 = <&spi0_default>;
// 	 pinctrl-1 = <&spi0_sleep>;
// 	 pinctrl-names = "default", "sleep";
//  };
 
//  &spi1 {
// 	 compatible = "nordic,nrf-spi";
// 	 status = "okay";
// 	 pinctrl-0 = <&spi1_default>;
// 	 pinctrl-1 = <&spi1_sleep>;
// 	 pinctrl-names = "default", "sleep";
//  };
 
//  &spi2 {
// 	 compatible = "nordic,nrf-spi";
// 	 status = "disabled";
// 	 pinctrl-0 = <&spi2_default>;
// 	 pinctrl-1 = <&spi2_sleep>;
// 	 pinctrl-names = "default", "sleep";
//  };
 
//  &qspi {
// 	 status = "okay";
// 	 pinctrl-0 = <&qspi_default>;
// 	 pinctrl-1 = <&qspi_sleep>;
// 	 pinctrl-names = "default", "sleep";
// 	 mx25r64: mx25r6435f@0 {
// 		 compatible = "nordic,qspi-nor";
// 		 reg = <0>;
// 		 /* MX25R64 supports only pp and pp4io */
// 		 writeoc = "pp4io";
// 		 /* MX25R64 supports all readoc options */
// 		 readoc = "read4io";
// 		 sck-frequency = <8000000>;
// 		 jedec-id = [c2 28 17];
// 		 sfdp-bfp = [
// 			 e5 20 f1 ff  ff ff ff 03  44 eb 08 6b  08 3b 04 bb
// 			 ee ff ff ff  ff ff 00 ff  ff ff 00 ff  0c 20 0f 52
// 			 10 d8 00 ff  23 72 f5 00  82 ed 04 cc  44 83 68 44
// 			 30 b0 30 b0  f7 c4 d5 5c  00 be 29 ff  f0 d0 ff ff
// 		 ];
// 		 size = <67108864>;
// 		 has-dpd;
// 		 t-enter-dpd = <10000>;
// 		 t-exit-dpd = <35000>;
// 	 };
//  };
 
//  arduino_spi: &spi3 {
// 	 status = "okay";
// 	// cs-gpios = <&arduino_header 16 GPIO_ACTIVE_LOW>; /* D10 */
// 	 pinctrl-0 = <&spi3_default>;
// 	 pinctrl-1 = <&spi3_sleep>;
// 	 pinctrl-names = "default", "sleep";
//  };
 
 &ieee802154 {
	 status = "okay";
 };
 
 &flash0 {
 
	 partitions {
		 compatible = "fixed-partitions";
		 #address-cells = <1>;
		 #size-cells = <1>;
 
		 boot_partition: partition@0 {
			 label = "mcuboot";
			 reg = <0x00000000 0x0000C000>;
		 };
		 slot0_partition: partition@c000 {
			 label = "image-0";
			 reg = <0x0000C000 0x00076000>;
		 };
		 slot1_partition: partition@82000 {
			 label = "image-1";
			 reg = <0x00082000 0x00076000>;
		 };
 
		 /*
		  * The flash starting at 0x000f8000 and ending at
		  * 0x000fffff is reserved for use by the application.
		  */
 
		 /*
		  * Storage partition will be used by FCB/LittleFS/NVS
		  * if enabled.
		  */
		 storage_partition: partition@f8000 {
			 label = "storage";
			 reg = <0x000f8000 0x00008000>;
		 };
	 };
 };
 
 zephyr_udc0: &usbd {
	 compatible = "nordic,nrf-usbd";
	 status = "okay";
 };


i am attempting to configure the magnetometer, however when ever i try to get the device instance for the magnetometer, i receive the following error at compilation. 

\lsm303agr_m\src\main.c:120: undefined reference to `__device_dts_ord_127'

This only happens when i call any functions that require me to pass a device instance. I have tried referencing the magnetometer with different macros, but this has not fixed the issue. i can also return parameters such as 'label' or 'status' from the magnetometer but i cannot get the device instance. if i comment out the line:

    if (!device_is_ready(m_lsm_dev))
and also:
    i2c_reg_read_byte(m_lsm_dev, LSM_TWI_ADDR, 0X1E, &data);
then it will compile.

main.c:

/*
 * Copyright (c) 2012-2014 Wind River Systems, Inc.
 *
 * SPDX-License-Identifier: Apache-2.0
 */
#include <zephyr/devicetree.h>
#include <zephyr/kernel.h>
#include "lsm303agr_reg.h"
#include <zephyr/logging/log.h>
#include <zephyr/device.h>
#include <zephyr/drivers/i2c.h>
#include <zephyr/drivers/gpio.h>
#include <zephyr/sys/util.h>
#include <inttypes.h>
#include <stdlib.h>


LOG_MODULE_REGISTER(foo, CONFIG_LOG_DEFAULT_LEVEL);
#define DT_DRV_COMPAT nordic_nrf_twi 
#define DT_I2C0_N DT_PATH(soc, i2c_40003000)
//#define gpio DT_PROP(DT_CHILD(LABEL, lsm303agr), label)
#define LSM_TWI_ADDR 0x19
////////////////////////////////////////////////////////
#define MAGNT DT_ALIAS(magnetometer)
#if !DT_NODE_HAS_STATUS(MAGNT, okay)
#error "Unsupported board: mag devicetree alias is not defined"
#endif

stmdev_ctx_t lsm_twi_rw;
char* labl = DT_PROP(MAGNT, label);

static const struct device* m_lsm_dev = DEVICE_DT_GET(MAGNT);
static struct gpio_callback lsm_cb;
static struct gpio_dt_spec lsmint = GPIO_DT_SPEC_GET_OR(MAGNT, int_gpios, {0});
lsm303agr_reg_t m_lsm_reg; 

//#define lsmint DT_PHANDLE(LSM, interrupts)

void init_lsm()
{
	uint8_t err, data;
	memset(&m_lsm_reg.cfg_reg_a_m , 0, sizeof(m_lsm_reg.cfg_reg_a_m ));

	m_lsm_reg.cfg_reg_a_m.comp_temp_en 				= true;
	m_lsm_reg.cfg_reg_a_m.lp 						= true;
	m_lsm_reg.cfg_reg_b_m.lpf 						= true;
	m_lsm_reg.cfg_reg_b_m.int_on_dataoff 			= true; 
	m_lsm_reg.cfg_reg_c_m.drdy_on_pin				= true;
	m_lsm_reg.cfg_reg_c_m.bdu						= true;
	m_lsm_reg.cfg_reg_a_m.md						= 0x03;
	m_lsm_reg.cfg_reg_a_m.odr						= LSM303AGR_MG_ODR_10Hz;
	i2c_reg_read_byte(m_lsm_dev, LSM_TWI_ADDR, 0X1E, &data);

	if (data != 0x40)
	{
		LOG_INF("WHOAMINOTFOUND");
		
	}

}

void lsm_set_rw()
{
	lsm_twi_rw.read_reg = i2c_write_read;
	lsm_twi_rw.write_reg = i2c_write;
	//lsm_twi_rw.handle = &m_lsm_dev;
}

void lsm_handler()
{
	LOG_INF("MAGREAD");
}

static void mag_handler()
{
	LOG_INF("MAG INT");
}

void main(void)
{

	uint8_t ret;
	LOG_INF("PIN = %u dt flags =%u Port = %u labl=%s", lsmint.pin, lsmint.dt_flags, lsmint.port, labl);
	const struct device *i2c_dev = DEVICE_DT_GET(DT_I2C0_N);

	if (!device_is_ready(m_lsm_dev)) 
	{
   		LOG_INF("Can't bind I2C device %s\n", m_lsm_dev);
    	return;
  	}

 	if (lsmint.port && !device_is_ready(lsmint.port)) {
		LOG_INF("Error %d: LED device %s is not ready; ignoring it\n",
		       ret, lsmint.port->name);
		lsmint.port = NULL;
	}
	if (lsmint.port) {
		ret = gpio_pin_configure_dt(&lsmint, GPIO_OUTPUT);
		if (ret != 0) {
			LOG_INF("Error %d: failed to configure LED device %s pin %d\n",
			       ret, lsmint.port->name, lsmint.pin);
			lsmint.port = NULL;
		} else {
			LOG_INF("Set up LED at %s pin %d\n", lsmint.port->name, lsmint.pin);
		}
	}


	ret = gpio_pin_interrupt_configure_dt(&lsmint,
					      GPIO_INT_EDGE_TO_ACTIVE);
	if (ret != 0) {
		LOG_INF("Error %d: failed to configure interrupt on %s pin %d\n",
			ret, lsmint.port->name, lsmint.pin);
		return;
	}
	gpio_init_callback(&lsm_cb, mag_handler, BIT(lsmint.pin));
	gpio_add_callback(lsmint.port, &lsm_cb);	

	//lsm_set_rw();
	init_lsm();
	//err_code = lsm303agr_twi_read(p_lsm303agr_m->p_i2c_instance, 0x1E, 0x4F, &data, sizeof(uint8_t));



	//lsm303agr_read_reg(&foo, 0, x, y);

}

 

thank you in advance.

Parents Reply
  • Hi

    I have just used that for now as a template binding as it a 'int-gpios' property which as far as im aware is required to call GPIO_DT_SPEC_GET_OR. there is a LSM magnetometer  binding, but it is for a slightly different model and has a property for 'interrupts' instead which seems to be an invalid syntax when calling GPIO_DT_SPEC_GET_OR. 

    Are there any other bindings you would suggest for arbitrary twi properties? 

Children
  • What do you mean it has a 'int-gpios' property?

    Where is it defined?

  • for me, rohm,bh1749.yaml is included in the v2.2.0 SDK: \v2.2.0\nrf\dts\bindings\sensor\rohm,bh1749.yaml

  • Oh I see.

    While it certainly isn't good code practice to use a binding for a different sensor, I don't really see why it shouldn't compile.

    It could be worth looking into how the NCS BH1749 sample is set up.

    One thing I suspect is that nrf/dts/bindings might not be included in your project by default, I notice that the NCS sample has target_sources(app PRIVATE ${app_sources}) included in their CMakeLists.txt, that might be worth trying if you're not already doing it.

    Also, this is what that sample's compiled dts node looks like:

    i2c2: i2c@a000 {
    				compatible = "nordic,nrf-twim";
    				#address-cells = < 0x1 >;
    				#size-cells = < 0x0 >;
    				reg = < 0xa000 0x1000 >;
    				clock-frequency = < 0x61a80 >;
    				interrupts = < 0xa 0x1 >;
    				status = "okay";
    				pinctrl-0 = < &i2c2_default >;
    				pinctrl-1 = < &i2c2_sleep >;
    				pinctrl-names = "default", "sleep";
    				bme680: bme680@76 {
    					compatible = "bosch,bme680";
    					reg = < 0x76 >;
    				};
    				bh1749: bh1749@38 {
    					compatible = "rohm,bh1749";
    					reg = < 0x38 >;
    					int-gpios = < &gpio0 0x1b 0x11 >;
    				};
    			};

    First thing I notice is they use nrf-twim rather than nrf-twi, maybe worth a shot.

    -Einar

Related