mirror of
https://github.com/NVIDIA/open-gpu-kernel-modules.git
synced 2025-01-19 03:52:11 +01:00
218 lines
7.7 KiB
C
218 lines
7.7 KiB
C
|
/*
|
||
|
* SPDX-FileCopyrightText: Copyright (c) 2011 NVIDIA CORPORATION & AFFILIATES. All rights reserved.
|
||
|
* SPDX-License-Identifier: MIT
|
||
|
*
|
||
|
* Permission is hereby granted, free of charge, to any person obtaining a
|
||
|
* copy of this software and associated documentation files (the "Software"),
|
||
|
* to deal in the Software without restriction, including without limitation
|
||
|
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
||
|
* and/or sell copies of the Software, and to permit persons to whom the
|
||
|
* Software is furnished to do so, subject to the following conditions:
|
||
|
*
|
||
|
* The above copyright notice and this permission notice shall be included in
|
||
|
* all copies or substantial portions of the Software.
|
||
|
*
|
||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
||
|
* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||
|
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
||
|
* DEALINGS IN THE SOFTWARE.
|
||
|
*/
|
||
|
|
||
|
#define __NO_VERSION__
|
||
|
|
||
|
#include "os-interface.h"
|
||
|
#include "nv-linux.h"
|
||
|
|
||
|
#if defined(CONFIG_CRAY_XT)
|
||
|
enum {
|
||
|
NV_FORMAT_STATE_ORDINARY,
|
||
|
NV_FORMAT_STATE_INTRODUCTION,
|
||
|
NV_FORMAT_STATE_FLAGS,
|
||
|
NV_FORMAT_STATE_FIELD_WIDTH,
|
||
|
NV_FORMAT_STATE_PRECISION,
|
||
|
NV_FORMAT_STATE_LENGTH_MODIFIER,
|
||
|
NV_FORMAT_STATE_CONVERSION_SPECIFIER
|
||
|
};
|
||
|
|
||
|
enum {
|
||
|
NV_LENGTH_MODIFIER_NONE,
|
||
|
NV_LENGTH_MODIFIER_CHAR,
|
||
|
NV_LENGTH_MODIFIER_SHORT_INT,
|
||
|
NV_LENGTH_MODIFIER_LONG_INT,
|
||
|
NV_LENGTH_MODIFIER_LONG_LONG_INT
|
||
|
};
|
||
|
|
||
|
#define NV_IS_FLAG(c) \
|
||
|
((c) == '#' || (c) == '0' || (c) == '-' || (c) == ' ' || (c) == '+')
|
||
|
#define NV_IS_LENGTH_MODIFIER(c) \
|
||
|
((c) == 'h' || (c) == 'l' || (c) == 'L' || (c) == 'q' || (c) == 'j' || \
|
||
|
(c) == 'z' || (c) == 't')
|
||
|
#define NV_IS_CONVERSION_SPECIFIER(c) \
|
||
|
((c) == 'd' || (c) == 'i' || (c) == 'o' || (c) == 'u' || (c) == 'x' || \
|
||
|
(c) == 'X' || (c) == 'e' || (c) == 'E' || (c) == 'f' || (c) == 'F' || \
|
||
|
(c) == 'g' || (c) == 'G' || (c) == 'a' || (c) == 'A' || (c) == 'c' || \
|
||
|
(c) == 's' || (c) == 'p')
|
||
|
|
||
|
#define NV_MAX_NUM_INFO_MMRS 6
|
||
|
|
||
|
NV_STATUS nvos_forward_error_to_cray(
|
||
|
struct pci_dev *dev,
|
||
|
NvU32 error_number,
|
||
|
const char *format,
|
||
|
va_list ap
|
||
|
)
|
||
|
{
|
||
|
NvU32 num_info_mmrs;
|
||
|
NvU64 x = 0, info_mmrs[NV_MAX_NUM_INFO_MMRS];
|
||
|
int state = NV_FORMAT_STATE_ORDINARY;
|
||
|
int modifier = NV_LENGTH_MODIFIER_NONE;
|
||
|
NvU32 i, n = 0, m = 0;
|
||
|
|
||
|
memset(info_mmrs, 0, sizeof(info_mmrs));
|
||
|
while (*format != '\0')
|
||
|
{
|
||
|
switch (state)
|
||
|
{
|
||
|
case NV_FORMAT_STATE_ORDINARY:
|
||
|
if (*format == '%')
|
||
|
state = NV_FORMAT_STATE_INTRODUCTION;
|
||
|
break;
|
||
|
case NV_FORMAT_STATE_INTRODUCTION:
|
||
|
if (*format == '%')
|
||
|
{
|
||
|
state = NV_FORMAT_STATE_ORDINARY;
|
||
|
break;
|
||
|
}
|
||
|
case NV_FORMAT_STATE_FLAGS:
|
||
|
if (NV_IS_FLAG(*format))
|
||
|
{
|
||
|
state = NV_FORMAT_STATE_FLAGS;
|
||
|
break;
|
||
|
}
|
||
|
else if (*format == '*')
|
||
|
{
|
||
|
state = NV_FORMAT_STATE_FIELD_WIDTH;
|
||
|
break;
|
||
|
}
|
||
|
case NV_FORMAT_STATE_FIELD_WIDTH:
|
||
|
if ((*format >= '0') && (*format <= '9'))
|
||
|
{
|
||
|
state = NV_FORMAT_STATE_FIELD_WIDTH;
|
||
|
break;
|
||
|
}
|
||
|
else if (*format == '.')
|
||
|
{
|
||
|
state = NV_FORMAT_STATE_PRECISION;
|
||
|
break;
|
||
|
}
|
||
|
case NV_FORMAT_STATE_PRECISION:
|
||
|
if ((*format >= '0') && (*format <= '9'))
|
||
|
{
|
||
|
state = NV_FORMAT_STATE_PRECISION;
|
||
|
break;
|
||
|
}
|
||
|
else if (NV_IS_LENGTH_MODIFIER(*format))
|
||
|
{
|
||
|
state = NV_FORMAT_STATE_LENGTH_MODIFIER;
|
||
|
break;
|
||
|
}
|
||
|
else if (NV_IS_CONVERSION_SPECIFIER(*format))
|
||
|
{
|
||
|
state = NV_FORMAT_STATE_CONVERSION_SPECIFIER;
|
||
|
break;
|
||
|
}
|
||
|
case NV_FORMAT_STATE_LENGTH_MODIFIER:
|
||
|
if ((*format == 'h') || (*format == 'l'))
|
||
|
{
|
||
|
state = NV_FORMAT_STATE_LENGTH_MODIFIER;
|
||
|
break;
|
||
|
}
|
||
|
else if (NV_IS_CONVERSION_SPECIFIER(*format))
|
||
|
{
|
||
|
state = NV_FORMAT_STATE_CONVERSION_SPECIFIER;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
switch (state)
|
||
|
{
|
||
|
case NV_FORMAT_STATE_INTRODUCTION:
|
||
|
modifier = NV_LENGTH_MODIFIER_NONE;
|
||
|
break;
|
||
|
case NV_FORMAT_STATE_LENGTH_MODIFIER:
|
||
|
switch (*format)
|
||
|
{
|
||
|
case 'h':
|
||
|
modifier = (modifier == NV_LENGTH_MODIFIER_NONE)
|
||
|
? NV_LENGTH_MODIFIER_SHORT_INT
|
||
|
: NV_LENGTH_MODIFIER_CHAR;
|
||
|
break;
|
||
|
case 'l':
|
||
|
modifier = (modifier == NV_LENGTH_MODIFIER_NONE)
|
||
|
? NV_LENGTH_MODIFIER_LONG_INT
|
||
|
: NV_LENGTH_MODIFIER_LONG_LONG_INT;
|
||
|
break;
|
||
|
case 'q':
|
||
|
modifier = NV_LENGTH_MODIFIER_LONG_LONG_INT;
|
||
|
default:
|
||
|
return NV_ERR_INVALID_ARGUMENT;
|
||
|
}
|
||
|
break;
|
||
|
case NV_FORMAT_STATE_CONVERSION_SPECIFIER:
|
||
|
switch (*format)
|
||
|
{
|
||
|
case 'c':
|
||
|
case 'd':
|
||
|
case 'i':
|
||
|
x = (unsigned int)va_arg(ap, int);
|
||
|
break;
|
||
|
case 'o':
|
||
|
case 'u':
|
||
|
case 'x':
|
||
|
case 'X':
|
||
|
switch (modifier)
|
||
|
{
|
||
|
case NV_LENGTH_MODIFIER_LONG_LONG_INT:
|
||
|
x = va_arg(ap, unsigned long long int);
|
||
|
break;
|
||
|
case NV_LENGTH_MODIFIER_LONG_INT:
|
||
|
x = va_arg(ap, unsigned long int);
|
||
|
break;
|
||
|
case NV_LENGTH_MODIFIER_CHAR:
|
||
|
case NV_LENGTH_MODIFIER_SHORT_INT:
|
||
|
case NV_LENGTH_MODIFIER_NONE:
|
||
|
x = va_arg(ap, unsigned int);
|
||
|
break;
|
||
|
}
|
||
|
break;
|
||
|
default:
|
||
|
return NV_ERR_INVALID_ARGUMENT;
|
||
|
}
|
||
|
state = NV_FORMAT_STATE_ORDINARY;
|
||
|
for (i = 0; i < ((modifier == NV_LENGTH_MODIFIER_LONG_LONG_INT)
|
||
|
? 2 : 1); i++)
|
||
|
{
|
||
|
if (m == NV_MAX_NUM_INFO_MMRS)
|
||
|
return NV_ERR_INSUFFICIENT_RESOURCES;
|
||
|
info_mmrs[m] = ((info_mmrs[m] << 32) | (x & 0xffffffff));
|
||
|
x >>= 32;
|
||
|
if (++n == 2)
|
||
|
{
|
||
|
m++;
|
||
|
n = 0;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
format++;
|
||
|
}
|
||
|
|
||
|
num_info_mmrs = (m + (n != 0));
|
||
|
if (num_info_mmrs > 0)
|
||
|
cray_nvidia_report_error(dev, error_number, num_info_mmrs, info_mmrs);
|
||
|
|
||
|
return NV_OK;
|
||
|
}
|
||
|
#endif
|