2020-03-04 18:05:57 +03:00
/* ------------------------------------------------------------------------ */
2023-01-09 14:23:19 +03:00
/* Copyright 2002-2023, OpenNebula Project, OpenNebula Systems */
2020-03-04 18:05:57 +03:00
/* */
/* 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. */
/* ------------------------------------------------------------------------ */
# ifndef HOST_SHARE_PCI_H_
# define HOST_SHARE_PCI_H_
# include "ObjectXML.h"
# include "Template.h"
# include <time.h>
# include <set>
# include <map>
/* -------------------------------------------------------------------------- */
/* -------------------------------------------------------------------------- */
/**
* This class represents a PCI DEVICE list for the host . The list is in the
* form :
* < PCI >
* < DOMAIN > PCI address domain
* < BUS > PCI address bus
* < SLOT > PCI address slot
* < FUNCTION > PCI address function
* < ADDRESS > PCI address , bus , slot and function
* < VENDOR > ID of PCI device vendor
* < DEVICE > ID of PCI device
* < CLASS > ID of PCI device class
* < VMID > ID using this device , - 1 if free
*
* The monitor probe may report additional information such as VENDOR_NAME ,
* DEVICE_NAME , CLASS_NAME . . .
*/
class HostSharePCI : public Template
{
public :
HostSharePCI ( ) : Template ( false , ' = ' , " PCI_DEVICES " ) { } ;
HostSharePCI ( const HostSharePCI & src ) ;
virtual ~ HostSharePCI ( )
{
clear ( ) ;
} ;
/**
* Builds the devices list from its XML representation . This function
* is used when importing it from the DB .
* @ param node xmlNode for the template
* @ return 0 on success
*/
int from_xml_node ( const xmlNodePtr node ) ;
/**
* Test whether this PCI device set has the requested devices available .
* @ param devs list of requested devices by the VM .
* @ return true if all the devices are available .
*/
2020-07-02 23:42:10 +03:00
bool test ( const std : : vector < VectorAttribute * > & devs ) const ;
2020-03-04 18:05:57 +03:00
/**
* Assign the requested devices to the given VM . The assigned devices will
* be labeled with the VM and the PCI attribute of the VM extended with
* the address of the assigned devices .
* @ param devs list of requested PCI devices , will include address of
* assigned devices .
* @ param vmid of the VM
2022-10-04 21:16:09 +03:00
*
2022-08-11 02:46:57 +03:00
* @ return true if the devices where added
*
* NOTE THIS FUNCTION DOES NOT PERFORM ANY ROLLBACK
2020-03-04 18:05:57 +03:00
*/
2022-08-11 02:46:57 +03:00
bool add ( std : : vector < VectorAttribute * > & devs , int vmid ) ;
2020-03-04 18:05:57 +03:00
/**
* Remove the VM assignment from the PCI device list
*/
2021-02-25 13:45:13 +03:00
void del ( const std : : vector < VectorAttribute * > & devs , int vmid ) ;
/**
2022-08-11 02:46:57 +03:00
* Revert the VM assignment from the PCI device list . It copies
* back the attributes from the previous PCI device
2021-02-25 13:45:13 +03:00
*/
void revert ( std : : vector < VectorAttribute * > & devs ) ;
2020-03-04 18:05:57 +03:00
/**
* Updates the PCI list with monitor data , it will create or
* remove PCIDevices as needed .
*/
void set_monitorization ( Template & ht ) ;
void clear ( ) override ;
/**
* Prints the PCI device list to an output stream . This function is used
* for logging purposes and * not * for generating DB content .
*/
2020-07-02 23:42:10 +03:00
friend std : : ostream & operator < < ( std : : ostream & o , const HostSharePCI & p ) ;
2020-03-04 18:05:57 +03:00
HostSharePCI & operator = ( const HostSharePCI & other ) ;
HostSharePCI & operator = ( HostSharePCI & & other ) noexcept ;
/**
* Gets a 4 hex digits value from attribute
* @ param name of the attribute
2022-10-04 21:16:09 +03:00
* @ param pci_device VectorAttribute representing the device
2020-03-04 18:05:57 +03:00
* @ return the 0 if not found , - 1 syntax error , > 0 valid hex value
*/
static int get_pci_value ( const char * name ,
const VectorAttribute * pci_device ,
unsigned int & value ) ;
/**
* Sets the PCI device address in the Virtual Machine as follows ;
* - VM_DOMAIN : 0x0000
* - VM_BUS : dbus or VM_BUS in PCI attribute
* - VM_SLOT : PCI_ID + 1
* - VM_FUNCTION : 0
* - VM_ADDRESS : BUS : SLOT .0
2022-08-11 02:46:57 +03:00
*
* Cleans internal attributes :
* - NUMA_NODE
* - UUID
* - BUS , SLOT , FUNCITION
* - ADDRESS , PREV_ADDRESS
2020-03-04 18:05:57 +03:00
* @ param pci_device to set the address in
* @ param default_bus if not set in PCI attribute ( PCI_PASSTHROUGH_BUS
* in oned . conf )
* @ return - 1 if wrong bus 0 on success
*/
2022-08-11 02:46:57 +03:00
static int set_pci_address ( VectorAttribute * pci_device , const std : : string & dbus ,
bool clean ) ;
2020-03-04 18:05:57 +03:00
private :
/**
* Internal structure to represent PCI devices for fast look up and
* update
*/
struct PCIDevice
{
PCIDevice ( VectorAttribute * _attrs ) ;
PCIDevice ( const PCIDevice & src ) ;
~ PCIDevice ( ) { } ;
unsigned int vendor_id ;
unsigned int device_id ;
unsigned int class_id ;
int vmid ;
2020-07-02 23:42:10 +03:00
std : : string address ;
2020-03-04 18:05:57 +03:00
VectorAttribute * attrs ;
} ;
2020-07-02 23:42:10 +03:00
std : : map < std : : string , PCIDevice * > pci_devices ;
2022-08-11 02:46:57 +03:00
/**
* Sets the internal class structures from the template
*/
void init ( ) ;
/**
* Test if there is a suitable PCI device for the VM request . The test
* is done using the VENDOR / DEVICE / CLASS attributes
*
* @ param dev VM attribute that represents the decive request
* @ param addrs PCI addresses that should be considered in use
*
* @ return true if the device can be allocated to this host
*/
bool test_by_name ( const VectorAttribute * dev , std : : set < std : : string > & addrs ) const ;
/**
* Test if there is a suitable PCI device for the VM request . The test
* is done using the a specific address
*
* @ param device VM attribute that represents the decive request
* @ param addr the requested address
*
* @ return PCI_ID of the tested device or - 1 if no PCI found
*/
bool test_by_addr ( const VectorAttribute * dev , const std : : string & addr ) const ;
/**
* Allocates the given VM device using the VENDOR / DEVICE / CLASS attributes
* @ param device VM attribute that represents the decive request
* @ param vmid of the VM
*/
bool add_by_name ( VectorAttribute * device , int vmid ) ;
/**
* Allocates the given VM device using the SHORT_ADDRESS attribute
* @ param device VM attribute that represents the decive request
* @ param vmid of the VM
*
* @ return pci_id of the allocated device or - 1 if not allocated
*/
bool add_by_addr ( VectorAttribute * device , const std : : string & addr , int vmid ) ;
/**
* Adds PCI attributes of the selected PCI to the VM PCI device
*
* @ param device VM attribute
* @ param pci Host device
* @ param sp if true set the " PREVIOUS_ADDRESS " attribute
*/
void pci_attribute ( VectorAttribute * device , PCIDevice * pci , bool sp ) ;
2020-03-04 18:05:57 +03:00
} ;
# endif /*HOST_SHARE_PCI_H_*/