Differences
This shows you the differences between two versions of the page.
| Both sides previous revision Previous revision Next revision | Previous revision | ||
| operstate [2018/08/26 15:55] – [Philosophy] rpjday | operstate [2018/08/27 11:07] (current) – [Testing under sysfs] rpjday | ||
|---|---|---|---|
| Line 66: | Line 66: | ||
| ==== include/ | ==== include/ | ||
| + | |||
| + | Possible values for '' | ||
| < | < | ||
| Line 116: | Line 118: | ||
| ===== Testing under sysfs ===== | ===== Testing under sysfs ===== | ||
| - | Unplugging and plugging: | + | Unplugging and plugging | 
| < | < | ||
| Line 150: | Line 152: | ||
| </ | </ | ||
| - | This should be set by '' | + | This should be set by '' | 
| ==== netif_oper_up() ==== | ==== netif_oper_up() ==== | ||
| - | This is the important one related to plugging and unplugging: | + | This is the important one related to plugging and unplugging | 
| < | < | ||
| Line 169: | Line 171: | ||
| </ | </ | ||
| ==== netif_carrier_ok() ==== | ==== netif_carrier_ok() ==== | ||
| + | |||
| + | This routine becomes important shortly: | ||
| < | < | ||
| Line 188: | Line 192: | ||
| Clearly(?) based on presence or absence of carrier (see tests above). | Clearly(?) based on presence or absence of carrier (see tests above). | ||
| + | |||
| + | ==== net/ | ||
| + | |||
| + | Consults '' | ||
| + | |||
| + | < | ||
| + | static ssize_t carrier_show(struct device *dev, | ||
| + | struct device_attribute *attr, char *buf) | ||
| + | { | ||
| + | struct net_device *netdev = to_net_dev(dev); | ||
| + | |||
| + | if (netif_running(netdev)) | ||
| + | return sprintf(buf, | ||
| + | |||
| + | return -EINVAL; | ||
| + | } | ||
| + | static DEVICE_ATTR_RW(carrier); | ||
| + | </ | ||
| + | |||
| + | defined as: | ||
| + | |||
| + | < | ||
| + | static inline bool netif_carrier_ok(const struct net_device *dev) | ||
| + | { | ||
| + | return !test_bit(__LINK_STATE_NOCARRIER, | ||
| + | } | ||
| + | </ | ||
| + | |||
| + | ==== Where is IF_OPER_DOWN set? ==== | ||
| + | |||
| + | From '' | ||
| + | |||
| + | < | ||
| + | static unsigned char default_operstate(const struct net_device *dev) | ||
| + | { | ||
| + | if (!netif_carrier_ok(dev)) | ||
| + | return (dev-> | ||
| + | IF_OPER_LOWERLAYERDOWN : IF_OPER_DOWN); | ||
| + | |||
| + | if (netif_dormant(dev)) | ||
| + | return IF_OPER_DORMANT; | ||
| + | |||
| + | return IF_OPER_UP; | ||
| + | } | ||
| + | </ | ||
| + | |||
| + | which brings us back to who sets '' | ||
| + | |||
| + | ==== Setting/ | ||
| + | |||
| + | From '' | ||
| + | |||
| + | < | ||
| + | /** | ||
| + |  | ||
| + |  | ||
| + | * | ||
| + | * Device has detected that carrier. | ||
| + | */ | ||
| + | void netif_carrier_on(struct net_device *dev) | ||
| + | { | ||
| + | if (test_and_clear_bit(__LINK_STATE_NOCARRIER, | ||
| + | if (dev-> | ||
| + | return; | ||
| + | atomic_inc(& | ||
| + | linkwatch_fire_event(dev); | ||
| + | if (netif_running(dev)) | ||
| + | __netdev_watchdog_up(dev); | ||
| + | } | ||
| + | } | ||
| + | EXPORT_SYMBOL(netif_carrier_on); | ||
| + | |||
| + | /** | ||
| + |  | ||
| + |  | ||
| + | * | ||
| + | * Device has detected loss of carrier. | ||
| + | */ | ||
| + | void netif_carrier_off(struct net_device *dev) | ||
| + | { | ||
| + | if (!test_and_set_bit(__LINK_STATE_NOCARRIER, | ||
| + | if (dev-> | ||
| + | return; | ||
| + | atomic_inc(& | ||
| + | linkwatch_fire_event(dev); | ||
| + | } | ||
| + | } | ||
| + | EXPORT_SYMBOL(netif_carrier_off); | ||
| + | </ | ||
| + | |||
| + | So who calls '' | ||