Skip to content

Commit

Permalink
Fix clunking sound from solenoids
Browse files Browse the repository at this point in the history
The "clunking" sound that is occasionnally heard when using AYAB
is caused by all solenoid armatures being released simultaneously
when all solenoids are turned off.

This does not happen with Brother electronics because they default
to keeping solenoids on.

This commit changes AYAB's behavior to match that behavior of the
Brother electronics. Note that this required abandoning the third-party
library used to drive the I/O expanders (Adafruit_MCP23008) since
part of its non-skippable initialization code was forcing all outputs
to off. It turns out the amount of code needed to directly drive the
MCP23008 expanders using the Arduino Wire library is only a dozen
lines of code anyway.

Fixes AllYarnsAreBeautiful#15.
  • Loading branch information
jonathanperret committed Sep 30, 2024
1 parent 85d0842 commit 8c30efc
Show file tree
Hide file tree
Showing 7 changed files with 45 additions and 35 deletions.
5 changes: 1 addition & 4 deletions .gitmodules
Original file line number Diff line number Diff line change
@@ -1,6 +1,3 @@
[submodule "libraries/PacketSerial"]
path = lib/PacketSerial
url = https://github.com/bakercp/PacketSerial.git
[submodule "libraries/Adafruit_MCP23008"]
url = https://github.com/adafruit/Adafruit-MCP23008-library.git
path = lib/Adafruit_MCP23008
url = https://github.com/bakercp/PacketSerial.git
1 change: 0 additions & 1 deletion lib/Adafruit_MCP23008
Submodule Adafruit_MCP23008 deleted from 05fc9b
53 changes: 39 additions & 14 deletions src/ayab/solenoids.cpp
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
/*!
/*!
* \file solenoids.cpp
* \brief Class containing methods that control the needles
* via solenoids connected to IO expanders on the device.
Expand All @@ -25,18 +25,29 @@

#include "solenoids.h"

// Minimal required constants from the MCP23008 datasheet:
// - fixed I2C address bits
#define MCP23008_ADDRESS 0x20
// - used register addresses
#define MCP23008_IODIR 0x00
#define MCP23008_GPIO 0x09

/*!
* \brief Initialize I2C connection for solenoids.
*/
void Solenoids::init() {
mcp_0.begin(I2Caddr_sol1_8);
mcp_1.begin(I2Caddr_sol9_16);
Wire.begin();

for (uint8_t i = 0; i < SOLENOID_BUFFER_SIZE / 2; i++) {
mcp_0.pinMode(i, OUTPUT);
mcp_1.pinMode(i, OUTPUT);
}
solenoidState = 0x0000U;
// Default to all solenoids ON
// This mimicks the behavior of Brother electronics and is done
// to avoid a "clunking" sound that occurs if solenoids are turned
// OFF after having been ON while the carriage was moving.
solenoidState = 0xffffU;
writeGPIO(solenoidState);

// Configure all MCP23008 pins as outputs
writeRegister(I2Caddr_sol1_8, MCP23008_IODIR, 0x00);
writeRegister(I2Caddr_sol9_16, MCP23008_IODIR, 0x00);
}

