mirror of https://github.com/ARMmbed/mbed-os.git
				
				
				
			
		
			
				
	
	
		
			189 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			C
		
	
	
			
		
		
	
	
			189 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			C
		
	
	
/* mbed Microcontroller Library
 | 
						|
 * Copyright (c) 2006-2013 ARM Limited
 | 
						|
 * SPDX-License-Identifier: Apache-2.0
 | 
						|
 *
 | 
						|
 * Licensed under the Apache License, Version 2.0 (the "License");
 | 
						|
 * you may not use this file except in compliance with the License.
 | 
						|
 * You may obtain a copy of the License at
 | 
						|
 *
 | 
						|
 *     http://www.apache.org/licenses/LICENSE-2.0
 | 
						|
 *
 | 
						|
 * Unless required by applicable law or agreed to in writing, software
 | 
						|
 * distributed under the License is distributed on an "AS IS" BASIS,
 | 
						|
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
						|
 * See the License for the specific language governing permissions and
 | 
						|
 * limitations under the License.
 | 
						|
 */
 | 
						|
#include "hal/pinmap.h"
 | 
						|
#include "platform/mbed_error.h"
 | 
						|
 | 
						|
void pinmap_pinout(PinName pin, const PinMap *map)
 | 
						|
{
 | 
						|
    if (pin == NC) {
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    while (map->pin != NC) {
 | 
						|
        if (map->pin == pin) {
 | 
						|
            pin_function(pin, map->function);
 | 
						|
 | 
						|
            pin_mode(pin, PullNone);
 | 
						|
            return;
 | 
						|
        }
 | 
						|
        map++;
 | 
						|
    }
 | 
						|
    MBED_ERROR1(MBED_MAKE_ERROR(MBED_MODULE_PLATFORM, MBED_ERROR_CODE_PINMAP_INVALID), "could not pinout", pin);
 | 
						|
}
 | 
						|
 | 
						|
uint32_t pinmap_merge(uint32_t a, uint32_t b)
 | 
						|
{
 | 
						|
    // both are the same (inc both NC)
 | 
						|
    if (a == b) {
 | 
						|
        return a;
 | 
						|
    }
 | 
						|
 | 
						|
    // one (or both) is not connected
 | 
						|
    if (a == (uint32_t)NC) {
 | 
						|
        return b;
 | 
						|
    }
 | 
						|
    if (b == (uint32_t)NC) {
 | 
						|
        return a;
 | 
						|
    }
 | 
						|
 | 
						|
    // mis-match error case
 | 
						|
    MBED_ERROR1(MBED_MAKE_ERROR(MBED_MODULE_PLATFORM, MBED_ERROR_CODE_PINMAP_INVALID), "pinmap mis-match", a);
 | 
						|
}
 | 
						|
 | 
						|
uint32_t pinmap_find_peripheral(PinName pin, const PinMap *map)
 | 
						|
{
 | 
						|
    while (map->pin != NC) {
 | 
						|
        if (map->pin == pin) {
 | 
						|
            return map->peripheral;
 | 
						|
        }
 | 
						|
        map++;
 | 
						|
    }
 | 
						|
    return (uint32_t)NC;
 | 
						|
}
 | 
						|
 | 
						|
uint32_t pinmap_peripheral(PinName pin, const PinMap *map)
 | 
						|
{
 | 
						|
    uint32_t peripheral = (uint32_t)NC;
 | 
						|
 | 
						|
    if (pin == (PinName)NC) {
 | 
						|
        return (uint32_t)NC;
 | 
						|
    }
 | 
						|
    peripheral = pinmap_find_peripheral(pin, map);
 | 
						|
    if ((uint32_t)NC == peripheral) { // no mapping available
 | 
						|
        MBED_ERROR1(MBED_MAKE_ERROR(MBED_MODULE_PLATFORM, MBED_ERROR_CODE_PINMAP_INVALID), "pinmap not found for peripheral", pin);
 | 
						|
    }
 | 
						|
    return peripheral;
 | 
						|
}
 | 
						|
 | 
						|
uint32_t pinmap_find_function(PinName pin, const PinMap *map)
 | 
						|
{
 | 
						|
    while (map->pin != NC) {
 | 
						|
        if (map->pin == pin) {
 | 
						|
            return map->function;
 | 
						|
        }
 | 
						|
        map++;
 | 
						|
    }
 | 
						|
    return (uint32_t)NC;
 | 
						|
}
 | 
						|
 | 
						|
uint32_t pinmap_function(PinName pin, const PinMap *map)
 | 
						|
{
 | 
						|
    uint32_t function = (uint32_t)NC;
 | 
						|
 | 
						|
    if (pin == (PinName)NC) {
 | 
						|
        return (uint32_t)NC;
 | 
						|
    }
 | 
						|
    function = pinmap_find_function(pin, map);
 | 
						|
    if ((uint32_t)NC == function) { // no mapping available
 | 
						|
        MBED_ERROR1(MBED_MAKE_ERROR(MBED_MODULE_PLATFORM, MBED_ERROR_CODE_PINMAP_INVALID), "pinmap not found for function", pin);
 | 
						|
    }
 | 
						|
    return function;
 | 
						|
}
 | 
						|
 | 
						|
bool pinmap_find_peripheral_pins(const PinList *whitelist, const PinList *blacklist, int per, const PinMap *const *maps, PinName **pins, uint32_t count)
 | 
						|
{
 | 
						|
    /*
 | 
						|
     * This function uses recursion to find a suitable set of pins which meet the requirements.
 | 
						|
     * Recursion is at max the number of pinmaps passed in - the 'count' parameter. Because of this
 | 
						|
     * there is no risk of a stack overflow due to unbounded recursion.
 | 
						|
     *
 | 
						|
     * Below is a psuedo code example of this function's operation when finding a set of 4 pins.
 | 
						|
     * The recursion depth is indicated by the number in front.
 | 
						|
     *
 | 
						|
     *  1. Given 4 maps and a peripheral find 4 suitable pins
 | 
						|
     *      2. Given 4 maps, a peripheral and 1 pin find 3 suitable pins
 | 
						|
     *          3. Given 4 maps, a peripheral and 2 pins find 2 suitable pins
 | 
						|
     *              4. Given 4 maps, a peripheral and 3 pins find 1 suitable pin
 | 
						|
     *              4. Return success if all pins are found, return failure if there are no suitable pins, otherwise choose the next pin and retry
 | 
						|
     *          3. Return success if all pins are found, return failure if there are no suitable pins, otherwise choose the next pin and retry
 | 
						|
     *      2. Return success if all pins are found, return failure if there are no suitable pins, otherwise choose the next pin and retry
 | 
						|
     *  1. Return success if all pins are found, return failure if there are no suitable pins, otherwise choose the next pin and retry
 | 
						|
     *
 | 
						|
     */
 | 
						|
 | 
						|
    for (uint32_t i = 0; i < count; i++) {
 | 
						|
        const PinMap *map = maps[i];
 | 
						|
        PinName *pin = pins[i];
 | 
						|
        if (*pin == NC) {
 | 
						|
            for (; map->pin != NC; map++) {
 | 
						|
                if (map->peripheral != per) {
 | 
						|
                    continue;
 | 
						|
                }
 | 
						|
                if (!pinmap_list_has_pin(whitelist, map->pin)) {
 | 
						|
                    // Not part of this form factor
 | 
						|
                    continue;
 | 
						|
                }
 | 
						|
                if (pinmap_list_has_pin(blacklist, map->pin)) {
 | 
						|
                    // Restricted pin
 | 
						|
                    continue;
 | 
						|
                }
 | 
						|
                bool already_in_use = false;
 | 
						|
                for (uint32_t j = 0; j < count; j++) {
 | 
						|
                    if (j == i) {
 | 
						|
                        // Don't compare with self
 | 
						|
                        continue;
 | 
						|
                    }
 | 
						|
                    if (map->pin == *pins[j]) {
 | 
						|
                        already_in_use = true;
 | 
						|
                        break;
 | 
						|
                    }
 | 
						|
                }
 | 
						|
                if (already_in_use) {
 | 
						|
                    continue;
 | 
						|
                }
 | 
						|
                *pin = map->pin;
 | 
						|
                if (pinmap_find_peripheral_pins(whitelist, blacklist, per, maps, pins, count)) {
 | 
						|
                    return true;
 | 
						|
                }
 | 
						|
            }
 | 
						|
            *pin = NC;
 | 
						|
            return false;
 | 
						|
        }
 | 
						|
    }
 | 
						|
    return true;
 | 
						|
}
 | 
						|
 | 
						|
bool pinmap_list_has_pin(const PinList *list, PinName pin)
 | 
						|
{
 | 
						|
    for (uint32_t i = 0; i < list->count; i++) {
 | 
						|
        if (list->pins[i] == pin) {
 | 
						|
            return true;
 | 
						|
        }
 | 
						|
    }
 | 
						|
    return false;
 | 
						|
}
 | 
						|
 | 
						|
bool pinmap_list_has_peripheral(const PeripheralList *list, int peripheral)
 | 
						|
{
 | 
						|
    for (uint32_t i = 0; i < list->count; i++) {
 | 
						|
        if (list->peripheral[i] == peripheral) {
 | 
						|
            return true;
 | 
						|
        }
 | 
						|
    }
 | 
						|
    return false;
 | 
						|
}
 |