1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22#include <core/device.h>
23#include <core/firmware.h>
24
25int
26nvkm_firmware_load_name(const struct nvkm_subdev *subdev, const char *base,
27 const char *name, int ver, const struct firmware **pfw)
28{
29 char path[64];
30 int ret;
31
32 snprintf(path, sizeof(path), "%s%s", base, name);
33 ret = nvkm_firmware_get(subdev, path, ver, pfw);
34 if (ret < 0)
35 return ret;
36
37 return 0;
38}
39
40int
41nvkm_firmware_load_blob(const struct nvkm_subdev *subdev, const char *base,
42 const char *name, int ver, struct nvkm_blob *blob)
43{
44 const struct firmware *fw;
45 int ret;
46
47 ret = nvkm_firmware_load_name(subdev, base, name, ver, &fw);
48 if (ret == 0) {
49 blob->data = kmemdup(fw->data, fw->size, GFP_KERNEL);
50 blob->size = fw->size;
51 nvkm_firmware_put(fw);
52 if (!blob->data)
53 return -ENOMEM;
54 }
55
56 return ret;
57}
58
59
60
61
62
63
64
65
66
67
68
69int
70nvkm_firmware_get(const struct nvkm_subdev *subdev, const char *fwname, int ver,
71 const struct firmware **fw)
72{
73 struct nvkm_device *device = subdev->device;
74 char f[64];
75 char cname[16];
76 int i;
77
78
79 strncpy(cname, device->chip->name, sizeof(cname));
80 cname[sizeof(cname) - 1] = '\0';
81 i = strlen(cname);
82 while (i) {
83 --i;
84 cname[i] = tolower(cname[i]);
85 }
86
87 if (ver != 0)
88 snprintf(f, sizeof(f), "nvidia/%s/%s-%d.bin", cname, fwname, ver);
89 else
90 snprintf(f, sizeof(f), "nvidia/%s/%s.bin", cname, fwname);
91
92 if (!firmware_request_nowarn(fw, f, device->dev)) {
93 nvkm_debug(subdev, "firmware \"%s\" loaded - %zu byte(s)\n",
94 f, (*fw)->size);
95 return 0;
96 }
97
98 nvkm_debug(subdev, "firmware \"%s\" unavailable\n", f);
99 return -ENOENT;
100}
101
102
103
104
105void
106nvkm_firmware_put(const struct firmware *fw)
107{
108 release_firmware(fw);
109}
110