fp@409: /****************************************************************************** fp@409: * fp@409: * $Id$ fp@409: * fp@1326: * Copyright (C) 2006-2008 Florian Pose, Ingenieurgemeinschaft IgH fp@409: * fp@409: * This file is part of the IgH EtherCAT Master. fp@409: * fp@1326: * The IgH EtherCAT Master is free software; you can redistribute it and/or fp@1326: * modify it under the terms of the GNU General Public License version 2, as fp@1326: * published by the Free Software Foundation. fp@1326: * fp@1326: * The IgH EtherCAT Master is distributed in the hope that it will be useful, fp@1326: * but WITHOUT ANY WARRANTY; without even the implied warranty of fp@1326: * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General fp@1326: * Public License for more details. fp@1326: * fp@1326: * You should have received a copy of the GNU General Public License along fp@1326: * with the IgH EtherCAT Master; if not, write to the Free Software fp@409: * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA fp@409: * fp@1363: * --- fp@1363: * fp@1363: * The license mentioned above concerns the source code only. Using the fp@1363: * EtherCAT technology and brand is only permitted in compliance with the fp@1363: * industrial property and similar rights of Beckhoff Automation GmbH. fp@409: * fp@409: *****************************************************************************/ fp@409: fp@862: /** \file fp@1327: * Canopen over EtherCAT SDO request functions. fp@862: */ fp@409: fp@409: /*****************************************************************************/ fp@409: fp@409: #include fp@881: #include fp@1975: #include fp@409: fp@847: #include "sdo_request.h" fp@409: fp@409: /*****************************************************************************/ fp@409: fp@1327: /** Default timeout in ms to wait for SDO transfer responses. fp@1053: */ fp@1794: #define EC_SDO_REQUEST_RESPONSE_TIMEOUT 1000 fp@1053: fp@1053: /*****************************************************************************/ fp@1053: fp@852: void ec_sdo_request_clear_data(ec_sdo_request_t *); fp@852: fp@852: /*****************************************************************************/ fp@852: fp@1327: /** SDO request constructor. fp@758: */ fp@847: void ec_sdo_request_init( fp@1327: ec_sdo_request_t *req /**< SDO request. */ fp@831: ) fp@831: { fp@1526: req->complete_access = 0; fp@430: req->data = NULL; fp@854: req->mem_size = 0; fp@854: req->data_size = 0; fp@1055: req->dir = EC_DIR_INVALID; fp@1035: req->issue_timeout = 0; // no timeout fp@1053: req->response_timeout = EC_SDO_REQUEST_RESPONSE_TIMEOUT; fp@1209: req->state = EC_INT_REQUEST_INIT; fp@1948: req->errno = 0; fp@1037: req->abort_code = 0x00000000; fp@430: } fp@430: fp@430: /*****************************************************************************/ fp@430: fp@1327: /** SDO request destructor. fp@847: */ fp@847: void ec_sdo_request_clear( fp@1327: ec_sdo_request_t *req /**< SDO request. */ fp@847: ) fp@430: { fp@852: ec_sdo_request_clear_data(req); fp@430: } fp@430: fp@430: /*****************************************************************************/ fp@852: fp@1327: /** Copy another SDO request. fp@1318: * fp@1318: * \attention Only the index subindex and data are copied. fp@2522: * fp@2522: * \return Zero on success, otherwise a negative error code. fp@1318: */ fp@1318: int ec_sdo_request_copy( fp@1332: ec_sdo_request_t *req, /**< SDO request. */ fp@1332: const ec_sdo_request_t *other /**< Other SDO request to copy from. */ fp@1318: ) fp@1318: { fp@1526: req->complete_access = other->complete_access; fp@1318: req->index = other->index; fp@1318: req->subindex = other->subindex; fp@1318: return ec_sdo_request_copy_data(req, other->data, other->data_size); fp@1318: } fp@1318: fp@1318: /*****************************************************************************/ fp@1318: fp@1327: /** SDO request destructor. fp@852: */ fp@852: void ec_sdo_request_clear_data( fp@1327: ec_sdo_request_t *req /**< SDO request. */ fp@852: ) fp@852: { fp@852: if (req->data) { fp@852: kfree(req->data); fp@852: req->data = NULL; fp@852: } fp@852: fp@854: req->mem_size = 0; fp@854: req->data_size = 0; fp@852: } fp@852: fp@852: /*****************************************************************************/ fp@852: fp@854: /** Pre-allocates the data memory. fp@854: * fp@854: * If the \a mem_size is already bigger than \a size, nothing is done. fp@1312: * fp@1312: * \return 0 on success, otherwise -ENOMEM. fp@854: */ fp@854: int ec_sdo_request_alloc( fp@1327: ec_sdo_request_t *req, /**< SDO request. */ fp@854: size_t size /**< Data size to allocate. */ fp@854: ) fp@854: { fp@854: if (size <= req->mem_size) fp@854: return 0; fp@854: fp@854: ec_sdo_request_clear_data(req); fp@854: fp@854: if (!(req->data = (uint8_t *) kmalloc(size, GFP_KERNEL))) { fp@1543: EC_ERR("Failed to allocate %zu bytes of SDO memory.\n", size); fp@1312: return -ENOMEM; fp@854: } fp@854: fp@854: req->mem_size = size; fp@854: req->data_size = 0; fp@854: return 0; fp@854: } fp@854: fp@854: /*****************************************************************************/ fp@854: fp@1327: /** Copies SDO data from an external source. fp@854: * fp@854: * If the \a mem_size is to small, new memory is allocated. fp@1313: * fp@1313: * \retval 0 Success. fp@1313: * \retval <0 Error code. fp@854: */ fp@854: int ec_sdo_request_copy_data( fp@1327: ec_sdo_request_t *req, /**< SDO request. */ fp@854: const uint8_t *source, /**< Source data. */ fp@854: size_t size /**< Number of bytes in \a source. */ fp@854: ) fp@854: { fp@1313: int ret = ec_sdo_request_alloc(req, size); fp@1313: if (ret < 0) fp@1313: return ret; fp@854: fp@854: memcpy(req->data, source, size); fp@854: req->data_size = size; fp@854: return 0; fp@854: } fp@854: fp@880: /*****************************************************************************/ fp@880: fp@880: /** Checks, if the timeout was exceeded. fp@880: * fp@880: * \return non-zero if the timeout was exceeded, else zero. fp@880: */ fp@1327: int ec_sdo_request_timed_out(const ec_sdo_request_t *req /**< SDO request. */) fp@880: { fp@1035: return req->issue_timeout fp@1035: && jiffies - req->jiffies_start > HZ * req->issue_timeout / 1000; fp@880: } fp@880: fp@858: /***************************************************************************** fp@1831: * Application interface. fp@858: ****************************************************************************/ fp@858: fp@2434: void ecrt_sdo_request_index(ec_sdo_request_t *req, uint16_t index, fp@2434: uint8_t subindex) fp@2434: { fp@2434: req->index = index; fp@2434: req->subindex = subindex; fp@2434: } fp@2434: fp@2434: /*****************************************************************************/ fp@2434: fp@858: void ecrt_sdo_request_timeout(ec_sdo_request_t *req, uint32_t timeout) fp@858: { fp@1035: req->issue_timeout = timeout; fp@858: } fp@858: fp@858: /*****************************************************************************/ fp@858: fp@858: uint8_t *ecrt_sdo_request_data(ec_sdo_request_t *req) fp@858: { fp@858: return req->data; fp@858: } fp@858: fp@858: /*****************************************************************************/ fp@858: fp@869: size_t ecrt_sdo_request_data_size(const ec_sdo_request_t *req) fp@869: { fp@869: return req->data_size; fp@869: } fp@869: fp@869: /*****************************************************************************/ fp@869: fp@1209: ec_request_state_t ecrt_sdo_request_state(const ec_sdo_request_t *req) fp@1209: { fp@1209: return ec_request_state_translation_table[req->state]; fp@858: } fp@858: fp@858: /*****************************************************************************/ fp@858: fp@858: void ecrt_sdo_request_read(ec_sdo_request_t *req) fp@854: { fp@859: req->dir = EC_DIR_INPUT; fp@1209: req->state = EC_INT_REQUEST_QUEUED; fp@1948: req->errno = 0; fp@1037: req->abort_code = 0x00000000; fp@1035: req->jiffies_start = jiffies; fp@854: } fp@854: fp@854: /*****************************************************************************/ fp@854: fp@858: void ecrt_sdo_request_write(ec_sdo_request_t *req) fp@854: { fp@859: req->dir = EC_DIR_OUTPUT; fp@1209: req->state = EC_INT_REQUEST_QUEUED; fp@1948: req->errno = 0; fp@1037: req->abort_code = 0x00000000; fp@1035: req->jiffies_start = jiffies; fp@854: } fp@854: fp@854: /*****************************************************************************/ fp@858: fp@863: /** \cond */ fp@863: fp@2434: EXPORT_SYMBOL(ecrt_sdo_request_index); fp@858: EXPORT_SYMBOL(ecrt_sdo_request_timeout); fp@858: EXPORT_SYMBOL(ecrt_sdo_request_data); fp@869: EXPORT_SYMBOL(ecrt_sdo_request_data_size); fp@858: EXPORT_SYMBOL(ecrt_sdo_request_state); fp@858: EXPORT_SYMBOL(ecrt_sdo_request_read); fp@858: EXPORT_SYMBOL(ecrt_sdo_request_write); fp@859: fp@863: /** \endcond */ fp@863: fp@863: /*****************************************************************************/