/*!
Expand All @@ -59,7 +70,7 @@ void Solenoids::setSolenoid(uint8_t solenoid, bool state) {
}
if (oldState != solenoidState) {
#ifndef AYAB_TESTS
write(solenoidState);
writeGPIO(solenoidState);
#endif
}
}
Expand All @@ -74,7 +85,7 @@ void Solenoids::setSolenoids(uint16_t state) {
if (state != solenoidState) {
solenoidState = state;
#ifndef AYAB_TESTS
write(state);
writeGPIO(state);
#endif
}
}
Expand All @@ -93,9 +104,23 @@ void Solenoids::setSolenoids(uint16_t state) {
* one bit per solenoid.
*/
// GCOVR_EXCL_START
void Solenoids::write(uint16_t newState) {
(void)newState;
mcp_0.writeGPIO(lowByte(newState));
mcp_1.writeGPIO(highByte(newState));
void Solenoids::writeGPIO(uint16_t newState) {
writeRegister(I2Caddr_sol1_8, MCP23008_GPIO, lowByte(newState));
writeRegister(I2Caddr_sol9_16, MCP23008_GPIO, highByte(newState));
}

/*!
* Write to an MCP23008 register via I2C
*
* \param i2caddr Address of the I/O expander, only the 3 lowest bits are
* used, mapping to the A0..A2 address pins on MCP23008
* \param reg Register address (see MCP23008 datasheet)
* \param value Value to set the register to
*/
void Solenoids::writeRegister(uint8_t i2caddr, uint8_t reg, uint8_t value) {
Wire.beginTransmission(MCP23008_ADDRESS | (i2caddr & 7));
Wire.write(reg);
Wire.write(value);
Wire.endTransmission();
}
// GCOVR_EXCL_STOP
8 changes: 3 additions & 5 deletions src/ayab/solenoids.h
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,6 @@
#include "board.h"
#include "encoders.h"
#include <Arduino.h>
#include <Adafruit_MCP23008.h>
#include <Wire.h>

// Different machines have a different number of solenoids.
Expand Down Expand Up @@ -81,11 +80,10 @@ class Solenoids : public SolenoidsInterface {
void setSolenoids(uint16_t state) final;

private:
uint16_t solenoidState = 0x0000U;
void write(uint16_t state);
uint16_t solenoidState;

Adafruit_MCP23008 mcp_0 = Adafruit_MCP23008();
Adafruit_MCP23008 mcp_1 = Adafruit_MCP23008();
void writeGPIO(uint16_t state);
void writeRegister(uint8_t i2caddr, uint8_t reg, uint8_t value);
};

#endif // SOLENOIDS_H_
2 changes: 1 addition & 1 deletion static_analysis.sh
Original file line number Diff line number Diff line change
@@ -1,2 +1,2 @@
#!/bin/bash
clang-tidy src/ayab/$1.cpp --fix -- -isystem /usr/share/arduino/hardware/arduino/cores/arduino/ -isystem /usr/lib/avr/include/ -isystem /usr/share/arduino/hardware/arduino/variants/standard -isystem libraries/Adafruit_MCP23008/ -isystem /usr/share/arduino/libraries/Wire/ -isystem libraries/PacketSerial/src/ -DCLANG_TIDY "${@:2}"
clang-tidy src/ayab/$1.cpp --fix -- -isystem /usr/share/arduino/hardware/arduino/cores/arduino/ -isystem /usr/lib/avr/include/ -isystem /usr/share/arduino/hardware/arduino/variants/standard -isystem /usr/share/arduino/libraries/Wire/ -isystem libraries/PacketSerial/src/ -DCLANG_TIDY "${@:2}"
9 changes: 0 additions & 9 deletions test/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -31,9 +31,6 @@ set(COMMON_INCLUDES
${SOURCE_DIRECTORY}
${LIBRARY_DIRECTORY}/PacketSerial/src
)
set(EXTERNAL_LIB_INCLUDES
${LIBRARY_DIRECTORY}/Adafruit_MCP23008
)
set(COMMON_SOURCES
${PROJECT_SOURCE_DIR}/test_boards.cpp

Expand Down Expand Up @@ -85,16 +82,10 @@ set(COMMON_LINKER_FLAGS
${CMAKE_THREAD_LIBS_INIT}
-lgcov
)
set(HARD_I2C_LIB
${LIBRARY_DIRECTORY}/Adafruit_MCP23008/Adafruit_MCP23008.cpp
)
function(add_board board)
set(processor # uno
__AVR_ATmega168__
)
set(I2C_LIB
${HARD_I2C_LIB}
)
add_executable(${PROJECT_NAME}_${board}
${COMMON_SOURCES}
# External libraries
Expand Down
2 changes: 1 addition & 1 deletion test/test_solenoids.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -50,7 +50,7 @@ TEST_F(SolenoidsTest, test_construct) {

TEST_F(SolenoidsTest, test_init) {
solenoids->init();
ASSERT_TRUE(solenoids->solenoidState == 0U);
ASSERT_TRUE(solenoids->solenoidState == 0xffffU);
}

TEST_F(SolenoidsTest, test_setSolenoid1) {
Expand Down

0 comments on commit 8c30efc

Please sign in to comment.