automount.c 10.2 KB
/************************************************************************************
 *
 *   Copyright (C) 2016-2018 Gregory Nutt. All rights reserved.
 *   Authors: Gregory Nutt <gnutt@nuttx.org>
 *            David Sidrane <david_s5@nscdg.com>
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions
 * are met:
 *
 * 1. Redistributions of source code must retain the above copyright
 *    notice, this list of conditions and the following disclaimer.
 * 2. Redistributions in binary form must reproduce the above copyright
 *    notice, this list of conditions and the following disclaimer in
 *    the documentation and/or other materials provided with the
 *    distribution.
 * 3. Neither the name NuttX nor the names of its contributors may be
 *    used to endorse or promote products derived from this software
 *    without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
 * "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
 * LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
 * FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
 * COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
 * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
 * BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
 * OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
 * AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
 * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN
 * ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
 * POSSIBILITY OF SUCH DAMAGE.
 *
 ************************************************************************************/

/************************************************************************************
 * Included Files
 ************************************************************************************/

#include <px4_platform_common/px4_config.h>

#if defined(CONFIG_FS_AUTOMOUNTER_DEBUG) && !defined(CONFIG_DEBUG_FS)
#  define CONFIG_DEBUG_FS 1
#endif

#include <debug.h>
#include <stddef.h>

#include <nuttx/irq.h>
#include <nuttx/clock.h>
#include <nuttx/fs/automount.h>

#include "board_config.h"
#ifdef HAVE_AUTOMOUNTER

/************************************************************************************
 * Pre-processor Definitions
 ************************************************************************************/

/************************************************************************************
 * Private Types
 ************************************************************************************/
/* This structure represents the changeable state of the automounter */

struct fmuk66_automount_state_s {
	volatile automount_handler_t handler;    /* Upper half handler */
	FAR void *arg;                           /* Handler argument */
	bool enable;                             /* Fake interrupt enable */
	bool pending;                            /* Set if there an event while disabled */
};

/* This structure represents the static configuration of an automounter */

struct fmuk66_automount_config_s {
	/* This must be first thing in structure so that we can simply cast from struct
	 * automount_lower_s to struct fmuk66_automount_config_s
	 */

	struct automount_lower_s lower;          /* Publicly visible part */
	FAR struct fmuk66_automount_state_s *state; /* Changeable state */
};

/************************************************************************************
 * Private Function Prototypes
 ************************************************************************************/

static int  fmuk66_attach(FAR const struct automount_lower_s *lower, automount_handler_t isr, FAR void *arg);
static void fmuk66_enable(FAR const struct automount_lower_s *lower, bool enable);
static bool fmuk66_inserted(FAR const struct automount_lower_s *lower);

/************************************************************************************
 * Private Data
 ************************************************************************************/

static struct fmuk66_automount_state_s g_sdhc_state;
static const struct fmuk66_automount_config_s g_sdhc_config = {
	.lower        =
	{
		.fstype     = CONFIG_FMUK66_SDHC_AUTOMOUNT_FSTYPE,
		.blockdev   = CONFIG_FMUK66_SDHC_AUTOMOUNT_BLKDEV,
		.mountpoint = CONFIG_FMUK66_SDHC_AUTOMOUNT_MOUNTPOINT,
		.ddelay     = MSEC2TICK(CONFIG_FMUK66_SDHC_AUTOMOUNT_DDELAY),
		.udelay     = MSEC2TICK(CONFIG_FMUK66_SDHC_AUTOMOUNT_UDELAY),
		.attach     = fmuk66_attach,
		.enable     = fmuk66_enable,
		.inserted   = fmuk66_inserted
	},
	.state        = &g_sdhc_state
};

/************************************************************************************
 * Private Functions
 ************************************************************************************/

/************************************************************************************
 * Name:  fmuk66_attach
 *
 * Description:
 *   Attach a new SDHC event handler
 *
 * Input Parameters:
 *   lower - An instance of the auto-mounter lower half state structure
 *   isr   - The new event handler to be attach
 *   arg   - Client data to be provided when the event handler is invoked.
 *
 *  Returned Value:
 *    Always returns OK
 *
 ************************************************************************************/

