001// Copyright (c) FIRST and other WPILib contributors. 002// Open Source Software; you can modify and/or share it under the terms of 003// the WPILib BSD license file in the root directory of this project. 004 005package edu.wpi.first.wpilibj; 006 007import static edu.wpi.first.util.ErrorMessages.requireNonNullParam; 008 009import java.util.concurrent.atomic.AtomicBoolean; 010import java.util.function.BiConsumer; 011 012/** 013 * Class for handling asynchronous interrupts using a callback thread. 014 * 015 * <p>By default, interrupts will occur on rising edge. Callbacks are disabled by default, and 016 * enable() must be called before they will occur. 017 * 018 * <p>Both rising and falling edges can be indicated in one callback if both a rising and falling 019 * edge occurred since the previous callback. 020 * 021 * <p>Synchronous (blocking) interrupts are handled by the SynchronousInterrupt class. 022 */ 023public class AsynchronousInterrupt implements AutoCloseable { 024 private final BiConsumer<Boolean, Boolean> m_callback; 025 private final SynchronousInterrupt m_interrupt; 026 027 private final AtomicBoolean m_keepRunning = new AtomicBoolean(false); 028 private Thread m_thread; 029 030 /** 031 * Construct a new asynchronous interrupt using a Digital Source. 032 * 033 * <p>At construction, the interrupt will trigger on the rising edge. 034 * 035 * <p>Callbacks will not be triggered until enable() is called. 036 * 037 * <p>The first bool in the callback indicates the rising edge triggered the interrupt, the second 038 * bool is falling edge. 039 * 040 * @param source The digital source to use. 041 * @param callback The callback to call on an interrupt 042 */ 043 public AsynchronousInterrupt(DigitalSource source, BiConsumer<Boolean, Boolean> callback) { 044 m_callback = requireNonNullParam(callback, "callback", "AsynchronousInterrupt"); 045 m_interrupt = new SynchronousInterrupt(source); 046 } 047 048 /** 049 * Closes the interrupt. 050 * 051 * <p>This does not close the associated digital source. 052 * 053 * <p>This will disable the interrupt if it is enabled. 054 */ 055 @Override 056 public void close() { 057 disable(); 058 m_interrupt.close(); 059 } 060 061 /** 062 * Enables interrupt callbacks. Before this, callbacks will not occur. Does nothing if already 063 * enabled. 064 */ 065 public void enable() { 066 if (m_keepRunning.get()) { 067 return; 068 } 069 070 m_keepRunning.set(true); 071 m_thread = new Thread(this::threadMain); 072 m_thread.start(); 073 } 074 075 /** Disables interrupt callbacks. Does nothing if already disabled. */ 076 public void disable() { 077 m_keepRunning.set(false); 078 m_interrupt.wakeupWaitingInterrupt(); 079 if (m_thread != null) { 080 if (m_thread.isAlive()) { 081 try { 082 m_thread.interrupt(); 083 m_thread.join(); 084 } catch (InterruptedException ex) { 085 Thread.currentThread().interrupt(); 086 } 087 } 088 m_thread = null; 089 } 090 } 091 092 /** 093 * Set which edges to trigger the interrupt on. 094 * 095 * @param risingEdge Trigger on rising edge 096 * @param fallingEdge Trigger on falling edge 097 */ 098 public void setInterruptEdges(boolean risingEdge, boolean fallingEdge) { 099 m_interrupt.setInterruptEdges(risingEdge, fallingEdge); 100 } 101 102 /** 103 * Get the timestamp of the last rising edge. 104 * 105 * <p>This function does not require the interrupt to be enabled to work. 106 * 107 * <p>This only works if rising edge was configured using setInterruptEdges. 108 * 109 * @return the timestamp in seconds relative to getFPGATime 110 */ 111 public double getRisingTimestamp() { 112 return m_interrupt.getRisingTimestamp(); 113 } 114 115 /** 116 * Get the timestamp of the last falling edge. 117 * 118 * <p>This function does not require the interrupt to be enabled to work. 119 * 120 * <p>This only works if falling edge was configured using setInterruptEdges. 121 * 122 * @return the timestamp in seconds relative to getFPGATime 123 */ 124 public double getFallingTimestamp() { 125 return m_interrupt.getFallingTimestamp(); 126 } 127 128 private void threadMain() { 129 while (m_keepRunning.get()) { 130 var result = m_interrupt.waitForInterrupt(10, false); 131 if (!m_keepRunning.get()) { 132 break; 133 } 134 boolean rising = false; 135 boolean falling = false; 136 switch (result) { 137 case kBoth -> { 138 rising = true; 139 falling = true; 140 } 141 case kFallingEdge -> { 142 falling = true; 143 } 144 case kRisingEdge -> { 145 rising = true; 146 } 147 default -> { 148 continue; 149 } 150 } 151 152 m_callback.accept(rising, falling); 153 } 154 } 155}