Files
harbour-core/extras/hbusb/core.c
Viktor Szakats 9f16c2bf8e 2017-08-13 18:27 UTC Viktor Szakats (vszakats users.noreply.github.com)
* *
    * update copyright headers with new FSF postal address
    * COPYING.txt -> LICENSE.txt (rest of repo to be synced)
2017-08-13 18:38:59 +00:00

371 lines
10 KiB
C

/*
* Copyright (C) 2012 Doug (dougf at people dot net dot au)
*
* This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation; either version 2, or (at your option)
* any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program; see the file LICENSE.txt. If not, write to
* the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
* Boston, MA 02110-1301 USA (or visit https://www.gnu.org/licenses/).
*
* As a special exception, the Harbour Project gives permission for
* additional uses of the text contained in its release of Harbour.
*
* The exception is that, if you link the Harbour libraries with other
* files to produce an executable, this does not by itself cause the
* resulting executable to be covered by the GNU General Public License.
* Your use of that executable is in no way restricted on account of
* linking the Harbour library code into it.
*
* This exception does not however invalidate any other reasons why
* the executable file might be covered by the GNU General Public License.
*
* This exception applies only to the code released by the Harbour
* Project under the name Harbour. If you copy code from other
* Harbour Project or Free Software Foundation releases into a copy of
* Harbour, as the General Public License permits, the exception does
* not apply to the code that you add in this way. To avoid misleading
* anyone as to the status of such modified files, you must delete
* this exception notice from them.
*
* If you write modifications of your own for Harbour, it is your choice
* whether to permit this exception to apply to your modifications.
* If you do not wish that, delete this exception notice.
*
*/
/* TODO: change raw pointers to GC collected ones? */
#include "hbapi.h"
#include "libusb.h"
/* ------------ library initialisation and deinitialisation -------------------- */
/* Initialises libusb.
Must be called before calling any other libusb functions.
Getting a context is optional. */
HB_FUNC( LIBUSB_INIT )
{
int success;
if( HB_ISBYREF( 1 ) )
{
libusb_context * context;
success = libusb_init( &context );
hb_storptr( success == 0 ? context : NULL, 1 );
}
else
success = libusb_init( NULL );
hb_retni( success );
}
/* Deinitialise libusb. */
HB_FUNC( LIBUSB_EXIT )
{
libusb_exit( ( libusb_context * ) hb_parptr( 1 ) );
}
/* Sets the message verbosity.
Refer to hbusb.ch for options */
HB_FUNC( LIBUSB_SET_DEBUG )
{
libusb_set_debug( ( libusb_context * ) hb_parptr( 1 ), hb_parni( 2 ) );
}
/* ------------------- device handling and enumeration ------------------------- */
/* Returns a list of USB devcices attached to your system. */
HB_FUNC( LIBUSB_GET_DEVICE_LIST )
{
libusb_device ** devicelist;
ssize_t count;
count = libusb_get_device_list( ( libusb_context * ) hb_parptr( 1 ), &devicelist );
hb_storptr( devicelist, 2 );
hb_retns( ( HB_SIZE ) count );
}
/* Frees the list returned by LIBUSB_GetDeviceList(). */
HB_FUNC( LIBUSB_FREE_DEVICE_LIST )
{
libusb_free_device_list( ( libusb_device ** ) hb_parptr( 1 ), hb_parni( 2 ) );
}
/* Gets the number of the bus that a device is attached to. */
HB_FUNC( LIBUSB_GET_BUS_NUMBER )
{
uint8_t busnumber;
libusb_device ** devicelist;
devicelist = ( libusb_device ** ) hb_parptr( 1 );
busnumber = libusb_get_bus_number( devicelist[ hb_parni( 2 ) ] );
hb_retni( busnumber );
}
/* Gets the address on the bus that a device is attached to. */
HB_FUNC( LIBUSB_GET_DEVICE_ADDRESS )
{
uint8_t deviceaddress;
libusb_device ** devicelist;
devicelist = ( libusb_device ** ) hb_parptr( 1 );
deviceaddress = libusb_get_device_address( devicelist[ hb_parni( 2 ) ] );
hb_retni( deviceaddress );
}
/* Gets the negotiated connection speed for a device.
Refer to hbusb.ch for speeds. */
#if 0
HB_FUNC( LIBUSB_GET_DEVICE_SPEED )
{
int devicespeed;
libusb_device ** devicelist;
devicelist = ( libusb_device ** ) hb_parptr( 1 );
devicespeed = libusb_get_device_speed( devicelist[ hb_parni( 2 ) ] );
hb_retni( devicespeed );
}
#endif
/* Gets the maximum packet size for a particular endpoint in the active device configuration. */
HB_FUNC( LIBUSB_GET_MAX_PACKET_SIZE )
{
int maxpacketsize;
libusb_device ** devicelist;
devicelist = ( libusb_device ** ) hb_parptr( 1 );
maxpacketsize = libusb_get_max_packet_size( devicelist[ hb_parni( 2 ) ], ( unsigned char ) hb_parni( 3 ) );
hb_retni( maxpacketsize );
}
/* Increments the reference count of a device. */
HB_FUNC( LIBUSB_REF_DEVICE )
{
libusb_device * device;
libusb_device ** devicelist;
devicelist = ( libusb_device ** ) hb_parptr( 1 );
device = libusb_ref_device( devicelist[ hb_parni( 2 ) ] );
HB_SYMBOL_UNUSED( device );
}
/* Decrements the reference count of a device */
HB_FUNC( LIBUSB_UNREF_DEVICE )
{
libusb_device ** devicelist;
devicelist = ( libusb_device ** ) hb_parptr( 1 );
libusb_unref_device( devicelist[ hb_parni( 2 ) ] );
}
/* Open a device and obtain a device handle */
HB_FUNC( LIBUSB_OPEN )
{
int success;
libusb_device_handle * handle;
libusb_device ** devicelist;
devicelist = ( libusb_device ** ) hb_parptr( 1 );
success = libusb_open( devicelist[ hb_parni( 2 ) ], &handle );
hb_storptr( success == 0 ? handle : NULL, 3 );
hb_retni( success );
}
/* Find a device with known Vendor IID and Product ID. */
HB_FUNC( LIBUSB_OPEN_DEVICE_WITH_VID_PID )
{
libusb_device_handle * handle;
handle = libusb_open_device_with_vid_pid( ( libusb_context * ) hb_parptr( 1 ), ( uint16_t ) hb_parni( 2 ), ( uint16_t ) hb_parni( 3 ) );
hb_retptr( handle );
}
/* Close a device handle. */
HB_FUNC( LIBUSB_CLOSE )
{
libusb_close( ( libusb_device_handle * ) hb_parptr( 1 ) );
}
/* Get the underlying device for a handle. */
HB_FUNC( LIBUSB_GET_DEVICE )
{
libusb_device * device;
device = libusb_get_device( ( libusb_device_handle * ) hb_parptr( 1 ) );
hb_retptr( device );
}
/* Get the configuration value of the currently active configuration. */
HB_FUNC( LIBUSB_GET_CONFIGURATION )
{
int configuration;
int success;
configuration = 0;
success = libusb_get_configuration( ( libusb_device_handle * ) hb_parptr( 1 ), &configuration );
hb_storni( success == 0 ? configuration : 0, 2 );
hb_retni( success );
}
/* Set the active configuration for a device. */
HB_FUNC( LIBUSB_SET_CONFIGURATION )
{
int success;
success = libusb_set_configuration( ( libusb_device_handle * ) hb_parptr( 1 ), hb_parni( 2 ) );
hb_retni( success );
}
/* Claim an interface on a given device handle.
Required before you can perform I/O on any of its endpoints. */
HB_FUNC( LIBUSB_CLAIM_INTERFACE )
{
int success;
success = libusb_claim_interface( ( libusb_device_handle * ) hb_parptr( 1 ), hb_parni( 2 ) );
hb_retni( success );
}
/* Release a previously claimed interface. */
HB_FUNC( LIBUSB_RELEASE_INTERFACE )
{
int success;
success = libusb_release_interface( ( libusb_device_handle * ) hb_parptr( 1 ), hb_parni( 2 ) );
hb_retni( success );
}
/* Activate an alternate setting for an interface. */
HB_FUNC( LIBUSB_SET_INTERFACE_ALT_SETTING )
{
int success;
success = libusb_set_interface_alt_setting( ( libusb_device_handle * ) hb_parptr( 1 ), hb_parni( 2 ), hb_parni( 3 ) );
hb_retni( success );
}
/* Clear the halt / stall condition for an endpoint */
HB_FUNC( LIBUSB_CLEAR_HALT )
{
int success;
success = libusb_clear_halt( ( libusb_device_handle * ) hb_parptr( 1 ), ( unsigned char ) hb_parni( 2 ) );
hb_retni( success );
}
/* Perform a USB port reset to reinitialise a device. */
HB_FUNC( LIBUSB_RESET_DEVICE )
{
int success;
success = libusb_reset_device( ( libusb_device_handle * ) hb_parptr( 1 ) );
hb_retni( success );
}
/* Determine if a kernel driver is active on an interfacc. */
HB_FUNC( LIBUSB_KERNEL_DRIVER_ACTIVE )
{
int isactive;
isactive = libusb_kernel_driver_active( ( libusb_device_handle * ) hb_parptr( 1 ), hb_parni( 2 ) );
hb_retni( isactive );
}
/* Detach a kernel driver from an interface. */
HB_FUNC( LIBUSB_DETACH_KERNEL_DRIVER )
{
hb_retni( libusb_detach_kernel_driver( ( libusb_device_handle * ) hb_parptr( 1 ), hb_parni( 2 ) ) );
}
/* Reattach a kernel driver which was previously detached. */
HB_FUNC( LIBUSB_ATTACH_KERNEL_DRIVER )
{
int success;
success = libusb_attach_kernel_driver( ( libusb_device_handle * ) hb_parptr( 1 ), hb_parni( 2 ) );
hb_retni( success );
}
/* ------------------------------ USB descriptors ------------------------------ */
/* Get the USB descriptor for a given device. */
HB_FUNC( LIBUSB_GET_DEVICE_DESCRIPTOR )
{
struct libusb_device_descriptor desc;
libusb_device ** devicelist;
int success;
devicelist = ( libusb_device ** ) hb_parptr( 1 );
success = libusb_get_device_descriptor( devicelist[ hb_parni( 2 ) ], &desc );
hb_storptr( &desc, 3 );
hb_storni( desc.idVendor, 4 );
hb_storni( desc.idProduct, 5 );
hb_storni( ( int ) desc.bNumConfigurations, 6 );
hb_retni( success );
}
/* --------------------------- synchronous device I/O -------------------------- */
/* Perform a USB bulk transfer. */
HB_FUNC( LIBUSB_BULK_TRANSFER )
{
int success;
unsigned char data[ 512 ];
int transferred;
success = libusb_bulk_transfer( ( libusb_device_handle * ) hb_parptr( 1 ), ( unsigned char ) hb_parni( 2 ), data, sizeof( data ), &transferred, hb_parni( 3 ) );
hb_storclen( ( char * ) data, ( HB_ISIZ ) transferred, 4 );
hb_storni( transferred, 5 );
hb_retni( success );
}
/* Perform a USB interrupt transfer. */
HB_FUNC( LIBUSB_INTERRUPT_TRANSFER )
{
int success;
unsigned char data[ 512 ];
int transferred;
success = libusb_interrupt_transfer( ( libusb_device_handle * ) hb_parptr( 1 ), ( unsigned char ) hb_parni( 2 ), data, sizeof( data ), &transferred, hb_parni( 3 ) );
hb_storclen( ( char * ) data, ( HB_ISIZ ) transferred, 4 );
hb_storni( transferred, 5 );
hb_retni( success );
}