static int fmuk66_attach(FAR const struct automount_lower_s *lower, automount_handler_t isr, FAR void *arg)
{
	FAR const struct fmuk66_automount_config_s *config;
	FAR struct fmuk66_automount_state_s *state;

	/* Recover references to our structure */

	config = (FAR struct fmuk66_automount_config_s *)lower;
	DEBUGASSERT(config != NULL && config->state != NULL);

	state = config->state;

	/* Save the new handler info (clearing the handler first to eliminate race
	 * conditions).
	 */

	state->handler = NULL;
	state->pending = false;
	state->arg     = arg;
	state->handler = isr;
	return OK;
}

/************************************************************************************
 * Name:  fmuk66_enable
 *
 * Description:
 *   Enable card insertion/removal event detection
 *
 * Input Parameters:
 *   lower - An instance of the auto-mounter lower half state structure
 *   enable - True: enable event detection; False: disable
 *
 *  Returned Value:
 *    None
 *
 ************************************************************************************/

static void fmuk66_enable(FAR const struct automount_lower_s *lower, bool enable)
{
	FAR const struct fmuk66_automount_config_s *config;
	FAR struct fmuk66_automount_state_s *state;
	irqstate_t flags;

	/* Recover references to our structure */

	config = (FAR struct fmuk66_automount_config_s *)lower;
	DEBUGASSERT(config != NULL && config->state != NULL);

	state = config->state;

	/* Save the fake enable setting */

	flags = enter_critical_section();
	state->enable = enable;

	/* Did an interrupt occur while interrupts were disabled? */

	if (enable && state->pending) {
		/* Yes.. perform the fake interrupt if the interrutp is attached */

		if (state->handler) {
			bool inserted = fmuk66_cardinserted();
			(void)state->handler(&config->lower, state->arg, inserted);
		}

		state->pending = false;
	}

	leave_critical_section(flags);
}

/************************************************************************************
 * Name: fmuk66_inserted
 *
 * Description:
 *   Check if a card is inserted into the slot.
 *
 * Input Parameters:
 *   lower - An instance of the auto-mounter lower half state structure
 *
 *  Returned Value:
 *    True if the card is inserted; False otherwise
 *
 ************************************************************************************/

static bool fmuk66_inserted(FAR const struct automount_lower_s *lower)
{
	return fmuk66_cardinserted();
}

/************************************************************************************
 * Public Functions
 ************************************************************************************/

/************************************************************************************
 * Name:  fmuk66_automount_initialize
 *
 * Description:
 *   Configure auto-mounters for each enable and so configured SDHC
 *
 * Input Parameters:
 *   None
 *
 *  Returned Value:
 *    None
 *
 ************************************************************************************/

void fmuk66_automount_initialize(void)
{
	FAR void *handle;

	finfo("Initializing automounter(s)\n");

	/* Initialize the SDHC0 auto-mounter */

	handle = automount_initialize(&g_sdhc_config.lower);

	if (!handle) {
		ferr("ERROR: Failed to initialize auto-mounter for SDHC0\n");
	}
}

/************************************************************************************
 * Name:  fmuk66_automount_event
 *
 * Description:
 *   The SDHC card detection logic has detected an insertion or removal event.  It
 *   has already scheduled the MMC/SD block driver operations.  Now we need to
 *   schedule the auto-mount event which will occur with a substantial delay to make
 *   sure that everything has settle down.
 *
 * Input Parameters:
 *   slotno - Identifies the SDHC0 slot: SDHC0_SLOTNO or SDHC1_SLOTNO.  There is a
 *      terminology problem here:  Each SDHC supports two slots, slot A and slot B.
 *      Only slot A is used.  So this is not a really a slot, but an HSCMI peripheral
 *      number.
 *   inserted - True if the card is inserted in the slot.  False otherwise.
 *
 *  Returned Value:
 *    None
 *
 *  Assumptions:
 *    Interrupts are disabled.
 *
 ************************************************************************************/

void fmuk66_automount_event(bool inserted)
{
	FAR const struct fmuk66_automount_config_s *config = &g_sdhc_config;
	FAR struct fmuk66_automount_state_s *state = &g_sdhc_state;

	/* Is the auto-mounter interrupt attached? */

	if (state->handler) {
		/* Yes.. Have we been asked to hold off interrupts? */

		if (!state->enable) {
			/* Yes.. just remember the there is a pending interrupt. We will
			 * deliver the interrupt when interrupts are "re-enabled."
			 */

			state->pending = true;

		} else {
			/* No.. forward the event to the handler */

			(void)state->handler(&config->lower, state->arg, inserted);
		}
	}
}

#endif /* HAVE_AUTOMOUNTER */