// SPDX-License-Identifier: GPL-2.0+ /* * (C) Copyright 2021 * Köry Maincent, Bootlin, * Based on initial code from Maxime Ripard */ #include #include #include #include #include #include #include #define for_each_w1_device(b, d) \ for (device_find_first_child(b, d); *d; device_find_next_child(d)) #define dip_convert(field) \ ( \ (sizeof(field) == 1) ? field : \ (sizeof(field) == 2) ? be16_to_cpu(field) : \ (sizeof(field) == 4) ? be32_to_cpu(field) : \ -1 \ ) #define DIP_MAGIC 0x50494843 /* CHIP */ struct dip_w1_header { u32 magic; /* CHIP */ u8 version; /* spec version */ u32 vendor_id; u16 product_id; u8 product_version; char vendor_name[32]; char product_name[32]; u8 rsvd[36]; /* rsvd for future spec versions */ u8 data[16]; /* user data, per-dip specific */ } __packed; static int sunxi_extension_board_scan(struct udevice *udev, struct alist *extension_list) { struct udevice *bus, *dev; int num_dip = 0; int ret; sunxi_gpio_set_pull(SUNXI_GPD(2), SUNXI_GPIO_PULL_UP); ret = w1_get_bus(0, &bus); if (ret) { printf("one wire interface not found\n"); return 0; } for_each_w1_device(bus, &dev) { struct extension dip = {0}; struct dip_w1_header w1_header; u32 vid; u16 pid; if (w1_get_device_family(dev) != W1_FAMILY_DS2431) continue; ret = device_probe(dev); if (ret) { printf("Couldn't probe device %s: error %d", dev->name, ret); continue; } w1_eeprom_read_buf(dev, 0, (u8 *)&w1_header, sizeof(w1_header)); if (w1_header.magic != DIP_MAGIC) continue; vid = dip_convert(w1_header.vendor_id); pid = dip_convert(w1_header.product_id); printf("DIP: %s (0x%x) from %s (0x%x)\n", w1_header.product_name, pid, w1_header.vendor_name, vid); snprintf(dip.overlay, sizeof(dip.overlay), "dip-%x-%x.dtbo", vid, pid); strlcpy(dip.name, w1_header.product_name, sizeof(dip.name)); strlcpy(dip.owner, w1_header.vendor_name, sizeof(dip.owner)); if (!alist_add(extension_list, dip)) return -ENOMEM; num_dip++; } return num_dip; } U_BOOT_EXTENSION(dip, sunxi_extension_board_scan); U_BOOT_DRVINFO(dip) = { .name = "dip", };