/**************************************************************************** * net/devif/devif_filesend.c * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. The * ASF licenses this file to you 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. * ****************************************************************************/ /**************************************************************************** * Included Files ****************************************************************************/ #include #include #include #include #include #include #include #include #ifdef CONFIG_MM_IOB /**************************************************************************** * Public Functions ****************************************************************************/ /**************************************************************************** * Name: devif_file_send * * Description: * Called from socket logic in response to a xmit or poll request from the * the network interface driver. * * This is identical to calling devif_file_send() except that the data is * in a available file handle. * * Assumptions: * Called with the network locked. * ****************************************************************************/ int devif_file_send(FAR struct net_driver_s *dev, FAR struct file *file, unsigned int len, unsigned int offset, unsigned int target_offset) { FAR struct iob_s *iob; unsigned int copyin; unsigned int remain; int ret; if (dev == NULL) { ret = -ENODEV; goto errout; } if (len == 0) { ret = -EINVAL; goto errout; } #ifndef CONFIG_NET_IPFRAG if (len > NETDEV_PKTSIZE(dev) - NET_LL_HDRLEN(dev) - target_offset) { ret = -EMSGSIZE; goto errout; } #endif /* Append the send buffer after device buffer */ if (len > iob_navail(false) * CONFIG_IOB_BUFSIZE || netdev_iob_prepare(dev, false, 0) != OK) { ret = -ENOMEM; goto errout; } iob_update_pktlen(dev->d_iob, target_offset, false); ret = file_seek(file, offset, SEEK_SET); if (ret < 0) { goto errout; } iob = dev->d_iob; remain = len; while (remain > 0) { if (iob->io_len + iob->io_offset == CONFIG_IOB_BUFSIZE) { if (iob->io_flink == NULL) { iob->io_flink = iob_tryalloc(false); if (iob->io_flink == NULL) { ret = -ENOMEM; goto errout; } } iob = iob->io_flink; } copyin = CONFIG_IOB_BUFSIZE - (iob->io_len + iob->io_offset); if (copyin > remain) { copyin = remain; } if (copyin > 0) { ret = file_read(file, iob->io_data + (iob->io_len + iob->io_offset), copyin); if (ret < 0) { goto errout; } iob->io_len += ret; remain -= ret; } } iob_update_pktlen(dev->d_iob, target_offset + len, false); dev->d_sndlen = len; return len; errout: if (ret < 0) { if (dev != NULL) { netdev_iob_release(dev); } nerr("ERROR: devif_iob_send error: %d\n", ret); } return ret; } #endif /* CONFIG_MM_IOB */