fp@409: /****************************************************************************** fp@409: * fp@409: * $Id$ fp@409: * fp@409: * Copyright (C) 2006 Florian Pose, Ingenieurgemeinschaft IgH fp@409: * fp@409: * This file is part of the IgH EtherCAT Master. fp@409: * fp@409: * The IgH EtherCAT Master is free software; you can redistribute it fp@409: * and/or modify it under the terms of the GNU General Public License fp@409: * as published by the Free Software Foundation; either version 2 of the fp@409: * License, or (at your option) any later version. fp@409: * fp@409: * The IgH EtherCAT Master is distributed in the hope that it will be fp@409: * useful, but WITHOUT ANY WARRANTY; without even the implied warranty of fp@409: * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the fp@409: * GNU General Public License for more details. fp@409: * fp@409: * You should have received a copy of the GNU General Public License fp@409: * along 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@409: * The right to use EtherCAT Technology is granted and comes free of fp@409: * charge under condition of compatibility of product made by fp@409: * Licensee. People intending to distribute/sell products based on the fp@409: * code, have to sign an agreement to guarantee that products using fp@409: * software based on IgH EtherCAT master stay compatible with the actual fp@409: * EtherCAT specification (which are released themselves as an open fp@409: * standard) as the (only) precondition to have the right to use EtherCAT fp@409: * Technology, IP and trade marks. fp@409: * fp@409: *****************************************************************************/ fp@409: fp@862: /** \file fp@862: * Canopen-over-EtherCAT Sdo request functions. fp@862: */ fp@409: fp@409: /*****************************************************************************/ fp@409: fp@409: #include fp@881: #include fp@409: fp@847: #include "sdo_request.h" fp@409: fp@409: /*****************************************************************************/ fp@409: fp@1053: /** Default timeout in ms to wait for Sdo transfer responses. fp@1053: */ fp@1053: #define EC_SDO_REQUEST_RESPONSE_TIMEOUT 3000 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@847: /** Sdo request constructor. fp@758: */ fp@847: void ec_sdo_request_init( fp@852: ec_sdo_request_t *req /**< Sdo request. */ fp@831: ) fp@831: { 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@1037: req->abort_code = 0x00000000; fp@430: } fp@430: fp@430: /*****************************************************************************/ fp@430: fp@847: /** Sdo request destructor. fp@847: */ fp@847: void ec_sdo_request_clear( fp@847: 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@852: /** Sdo request destructor. fp@852: */ fp@852: void ec_sdo_request_clear_data( fp@852: 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: /** Set the Sdo address. fp@852: */ fp@854: void ec_sdo_request_address( fp@852: ec_sdo_request_t *req, /**< Sdo request. */ fp@852: uint16_t index, /**< Sdo index. */ fp@852: uint8_t subindex /**< Sdo subindex. */ fp@852: ) fp@852: { fp@852: req->index = index; fp@852: req->subindex = subindex; fp@852: } fp@852: fp@852: /*****************************************************************************/ fp@854: 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@854: 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@854: EC_ERR("Failed to allocate %u 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@854: /** Copies Sdo data from an external source. fp@854: * fp@854: * If the \a mem_size is to small, new memory is allocated. fp@854: */ fp@854: int ec_sdo_request_copy_data( fp@854: 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@854: if (ec_sdo_request_alloc(req, size)) fp@854: return -1; 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@880: 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@858: * Realtime interface. fp@858: ****************************************************************************/ fp@858: 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@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@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@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: /*****************************************************************************/